Como posso encontrar o primeiro diretório ausente em um caminho longo?

14

Imagine que eu tenha um caminho que não existe:

$ ls /foo/bar/baz/hello/world
ls: cannot access /foo/bar/baz/hello/world: No such file or directory

Mas digamos que /foo/bar existe . Existe uma maneira rápida para eu determinar que baz é o ponto de ruptura no caminho?

Estou usando o Bash.

    
por kuzzooroo 04.02.2016 / 00:48

5 respostas

5

Dado um nome de caminho canônico, como o seu, isso funcionará:

set -f --; IFS=/
for p in $pathname
do    [ -e "$*/$p" ] || break
      set -- "$@" "$p"
done; printf %s\n "$*"

Isso imprime pelo último componente totalmente existente / acessível de $pathname e coloca cada um deles separadamente no array arg. O primeiro componente inexistente não é impresso, mas é salvo em $p .

Você pode abordá-lo de forma oposta:

until cd -- "$path" && cd -
do    case   $path  in
      (*[!/]/*)
              path="${path%/*}"
;;    (*)   ! break
      esac
done  2>/dev/null   && cd -

Isso retornará adequadamente ou reduzirá $path , conforme necessário. Ele se recusa a tentar uma alteração para / , mas se for bem-sucedida, imprimirá seu diretório de trabalho atual e o diretório para o qual ele será alterado para stdout. Seu $PWD atual também será colocado em $OLDPWD .

    
por 04.02.2016 / 01:22
11

Um dos meus utilitários favoritos é namei , parte de util-linux e, portanto, geralmente presente apenas no Linux:

$ namei /usr/share/foo/bar
f: /usr/share/foo/bar
 d /
 d usr
 d share
   foo - No such file or directory

Mas sua saída não é muito parseable. Então, se você quiser apenas apontar que algo está faltando, namei pode ser útil.

É útil para solucionar problemas gerais ao acessar um caminho, já que você pode fazer com que ele indique se um componente é um link ou um ponto de montagem, bem como suas permissões:

$ ln -sf /usr/foo/bar /tmp/
$ namei -lx /tmp/bar
f: /tmp/bar
Drwxr-xr-x root    root    /
Drwxrwxrwt root    root    tmp
lrwxrwxrwx muru    muru    bar -> /usr/foo/bar
Drwxr-xr-x root    root      /
drwxr-xr-x root    root      usr
                             foo - No such file or directory

O capital D indica um ponto de montagem.

    
por 04.02.2016 / 06:25
1

Algo parecido com isso (contabilizando nomes de caminho com espaços em branco incorporados):

#!/bin/sh
explain() {
    if [ -d "$1" ]
    then
        printf "\t%s: is a directory\n" "$1"
    elif [ -e "$1" ]
    then
        printf "\t%s: is not a directory\n" "$1"
    else
        printf "\t%s: does not exist\n" "$1"
    fi
}

for item in "$@"
do
    last=
    test="$item"
    printf "testing: '%s'\n" "$item"
    while [ ! -d "$test" ]
    do
        last="$test"
        test=$(dirname "$test")
        [ -z "$test" ] && break
    done
    if [ -n "$last" ]
    then
        explain "$test"
        explain "$last"
    else
        printf "\t%s: ok\n" "$item"
    fi
done
    
por 04.02.2016 / 01:12
1

Apenas uma solução alternativa para o bash, assumindo que o caminho é um caminho absoluto (começa com /):

#!/bin/bash

pathname="$1"

IFS='/' read -r -a p <<<"${pathname#/}"

pa=""    max="${#p[@]}"    i=0
while (( i<"$max" )); do
      pa="$pa/${p[i++]}"
      if     [[ ! -e $pa ]]; then
             printf 'failed at: \t"%s"\t"%s"\n' "${pa##*/}" "${pa}"
             break
      fi
done
$ ./script "/foo/ba r/baz/hello/world"
failed at:      "hello"        "/foo/ba r/baz/hello"
    
por 07.02.2016 / 04:27
0
(( dirct=$(echo ${dir}|tr "/" " "|wc -w)+1 ))
i=2
while [ ${i} -le ${dirct} ]
do
  sdir=$(echo ${dir}|cut -d/ -f1,${i})
  if [ ! -d ${sdir} ]
  then
    echo "Path is broken at ${sdir}"
  fi
  (( i++ ))
done

não é simples, mas você pode colocá-lo em um script, torná-lo executável e colocá-lo em algum lugar do seu caminho, se você for usá-lo com frequência.

Emptor de aviso: Se o nome do diretório em qualquer nível contiver um caractere space , isso NÃO funcionará.

    
por 04.02.2016 / 01:02