mirror of
git://git.code.sf.net/p/zsh/code
synced 2025-01-01 17:24:50 +01:00
Committed 15720 and 16927.
This commit is contained in:
parent
4e70abb13b
commit
74d50826cb
1 changed files with 161 additions and 35 deletions
|
@ -1,6 +1,35 @@
|
||||||
# function zmv {
|
# function zmv {
|
||||||
# zmv, zcp, zln:
|
# zmv, zcp, zln:
|
||||||
#
|
#
|
||||||
|
# This is a multiple move based on zsh pattern matching. To get the full
|
||||||
|
# power of it, you need a postgraduate degree in zsh. However, simple
|
||||||
|
# tasks work OK, so if that's all you need, here are some basic examples:
|
||||||
|
# zmv '(*).txt' '$1.lis'
|
||||||
|
# Rename foo.txt to foo.lis, etc. The parenthesis is the thing that
|
||||||
|
# gets replaced by the $1 (not the `*', as happens in mmv, and note the
|
||||||
|
# `$', not `=', so that you need to quote both words).
|
||||||
|
# zmv '(**/)(*).txt '$1$2.lis'
|
||||||
|
# The same, but scanning through subdirectories. The $1 becomes the full
|
||||||
|
# path. Note that you need to write it like this; you can't get away with
|
||||||
|
# '(**/*).txt'.
|
||||||
|
# zmv -w '**/*.txt' '$1$2.lis'
|
||||||
|
# noglob zmv -W **/*.txt **/*.lis
|
||||||
|
# These are the lazy version of the one above; with -w, zsh inserts the
|
||||||
|
# parentheses for you in the search pattern, and with -W it also inserts
|
||||||
|
# the numbered variables for you in the replacement pattern. The catch
|
||||||
|
# in the first version is that you don't need the / in the replacement
|
||||||
|
# pattern. (It's not really a catch, since $1 can be empty.) Note that
|
||||||
|
# -W actually inserts ${1}, ${2}, etc., so it works even if you put a
|
||||||
|
# number after a wildcard (such as zmv -W '*1.txt' '*2.txt').
|
||||||
|
# zmv -C '**/(*).txt' ~/save/'$1'.lis
|
||||||
|
# Copy, instead of move, all .txt files in subdirectories to .lis files
|
||||||
|
# in the single directory `~/save'. Note that the ~ was not quoted.
|
||||||
|
# You can test things safely by using the `-n' (no, not now) option.
|
||||||
|
# Clashes, where multiple files are renamed or copied to the same one, are
|
||||||
|
# picked up.
|
||||||
|
#
|
||||||
|
# Here's a more detailed description.
|
||||||
|
#
|
||||||
# Use zsh pattern matching to move, copy or link files, depending on
|
# Use zsh pattern matching to move, copy or link files, depending on
|
||||||
# the last two characters of the function name. The general syntax is
|
# the last two characters of the function name. The general syntax is
|
||||||
# zmv '<inpat>' '<outstring>'
|
# zmv '<inpat>' '<outstring>'
|
||||||
|
@ -8,19 +37,40 @@
|
||||||
# immediate expansion, while <outstring> is a string that will be
|
# immediate expansion, while <outstring> is a string that will be
|
||||||
# re-evaluated and hence may contain parameter substitutions, which should
|
# re-evaluated and hence may contain parameter substitutions, which should
|
||||||
# also be quoted. Each set of parentheses in <inpat> (apart from those
|
# also be quoted. Each set of parentheses in <inpat> (apart from those
|
||||||
# around glob qualifiers and globbing flags) may be referred to by a
|
# around glob qualifiers, if you use the -Q option, and globbing flags) may
|
||||||
# positional parameter in <outstring>, i.e. the first (...) matched is
|
# be referred to by a positional parameter in <outstring>, i.e. the first
|
||||||
# given by $1, and so on. For example,
|
# (...) matched is given by $1, and so on. For example,
|
||||||
# zmv '([a-z])(*).txt' '${(U)1}$2.txt'
|
# zmv '([a-z])(*).txt' '${(C)1}$2.txt'
|
||||||
# renames algernon.txt to Algernon.txt, boris.txt to Boris.txt and so on.
|
# renames algernon.txt to Algernon.txt, boris.txt to Boris.txt and so on.
|
||||||
# The original file matched can be referred to as $f in the second
|
# The original file matched can be referred to as $f in the second
|
||||||
# argument; accidental or deliberate use of other parameters is at owner's
|
# argument; accidental or deliberate use of other parameters is at owner's
|
||||||
# risk and is not covered by the (non-existent) guarantee.
|
# risk and is not covered by the (non-existent) guarantee.
|
||||||
#
|
#
|
||||||
# Any error --- a substitution resulted in an empty string, a
|
# As usual in zsh, /'s don't work inside parentheses. There is a special
|
||||||
# substitution did not change the file name, two substitutions gave the
|
# case for (**/) and (***/): these have the expected effect that the
|
||||||
# same result, the destination was an existing regular file and -f was not
|
# entire relevant path will be substituted by the appropriate positional
|
||||||
# given --- causes the entire function to abort without doing anything.
|
# parameter.
|
||||||
|
#
|
||||||
|
# There is a shortcut avoiding the use of parenthesis with the option -w
|
||||||
|
# (with wildcards), which picks out any expressions `*', `?', `<range>'
|
||||||
|
# (<->, <1-10>, etc.), `[...]', possibly followed by `#'s, `**/', `***/', and
|
||||||
|
# automatically parenthesises them. (You should quote any ['s or ]'s which
|
||||||
|
# appear inside [...] and which do not come from ranges of the form
|
||||||
|
# `[:alpha:]'.) So for example, in
|
||||||
|
# zmv -w '[[:upper:]]*' '${(L)1}$2'
|
||||||
|
# the $1 refers to the expression `[[:upper:]]' and the $2 refers to
|
||||||
|
# `*'. Thus this finds any file with an upper case first character and
|
||||||
|
# renames it to one with a lowercase first character. Note that any
|
||||||
|
# existing parentheses are active, too, so you must count accordingly.
|
||||||
|
# Furthermore, an expression like '(?)' will be rewritten as '((?))' --- in
|
||||||
|
# other words, parenthesising of wildcards is independent of any existing
|
||||||
|
# parentheses.
|
||||||
|
#
|
||||||
|
# Any file whose name is not changed by the substitution is simply ignored.
|
||||||
|
# Any error --- a substitution resulted in an empty string, two
|
||||||
|
# substitutions gave the same result, the destination was an existing
|
||||||
|
# regular file and -f was not given --- causes the entire function to abort
|
||||||
|
# without doing anything.
|
||||||
#
|
#
|
||||||
# Options:
|
# Options:
|
||||||
# -f force overwriting of destination files. Not currently passed
|
# -f force overwriting of destination files. Not currently passed
|
||||||
|
@ -30,7 +80,10 @@
|
||||||
# to execute it. Y or y will execute it, anything else will skip it.
|
# to execute it. Y or y will execute it, anything else will skip it.
|
||||||
# Note that you just need to type one character.
|
# Note that you just need to type one character.
|
||||||
# -n no execution: print what would happen, but don't do it.
|
# -n no execution: print what would happen, but don't do it.
|
||||||
# -q don't allow bare glob qualifiers in the filename pattern, see below.
|
# -q Turn bare glob qualifiers off: now assumed by default, so this
|
||||||
|
# has no effect.
|
||||||
|
# -Q Force bare glob qualifiers on. Don't turn this on unless you are
|
||||||
|
# actually using glob qualifiers in a pattern (see below).
|
||||||
# -s symbolic, passed down to ln; only works with zln or z?? -L.
|
# -s symbolic, passed down to ln; only works with zln or z?? -L.
|
||||||
# -v verbose: print line as it's being executed.
|
# -v verbose: print line as it's being executed.
|
||||||
# -o <optstring>
|
# -o <optstring>
|
||||||
|
@ -41,6 +94,10 @@
|
||||||
# Call <program> instead of cp, ln or mv. Whatever it does, it should
|
# Call <program> instead of cp, ln or mv. Whatever it does, it should
|
||||||
# at least understand the form '<program> -- <oldname> <newname>',
|
# at least understand the form '<program> -- <oldname> <newname>',
|
||||||
# where <oldname> and <newname> are filenames generated.
|
# where <oldname> and <newname> are filenames generated.
|
||||||
|
# -w Pick out wildcard parts of the pattern, as described above, and
|
||||||
|
# implicitly add parentheses for referring to them.
|
||||||
|
# -W Just like -w, with the addition of turning wildcards in the
|
||||||
|
# replacement pattern into sequential ${1} .. ${N} references.
|
||||||
# -C
|
# -C
|
||||||
# -L
|
# -L
|
||||||
# -M Force cp, ln or mv, respectively, regardless of the name of the
|
# -M Force cp, ln or mv, respectively, regardless of the name of the
|
||||||
|
@ -48,53 +105,64 @@
|
||||||
#
|
#
|
||||||
# Bugs:
|
# Bugs:
|
||||||
# Parenthesised expressions can be confused with glob qualifiers, for
|
# Parenthesised expressions can be confused with glob qualifiers, for
|
||||||
# example a trailing '(*)' is treated as a glob qualifier. Use -q to
|
# example a trailing '(*)' would be treated as a glob qualifier in
|
||||||
# turn off glob qualifiers, or (yuk) add a suitable dummy qualifier
|
# ordinary globbing. This has proved so annoying that glob qualifiers
|
||||||
# (e.g. `(.)') or dummy pattern (e.g. `(|)') at the end.
|
# are now turned off by default. To force the use of glob qualifiers,
|
||||||
|
# give the flag -Q.
|
||||||
#
|
#
|
||||||
# The second argument is re-evaluated in order to expand the parameters,
|
# The pattern is always treated as an extendedglob pattern. This
|
||||||
# so quoting may be a bit haphazard. In particular, a double quote
|
# can also be interpreted as a feature.
|
||||||
# will need an extra level of quoting.
|
|
||||||
#
|
|
||||||
# The pattern is always treated as an extendedglob pattern.
|
|
||||||
#
|
#
|
||||||
# Unbugs:
|
# Unbugs:
|
||||||
# You don't need braces around the 1 in expressions like '$1t' as
|
# You don't need braces around the 1 in expressions like '$1t' as
|
||||||
# non-positional parameters may not start with a number, although
|
# non-positional parameters may not start with a number, although
|
||||||
# paranoiacs like the author will probably put them there anyway.
|
# paranoiacs like the author will probably put them there anyway.
|
||||||
|
|
||||||
emulate -L zsh
|
emulate -RL zsh
|
||||||
setopt extendedglob
|
setopt extendedglob
|
||||||
|
|
||||||
local f g args match mbegin mend files action myname tmpf opt exec
|
local f g args match mbegin mend files action myname tmpf opt exec
|
||||||
local opt_f opt_i opt_n opt_q opt_s opt_M opt_C opt_L opt_o opt_p
|
local opt_f opt_i opt_n opt_q opt_Q opt_s opt_M opt_C opt_L
|
||||||
local pat repl errstr
|
local opt_o opt_p opt_v opt_w opt_W MATCH MBEGIN MEND
|
||||||
|
local pat repl errstr fpat hasglobqual opat
|
||||||
typeset -A from to
|
typeset -A from to
|
||||||
integer stat
|
integer stat
|
||||||
|
|
||||||
while getopts ":o:p:MCLfinqsv" opt; do
|
while getopts ":o:p:MCLfinqQsvwW" opt; do
|
||||||
if [[ $opt = "?" ]]; then
|
if [[ $opt = "?" ]]; then
|
||||||
print -P "%N: unrecognized option: -$OPTARG" >&2
|
print -P "%N: unrecognized option: -$OPTARG" >&2
|
||||||
return 1
|
return 1
|
||||||
fi
|
fi
|
||||||
eval "opt_$opt=${OPTARG:--$opt}"
|
eval "opt_$opt=${(q)OPTARG:--$opt}"
|
||||||
done
|
done
|
||||||
(( OPTIND > 1 )) && shift $(( OPTIND - 1 ))
|
(( OPTIND > 1 )) && shift $(( OPTIND - 1 ))
|
||||||
|
|
||||||
[[ -n $opt_q ]] && setopt nobareglobqual
|
[[ -z $opt_Q ]] && setopt nobareglobqual
|
||||||
[[ -n $opt_M ]] && action=mv
|
[[ -n $opt_M ]] && action=mv
|
||||||
[[ -n $opt_C ]] && action=cp
|
[[ -n $opt_C ]] && action=cp
|
||||||
[[ -n $opt_L ]] && action=ln
|
[[ -n $opt_L ]] && action=ln
|
||||||
[[ -n $opt_p ]] && action=$opt_p
|
[[ -n $opt_p ]] && action=$opt_p
|
||||||
|
|
||||||
if (( $# != 2 )); then
|
if (( $# != 2 )); then
|
||||||
print -P "Usage: %N oldpattern newpattern
|
print -P "Usage:
|
||||||
e.g. %N '(*).lis' '\$1.txt'" >&2
|
%N [OPTIONS] oldpattern newpattern
|
||||||
|
where oldpattern contains parenthesis surrounding patterns which will
|
||||||
|
be replaced in turn by \$1, \$2, ... in newpattern. For example,
|
||||||
|
%N '(*).lis' '\\\\\$1.txt'
|
||||||
|
renames 'foo.lis' to 'foo.txt', 'my.old.stuff.lis' to 'my.old.stuff.txt',
|
||||||
|
and so on. Something simpler (for basic commands) is the -W option:
|
||||||
|
%N -W '*.lis' '*.txt'
|
||||||
|
This does the same thing as the first command, but with automatic conversion
|
||||||
|
of the wildcards into the appropriate syntax. If you combine this with
|
||||||
|
noglob, you don't even need to quote the arguments. For example,
|
||||||
|
alias mmv='noglob zmv -W'
|
||||||
|
mmv *.c.orig orig/*.c" >&2
|
||||||
return 1
|
return 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
pat=$1
|
pat=$1
|
||||||
repl=$2
|
repl=$2
|
||||||
|
shift 2
|
||||||
|
|
||||||
if [[ -z $action ]]; then
|
if [[ -z $action ]]; then
|
||||||
# We can't necessarily get the name of the function directly, because
|
# We can't necessarily get the name of the function directly, because
|
||||||
|
@ -118,23 +186,80 @@ if [[ -n $opt_s && $action != ln ]]; then
|
||||||
return 1
|
return 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
files=(${~pat})
|
if [[ -n $opt_w || -n $opt_W ]]; then
|
||||||
|
# Parenthesise all wildcards.
|
||||||
if [[ -o bareglobqual && $pat = (#b)(*)\([^\)\|\~]##\) ]]; then
|
local tmp find
|
||||||
# strip off qualifiers for use as ordinary pattern
|
integer cnt=0
|
||||||
pat=$match[1]
|
# Well, this seems to work.
|
||||||
|
# The tricky bit is getting all forms of [...] correct, but as long
|
||||||
|
# as we require inactive bits to be backslashed its not so bad.
|
||||||
|
find='(#m)(\*\*#[/]|[*?]|\<[0-9]#-[0-9]#\>|\[(\[:[a-z]##:\]|\\\[|\\\]|[^\[\]]##)##\])\##'
|
||||||
|
tmp="${pat//${~find}/$[++cnt]}"
|
||||||
|
if [[ $cnt = 0 ]]; then
|
||||||
|
print -P "%N: warning: no wildcards were found in search pattern" >&2
|
||||||
|
else
|
||||||
|
pat="${pat//${~find}/($MATCH)}"
|
||||||
|
fi
|
||||||
|
if [[ -n $opt_W ]]; then
|
||||||
|
# Turn wildcards into ${1} .. ${N} references.
|
||||||
|
local open='${' close='}'
|
||||||
|
integer N=0
|
||||||
|
repl="${repl//${~find}/$open$[++N]$close}"
|
||||||
|
if [[ $N != $cnt ]]; then
|
||||||
|
print -P "%N: error: number of wildcards in each pattern must match" >&2
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
if [[ $N = 0 ]]; then
|
||||||
|
print -P "%N: warning: no wildcards were found in replacement pattern" >&2
|
||||||
|
fi
|
||||||
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
if [[ -n $opt_Q && $pat = (#b)(*)\([^\)\|\~]##\) ]]; then
|
||||||
|
hasglobqual=q
|
||||||
|
# strip off qualifiers for use as ordinary pattern
|
||||||
|
opat=$match[1]
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ $pat = (#b)(*)\((\*\*##/)\)(*) ]]; then
|
||||||
|
fpat="$match[1]$match[2]$match[3]"
|
||||||
|
# Now make sure we do depth-first searching.
|
||||||
|
# This is so that the names of any files are altered before the
|
||||||
|
# names of the directories they are in.
|
||||||
|
if [[ -n $opt_Q && -n $hasglobqual ]]; then
|
||||||
|
fpat[-1]="odon)"
|
||||||
|
else
|
||||||
|
setopt bareglobqual
|
||||||
|
fpat="${fpat}(odon)"
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
fpat=$pat
|
||||||
|
fi
|
||||||
|
files=(${~fpat})
|
||||||
|
|
||||||
|
[[ -n $hasglobqual ]] && pat=$opat
|
||||||
|
|
||||||
errs=()
|
errs=()
|
||||||
|
|
||||||
for f in $files; do
|
for f in $files; do
|
||||||
|
if [[ $pat = (#b)(*)\(\*\*##/\)(*) ]]; then
|
||||||
|
# This looks like a recursive glob. This isn't good enough,
|
||||||
|
# because we should really enforce that $match[1] and $match[2]
|
||||||
|
# don't match slashes unless they were explicitly given. But
|
||||||
|
# it's a start. It's fine for the classic case where (**/) is
|
||||||
|
# at the start of the pattern.
|
||||||
|
pat="$match[1](*/|)$match[2]"
|
||||||
|
fi
|
||||||
[[ -e $f && $f = (#b)${~pat} ]] || continue
|
[[ -e $f && $f = (#b)${~pat} ]] || continue
|
||||||
set -- $match
|
set -- "$match[@]"
|
||||||
eval g=\"$repl\"
|
g=${(e)repl}
|
||||||
if [[ -z $g ]]; then
|
if [[ -z $g ]]; then
|
||||||
errs=($errs "$f expanded to empty string")
|
errs=($errs "\`$f' expanded to an empty string")
|
||||||
elif [[ $f = $g ]]; then
|
elif [[ $f = $g ]]; then
|
||||||
errs=($errs "$f not altered by substitution")
|
# don't cause error: more useful just to skip
|
||||||
|
# errs=($errs "$f not altered by substitution")
|
||||||
|
[[ -n $opt_v ]] && print "$f not altered, ignored"
|
||||||
|
continue
|
||||||
elif [[ -n $from[$g] && ! -d $g ]]; then
|
elif [[ -n $from[$g] && ! -d $g ]]; then
|
||||||
errs=($errs "$f and $from[$g] both map to $g")
|
errs=($errs "$f and $from[$g] both map to $g")
|
||||||
elif [[ -f $g && -z $opt_f ]]; then
|
elif [[ -f $g && -z $opt_f ]]; then
|
||||||
|
@ -151,8 +276,9 @@ if (( $#errs )); then
|
||||||
fi
|
fi
|
||||||
|
|
||||||
for f in $files; do
|
for f in $files; do
|
||||||
|
[[ -z $to[$f] ]] && continue
|
||||||
exec=($action ${=opt_o} $opt_s -- $f $to[$f])
|
exec=($action ${=opt_o} $opt_s -- $f $to[$f])
|
||||||
[[ -n $opt_i$opt_n$opt_v ]] && print -- $exec
|
[[ -n $opt_i$opt_n$opt_v ]] && print -r -- ${(q)exec}
|
||||||
if [[ -n $opt_i ]]; then
|
if [[ -n $opt_i ]]; then
|
||||||
read -q 'opt?Execute? ' || continue
|
read -q 'opt?Execute? ' || continue
|
||||||
fi
|
fi
|
||||||
|
|
Loading…
Reference in a new issue