Estou procurando um comando simples que possa ser usado no Bash para encontrar o caminho absoluto e canônico para um arquivo em um OS X (semelhante a ``readlink -f'` no Linux).
O exemplo de sessão bash a seguir descreve um utilitário [fictício] chamado ``abspath'` que exibe o comportamento desejado:
$ pwd
/Users/guyfleegman
$ ls -lR
drwxr-xr-x 4 guyfleegman crew 136 Oct 30 02:09 foo
./foo:
-rw-r--r-- 1 guyfleegman crew 0 Oct 30 02:07 bar.txt
lrwxr-xr-x 1 guyfleegman crew 7 Oct 30 02:09 baz.txt -> bar.txt
$ abspath .
/Users/guyfleegman
$ abspath foo
/Users/guyfleegman/foo
$ abspath ./foo/bar.txt
/Users/guyfleegman/foo/bar.txt
$ abspath foo/baz.txt
/Users/guyfleegman/foo/baz.txt
Assim como na última invocação de ``abspath'` no exemplo acima, eu preferiria que ele não resolvesse links simbólicos automaticamente, mas não serei muito exigente aqui.
Responder1
function abspath() { pushd . > /dev/null; if [ -d "$1" ]; then cd "$1"; dirs -l +0; else cd "`dirname \"$1\"`"; cur_dir=`dirs -l +0`; if [ "$cur_dir" == "/" ]; then echo "$cur_dir`basename \"$1\"`"; else echo "$cur_dir/`basename \"$1\"`"; fi; fi; popd > /dev/null; }
Exemplos:
abspath / => /
abspath /.DS_Store => /.DS_Store
abspath ~ => /Users/mschrag
cd /tmp; abspath . => /tmp
cd /; abspath .DS_Store => /.DS_Store
Responder2
Não acho que exista um comando buildin que faça isso.Jessé Wilsonescreveu um script bash para isso:
#!/bin/bash
cd -P -- "$(dirname -- "$1")" &&
printf '%s\n' "$(pwd -P)/$(basename -- "$1")"
No entanto, não funciona bem para caminhos diretamente abaixo de /
, como /etc
(printing //etc
), bem como .
e ..
(printing /cwd/.
em ambos os casos). Tentei modificá-lo, mas meu bash-fu insuficiente falhou.
Aqui está minha sugestão:
#!/usr/bin/env python
import os.path
import sys
for arg in sys.argv[1:]:
print os.path.abspath(arg)
Salve como /usr/bin/abspath
ou algo parecido e torne-o executável. Exemplo de saída:
Servus08:~ danielbeck$ abspath .
/Users/danielbeck
Servus08:~ danielbeck$ abspath /tmp
/tmp
Servus08:~ danielbeck$ abspath Documents
/Users/danielbeck/Documents
Servus08:~ danielbeck$ abspath . /tmp Documents
/Users/danielbeck
/tmp
/Users/danielbeck/Documents
Se vocêfazerdeseja resolução de link simbólico, altere a print
linha assim:
print os.path.realpath(os.path.abspath(arg))
para conseguir esta:
Servus08:~ danielbeck$ abspath . /tmp Documents
/Users/danielbeck
/private/tmp
/Users/danielbeck/Documents
Responder3
Uma opção seria apenas instalar o coreutils e usar o greadlink -f
. Ele resolve links simbólicos e funciona com /Foo/
ou ~/foo.txt
se eles não existirem, mas não com /Foo/foo.txt
se /Foo/
não existir.
$ brew install coreutils
$ greadlink -f /etc
/private/etc
$ greadlink -f ~/Documents/
/Users/lauri/Documents
$ greadlink -f ..
/Users
$ greadlink -f //etc/..////
/private
$ greadlink -f /Foo
/Foo
$ greadlink -f /Foo/foo.txt
$
Isso não resolve links simbólicos e também não funciona /Foo/foo.txt
.
abspath() {
if [ -d "$1" ]; then
( cd "$1"; dirs -l +0 )
else
( cd "$(dirname "$1")"; d=$(dirs -l +0); echo "${d%/}/${1##*/}" )
fi
}
abspath /etc # /etc
abspath ~/Foo/foo.txt # doesn't work
abspath ~/Foo # works
abspath .
abspath ./
abspath ../
abspath ..
abspath /
abspath ~
abspath ~/
abspath ~/Documents
abspath /\"\ \'
abspath /etc/../etc/
abspath /private//etc/
abspath /private//
abspath //private # //private
abspath ./aa.txt
abspath aa.tar.gz
abspath .aa.txt
abspath /.DS_Store
abspath ~/Documents/Books/
dirs -l
executa a expansão do til. dirs +0
imprime apenas o diretório superior se houver outros diretórios na pilha.
Responder4
Se você tiver o módulo File::Spec instalado para perl, você pode fazer isso:
perl -MFile::Spec -e 'print File::Spec->rel2abs("../however/complex/../you/want/to.get"), "\n"'