Назад | Перейти на главную страницу

Как составить список цепочек символических ссылок?

Учитывая этот пример:

mkdir a
ln -s a b
ln -s b c
ln -s c d

Если я выполню:

ls -l d

Он покажет:

d -> c

Есть ли способ ls или любую другую команду Linux, чтобы показать d -> c -> b -> a вместо?

Просто используйте namei:

$ namei d
f: d
 l d -> c
   l c -> b
     l b -> a
       d a

readlink -e <link>

readlink [ОПЦИЯ] ... ФАЙЛ

  • -e, --canonicalize-existing
    canonicalize, рекурсивно следуя каждой символической ссылке в каждом компоненте заданного имени, все компоненты должны существовать
$ mkdir testlink
$ cd testlink
pjb@pjb-desktop:~/testlink$ ln -s c b
pjb@pjb-desktop:~/testlink$ ln -s b a
pjb@pjb-desktop:~/testlink$ ls -l 
total 0
lrwxrwxrwx 1 pjb pjb 1 2010-02-23 08:48 a -> b
lrwxrwxrwx 1 pjb pjb 1 2010-02-23 08:48 b -> c
pjb@pjb-desktop:~/testlink$ echo foo > c
pjb@pjb-desktop:~/testlink$ cat a
foo
pjb@pjb-desktop:~/testlink$ readlink -e a
/home/pjb/testlink/c

примечание: readlink a сам по себе возвращает b

примечание № 2: вместе с find -l утилита для вывода списка цепочек может быть легко написана на Perl, но также должна быть достаточно умной, чтобы обнаруживать циклы

readlink ничего не выведет, если у вас есть цикл. Полагаю, это лучше, чем застревать.

pjb@pjb-desktop:~/testlink$ ln -sf a c
pjb@pjb-desktop:~/testlink$ ls -l 
total 0
lrwxrwxrwx 1 pjb pjb 1 2010-02-23 08:48 a -> b
lrwxrwxrwx 1 pjb pjb 1 2010-02-23 08:48 b -> c
lrwxrwxrwx 1 pjb pjb 1 2010-02-23 09:03 c -> a
pjb@pjb-desktop:~/testlink$ readlink -e a
pjb@pjb-desktop:~/testlink$ # (note: no output)

Вот рекурсивная функция в Bash:

chain() { export chain; local link target; if [[ -z $chain ]]; then chain="$1"; fi; link=$(stat --printf=%N $1); while [[ $link =~ \-\> ]]; do target="${link##*\`}"; target="${target%\'}"; chain+=" -> $target"; chain "$target"; return; done; echo "$chain"; unset chain; }

В нескольких строках:

chain() {
    export chain
    local link target
    if [[ -z $chain ]]
    then
        chain="$1"
    fi
    link=$(stat --printf=%N "$1")
    while [[ $link =~ \-\> ]]
    do
        target="${link##*\`}"
        target="${target%\'}"
        chain+=" -> $target"
        if [[ ! $target =~ / && $1 =~ / ]]
        then
            target="${1%/*}/$target"
        fi
        chain "$target"
        return
    done
    echo "$chain"
    unset chain
}

Примеры:

$ chain d
d -> c -> b -> a
$ chain c
c -> b -> a
$ chain a
a

Это требует stat(1) которые могут отсутствовать в некоторых системах.

Он не будет работать, если имена содержат обратные кавычки, одинарные кавычки или "->". Он застревает в цикле с циклами символических ссылок (это можно решить с помощью ассоциативного массива в Bash 4). Он экспортирует переменную, называемую «цепочкой», независимо от того, используется ли она уже.

С ним могут быть и другие проблемы.

Редактировать:

Исправлена ​​проблема с некоторыми относительными символическими ссылками. Некоторые по-прежнему не работают, но для версии ниже не требуется, чтобы существовала цель ссылки.

Добавлена ​​версия, использующая ссылку для чтения:

chain ()
{
    export chain;
    local target;
    if [[ -z $chain ]]; then
        chain="$1";
    fi;
    target=$(readlink "$1");
    while [[ $target ]]; do
        chain+=" -> $target";
        if [[ ! $target =~ / && $1 =~ / ]]
        then
            target="${1%/*}/$target"
        fi
        chain "$target";
        return;
    done;
    echo "$chain";
    unset chain
}

Вы можете просто обработать вывод namei чем-то вроде awk или grep чтобы получить только нужные строки:

namei d | awk '$1=="l"'

или

namei d | egrep -e "->"