
長いパスを誤って入力したり、存在しないパスのログファイルエントリを受け取ったりすることがよくあります。一致するパスが見つかるまでパス階層をナビゲートできるコマンドまたはシェル関数はありますか?
コマンドライン交換は次のとおりです。
$ ls /var/lib/my/supper/complicated/path
File or directory not found
$ fixpath /var/lib/my/supper/complicated/path
Found /var/lib/my
ボーナス機能を使用すると、類似性の検出が役に立ちます。
$ fixpath /var/lib/my/supper/complicated/path
Found /var/lib/my
Did you mean /var/lib/my/super/complicated/path
答え1
$ fixpath /home/user/docus/collection/unix/djakl/jfkdsl/dfjksld/fsdkl
Found /home/user/docus/collection/unix
$ type fixpath
fixpath is a function
fixpath ()
{
CURDIR="$PWD";
DIR="$1";
E=1;
while [ $E -eq 1 ]; do
cd "$DIR" 2> /dev/null && {
E=0;
echo Found "$DIR"
} || {
DIR="${DIR%/*}";
DIR="${DIR:-/}"
};
done;
cd "$CURDIR"
}
ほとんどの最新のシェルは、自己類似性の自動修正を実行できます。少なくともbashとzshの場合です。
答え2
次のシェル関数はパス名を使用して、実際に存在する最も長いビットを決定します。パスが完全に空であるか、実際に存在するファイル(またはディレクトリなど)に対応するまで、パスから最後のパス要素を削除してこれを行います。
fixpath () (
pathname="$1"
while [ -n "$pathname" ] && [ ! -e "$pathname" ]; do
pathname=${pathname%/*}
case "$pathname" in
*/*) continue ;;
*) [ -e "$pathname" ] || pathname=''
break
esac
done
printf 'Longest existing path in "%s" is "%s"\n' "$1" "$pathname"
)
この文は、変数なしでcase
ループを終了するために必要です(つまり、変数置換によって変更されないことを意味します)。これにより、場合によっては無限ループを回避できます。$pathname
/
テストしてみてください:
$ fixpath /home/kk/blaha/123
Longest existing path in "/home/kk/blaha/123" is "/home/kk"
$ fixpath "$HOME/.profile"
Longest existing path in "/home/kk/.profile" is "/home/kk/.profile"
$ fixpath "n"
Longest existing path in "n" is ""