Ich habe zum Beispiel ein Verzeichnis
/path/to/directory
und ich möchte die Berechtigungen seiner Unterverzeichnisse auf einen bestimmten Wert setzen. Das ist ganz einfach:
find /path/to/directory -type d -exec chmod something {} +
Aber wie mache ich es umgekehrt? Ich muss die gleiche Berechtigung vergeben für
/path
und zu
/path/to
und zu
/path/to/directory
Ich habe viele Verzeichnisse wie dieses und suche nach einer Skriptlösung
Antwort1
Gehen Sie rekursiv nach oben vor, bis /
erreicht ist, und rufen Sie für jedes Verzeichnis eine Funktion auf.
#!/bin/sh
function mangleperms {
echo DEBUG would chmod 755 "$1"
}
function walktoroot {
DIR="$1"
HANDLE="$2"
if [ "$DIR" = "/" ]; then
return
fi
"$HANDLE" "$DIR"
# recursion (noun): see recursion
PARENTDIR=`dirname "$DIR"`
walktoroot "$PARENTDIR" "$HANDLE"
}
if [ -z "$1" ]; then
echo >&2 "Usage: walktoroot dir"
exit 1
fi
# TODO probably more edge cases on relative dirs, though there are means
# to fully qualify those
if [ "$1" = "." ]; then
DIR=`pwd`
else
DIR=$1
fi
walktoroot "$DIR" mangleperms
Z.B
$ pwd
/var/tmp/a/b/c
$ /home/jdoe/walktoroot .
DEBUG would chmod 755 /var/tmp/a/b/c
DEBUG would chmod 755 /var/tmp/a/b
DEBUG would chmod 755 /var/tmp/a
DEBUG would chmod 755 /var/tmp
DEBUG would chmod 755 /var
$
Antwort2
Mit find
können wir dies wie folgt tun:
find /path/to/dir -type d -prune -exec chmod 755 {} \; -exec sh -c '
while { set -- "${1%/*}"; case $1 in "" ) break ;; esac; }
do find "$1" -type d -prune -exec chmod 755 \{\} \;; done
' {} {} \;
Eine andere Methode mit einer recursive
Funktion:
fx() {
case $1 in ?* ) chmod 755 "$1"; fx "${1%/*}" ;; esac
}
# and then...
fx /path/to/dir
Antwort3
Nun, ein Pfad ist einfach eine Zeichenfolge. Sie können ihn also aufteilen /
und Ihren Befehl auf jedes Ergebnis anwenden. Sie können beispielsweise einen kleinen Perl-Einzeiler verwenden, um die übergeordneten Pfade auszudrucken:
$ path=/path/to/some/deeply/buried/thing
$ echo "$path" | perl -F"/" -lane 'for(1..$#F){print join("/",@F[0..$_])}'
/path
/path/to
/path/to/some
/path/to/some/deeply
/path/to/some/deeply/buried
/path/to/some/deeply/buried/thing
Wenn Sie jetzt aus diesem Perl-Befehl eine Funktion machen (führen Sie diesen Befehl aus oder fügen Sie ihn der Initialisierungsdatei Ihrer Shell hinzu, ~/.bashrc
wenn Sie verwenden bash
):
pexpand(){
printf '%s\n' "$1" | perl -F"/" -lane 'for(1..$#F){print join("/",@F[0..$_])}';
}
Jetzt können Sie es folgendermaßen verwenden:
$ pexpand $path
/path
/path/to
/path/to/some
/path/to/some/deeply
/path/to/some/deeply/buried
/path/to/some/deeply/buried/thing
Und Sie können Befehle in jedem Verzeichnis in diesem Pfad mit folgendem Befehl ausführen:
chmod 733 "$(pexpand "$path")"
Der obige Ansatz schlägt fehl, wenn Ihr Pfad Zeilenumbrüche enthalten kann. Eine robustere Version ist:
pexpand(){
perl -le '@F=split(/\//, $ARGV[0]);
for(1..$#F){printf "%s\0",join("/",@F[0..$_])}' "$1";
}
Dies ist allerdings umständlicher in der Anwendung:
pexpand "$path" | while IFS= read -r -d '' d; do chmod 733 "$d"; done
Wenn dies also mit beliebigen Eingaben funktionieren soll, würde ich stattdessen empfehlen, ein kleines Skript zu schreiben, das sowohl den Pfad als auch den Befehl als Eingabe verwendet:
#!/usr/bin/perl
my $comm = $ARGV[0] || die "At least 2 arguments are necessary\n";
my $path = $ARGV[1] || die "At least 2 arguments are necessary\n";
my @paths=split(/\//, $path);
for (1..$#paths) {
system("$comm \"" . join("/",@paths[0..$_]) . "\"");
}
Speichern Sie das Skript als etwas ( foo.pl
oder was auch immer) in einem Verzeichnis in Ihrem $PATH
, machen Sie es ausführbar ( chmod a+x ~/bin/foo.pl
), und Sie können es jetzt wie folgt ausführen:
foo.pl "chmod 700" /path/to/some/deeply/buried/thing