Use a expansão de parâmetro para remover a última etapa de um caminho. Faça isso em loop até encontrar um diretório existente:
while [[ $dir && ! -d $dir ]] ; do
dir=${dir%/*}
done
if [[ -d $dir ]] ; then
echo $dir exists.
fi
Eu tenho um código como este em um arquivo de script (bash):
if [ ! -d $dir ]; then
cd $dir/..
git clone http://...
fi
É claro que, como está escrito, isso não funciona, pois $dir
não existe. Existe uma boa maneira de remover o diretório de nível superior aqui?
chkdir() for d do ${d:+:} continue
( cd -- "$d" && d= ||
while cd -- "${d%"${d#*/}"}." && ! {
[ -n "${d##*/*}" ] && break
}; do d=${d#*/}; d=${d#"${d%%[!/]*}"}; done
pwd -P; printf "${d:+./%s\n}\n" "$d"
); done 2>/dev/null
Essa é uma pequena função que deve lidar com qualquer caminho que você jogue nela. Ele tentará andar na árvore se não puder simplesmente mudar diretamente para o diretório. Ele quebra o loop quando deveria. Ele lida com vários argumentos. Para cada um, ele imprimirá apenas todo o caminho canônico para o diretório ou imprimirá, na medida em que conseguisse obter então , o que restasse do caminho do seu argumento. Salta argumentos vazios.
Aqui está em ação:
$ chkdir /tmp/chrome/some/nonexistent/path .. ../test *
/tmp/chrome
./some/nonexistent/path
/home/mikeserv
/home/mikeserv/test
/home/mikeserv/test
./file1
/home/mikeserv/test
./file2
/home/mikeserv/test
./file3
Se eu remover o stderr
redirect no fim e ativar a depuração, você poderá ver como funciona.
+ chkdir .. ///tmp/.//////chrome/not/
+ : continue
+ cd -- ..
+ d=
+ pwd -P
/home/mikeserv
+ printf \n
+ : continue
+ cd -- ///tmp/.//////chrome/not/
dash: 2: cd: can't cd to ///tmp/.//////chrome/not/
+ cd -- /.
+ [ -n ]
+ d=//tmp/.//////chrome/not/
+ d=tmp/.//////chrome/not/
+ cd -- tmp/.
+ [ -n ]
+ d=.//////chrome/not/
+ d=.//////chrome/not/
+ cd -- ./.
+ [ -n ]
+ d=/////chrome/not/
+ d=chrome/not/
+ cd -- chrome/.
+ [ -n ]
+ d=not/
+ d=not/
+ cd -- not/.
dash: 3: cd: can't cd to not/.
+ pwd -P
/tmp/chrome
+ printf ./%s\n\n not/
./not/