18

Я ищу простую команду, которую можно использовать в Bash, чтобы найти абсолютный и канонизированный путь к файлу в OS X (аналогично `` readlink -f'` в Linux).

В следующем примере сеанса bash описывается [вымышленная] утилита с именем `` abspath'`, которая демонстрирует желаемое поведение:

$ 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

Как и в случае последнего вызова `` abspath'` в вышеприведенном примере, я бы предпочел, чтобы он не разрешал автоматически символические ссылки, но я не собираюсь быть слишком требовательным.

9 ответов9

16
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; }

Примеры:

abspath / => /

abspath /.DS_Store => /.DS_Store

abspath ~ => /Users/mschrag

cd /tmp; abspath . => /tmp

cd /; abspath .DS_Store => /.DS_Store
11

Я не думаю, что есть команда buildin, которая делает это. Джесси Уилсон написал сценарий bash для этого:

#!/bin/bash
cd -P -- "$(dirname -- "$1")" &&
printf '%s\n' "$(pwd -P)/$(basename -- "$1")"

Тем не менее, он не работает хорошо для путей непосредственно под / , таких как /etc (печать //etc), а также . и .. (печать /cwd/. в обоих случаях). Я попытался изменить это, но мой неэффективный bash-fu подвел меня.

Вот мое предложение:

#!/usr/bin/env python
import os.path
import sys

for arg in sys.argv[1:]:
    print os.path.abspath(arg)

Сохраните как /usr/bin/abspath или что-то в этом роде и сделайте его исполняемым. Образец вывода:

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

Если вы хотите разрешение линка, изменить print строку:

    print os.path.realpath(os.path.abspath(arg))

чтобы получить это:

Servus08:~ danielbeck$ abspath . /tmp Documents
/Users/danielbeck
/private/tmp
/Users/danielbeck/Documents
8

Одним из вариантов будет просто установить coreutils и использовать greadlink -f . Он разрешает символические ссылки и работает с /Foo/ или ~/foo.txt если они не существуют, но не с /Foo/foo.txt если /Foo/ не существует.

$ 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
$ 

Это не разрешает символические ссылки, и не работает с /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 выполняет расширение тильды. dirs +0 печатает только самый верхний каталог, если в стеке есть другие каталоги.

2

Я думаю, вы могли бы сделать это с помощью Python или Ruby.

$ ruby -e 'puts File.expand_path("~/somepath")'

или сделать команду с

#!/usr/bin/env ruby
puts File.expand_path(ARGV[0])
1

Если у вас установлен модуль File::Spec для perl, вы можете просто сделать это:

perl -MFile::Spec -e 'print File::Spec->rel2abs("../however/complex/../you/want/to.get"), "\n"'
0

Для скриптов bash/sh вы можете использовать эту рекурсивную функцию:

canonical_readlink ()
{
    cd `dirname $1`;
    __filename=`basename $1`;
    if [ -h "$__filename" ]; then
        canonical_readlink `readlink $__filename`;
    else
        echo "`pwd -P`";
    fi
}

answer=$(canonical_readlink $0)
0

Если установка coreutils не является опцией, следующие комбинации обрабатывают символические ссылки. и .. и работает с файлами и папками, как GNU realpath:

#!/usr/bin/env bash
realpath()
{
    if ! pushd $1 &> /dev/null; then 
        pushd ${1##*/} &> /dev/null
        echo $( pwd -P )/${1%/*}
    else
        pwd -P
    fi
    popd > /dev/null
}

Но он не поддерживает realpath -relative-to. Для этого потребуется https://stackoverflow.com/a/12498485/869951.

0

Установите следующую библиотеку для OSX:

brew install coreutils

greadlink -f file.txt
0

Учитывая, что ограничение - MacOS (OS X в то время), PHP доступен по умолчанию. Это вернет корневой каталог файла. Удалите dirname чтобы получить файл тоже.

export SOURCE_DIRECTORY="$(php -r 'echo dirname(realpath($argv[1]));' -- "${BASH_SOURCE[0]}")"

Всё ещё ищете ответ? Посмотрите другие вопросы с метками .