mirror of
git://git.code.sf.net/p/zsh/code
synced 2025-09-02 22:11:54 +02:00
This function is used only by revert, diff, and commit, none of which can run on unversioned files (those with status '?').
524 lines
18 KiB
Text
524 lines
18 KiB
Text
#compdef svn svnlite=svn svnadmin svnadmin-static=svnadmin svnliteadmin=svnadmin
|
|
|
|
_svn () {
|
|
local curcontext="$curcontext" state line expl ret=1
|
|
typeset -A opt_args
|
|
|
|
# Colons in values must be escaped.
|
|
local -A show_item_keys=(
|
|
kind "the kind of TARGET (file or dir)"
|
|
url "the URL of TARGET in the repository"
|
|
relative-url "the repository-relative URL"
|
|
repos-root-url "the repository root URL"
|
|
repos-uuid "the repository UUID"
|
|
repos-size "the size of TARGET in the repository (for files only)"
|
|
revision "the revision of TARGET"
|
|
last-changed-revision "the most recent revision in which TARGET was changed"
|
|
last-changed-date "the date of the last-changed revision"
|
|
last-changed-author "the author of the last-changed revision"
|
|
wc-root "the working copy root path"
|
|
schedule "'normal', 'add', 'delete', 'replace'"
|
|
depth "'infinity', 'immediates', 'files', 'empty', 'exclude'"
|
|
changelist "the changelist this file was added to, if any"
|
|
)
|
|
|
|
local update_policy
|
|
zstyle -s ":completion:*:*:$service:*" cache-policy update_policy
|
|
if [[ -z "$update_policy" ]]; then
|
|
zstyle ":completion:*:*:$service:*" cache-policy _svn_caching_policy
|
|
fi
|
|
|
|
_arguments -C -A "-*" \
|
|
'(-)--help[print help information]' \
|
|
'(*)--version[print client version information]' \
|
|
'1: :->cmds' \
|
|
'*:: :->args' && ret=0
|
|
|
|
local _svn_help_takes_verbose
|
|
if _cache_invalid svn-help-takes-verbose || ! _retrieve_cache svn-help-takes-verbose; then
|
|
[[ $(_call_program svn-help-v svn help help) == *--verbose* ]]
|
|
_svn_help_takes_verbose=$(( ! $? ))
|
|
_store_cache svn-help-takes-verbose _svn_help_takes_verbose
|
|
fi
|
|
if (( _svn_help_takes_verbose )); then
|
|
readonly dash_v="-v"
|
|
else
|
|
readonly dash_v
|
|
fi
|
|
unset _svn_help_takes_verbose
|
|
|
|
if [[ -n $state ]] && (( ! $+_svn_cmds )); then
|
|
typeset -gHA _svn_cmds
|
|
if _cache_invalid svn-cmds || ! _retrieve_cache svn-cmds; then
|
|
_svn_cmds=(
|
|
${=${(f)${${"$(_call_program commands svn help $dash_v)"#l#*Available subcommands:}%%Subversion is a tool*}}/(#s)[[:space:]]#(#b)([a-z-]##)[[:space:]]#(\([a-z, ?-]##\))#/$match[1] :$match[1]${match[2]:+:${${match[2]//[(),]}// /:}}:}
|
|
)
|
|
if (( $? == 0 )); then
|
|
_store_cache svn-cmds _svn_cmds
|
|
else
|
|
# Ensure we enter this block again on the next <TAB>.
|
|
unset _svn_cmds
|
|
fi
|
|
fi
|
|
fi
|
|
|
|
case $state in
|
|
cmds)
|
|
_wanted commands expl 'svn command' _svn_commands && ret=0
|
|
;;
|
|
args)
|
|
local cmd args usage idx
|
|
typeset -gHA _cache_svn_status _cache_svn_mtime
|
|
|
|
cmd="${${(k)_svn_cmds[(R)*:$words[1]:*]}:-${(k)_svn_cmds[(i):$words[1]:]}}"
|
|
if (( $#cmd )); then
|
|
curcontext="${curcontext%:*:*}:svn-${cmd}:"
|
|
|
|
if _cache_invalid svn-${cmd}-usage || \
|
|
! _retrieve_cache svn-${cmd}-usage;
|
|
then
|
|
usage=${${(M)${(f)"$(_call_program options svn help $dash_v -- $cmd)"}:#usage:*}#usage:*$cmd] }
|
|
_store_cache svn-${cmd}-usage usage
|
|
fi
|
|
if _cache_invalid svn-${cmd}-usage || \
|
|
! _retrieve_cache svn-${cmd}-args;
|
|
then
|
|
args=(
|
|
${=${${${(M)${(f)"$(_call_program options svn help $dash_v -- $cmd)"#(*Valid options:|(#e))}:#* :*}%% #:*}/ (arg|ARG)/:arg:}/(#b)(-##)([[:alpha:]]##) \[--([a-z-]##)\](:arg:)#/(--$match[3])$match[1]$match[2]$match[4] ($match[1]$match[2])--$match[3]$match[4]}
|
|
)
|
|
while (( idx=$args[(I)*--accept:arg:] )); do
|
|
args[idx]=( --accept'=:automatic conflict resolution action:((working\:working base\:base recommended\:recommended '"`for i j in p postpone mc mine-conflict tc theirs-conflict mf mine-full tf theirs-full e edit l launch; do print -rn $i\\\\:$j $j\\\\:$j ""; done `"'))' )
|
|
done
|
|
while (( idx=$args[(I)*--c(l|hangelist):arg:] )); do
|
|
args[idx]=( \*{--cl,--changelist}'=:change list:_svn_changelists' )
|
|
done
|
|
while (( idx=$args[(I)*--config-dir:arg:] )); do
|
|
args[idx]=( --config-dir'=:config dir:_directories' )
|
|
done
|
|
while (( idx=$args[(I)*--config-option:arg:] )); do
|
|
args[idx]=( '*--config-option=: :_svn_config_options' )
|
|
done
|
|
while (( idx=$args[(I)*--depth:arg:] )); do
|
|
args[idx]=( --depth'=:operation depth (how far to recurse):(empty files immediates infinity)' )
|
|
done
|
|
while (( idx=$args[(I)*(-F|--file):arg:] )); do
|
|
args[idx]=( '(-F --file)'{-F+,--file=}':log message file:_files' )
|
|
done
|
|
while (( idx=$args[(I)*--set-depth:arg:] )); do
|
|
args[idx]=( --set-depth'=[make working copy deeper or shallower]:new depth:(exclude empty files immediates infinity)' )
|
|
done
|
|
while (( idx=$args[(I)*--trust-server-cert-failures:arg:] )); do
|
|
args[idx]=( --trust-server-cert-failures'=:failures:_values -s , "certificate failures to ignore" "unknown-ca[unknown authority]" "cn-mismatch[hostname mismatch]" "expired[certificate expired]" "not-yet-valid[certificate not yet valid]" "other[all other failures]"' )
|
|
done
|
|
while (( idx=$args[(I)*--show-item:arg:] )); do
|
|
# (q) to quote the parentheses in the value
|
|
args[idx]=( --show-item'=:item key:(('"`for i j in ${(kv)show_item_keys}; do print -rn - $i\\\\:"${(q)j}" ""; done`"'))' )
|
|
done
|
|
# All other options get {-x+,--long-x=}
|
|
args=( ${args/(#b)(--[A-Za-z0-9-]##):arg:/$match[1]=:arg:} )
|
|
args=( ${args/(#b)([^=]):arg:/$match[1]+:arg:} )
|
|
_store_cache svn-${cmd}-args args
|
|
fi
|
|
|
|
case $cmd in;
|
|
(add)
|
|
args+=(
|
|
'*:file:_files -g "*(e:_svn_uncontrolled:)"'
|
|
)
|
|
;;
|
|
(auth)
|
|
args+=(
|
|
'*:auth pattern: '
|
|
)
|
|
;;
|
|
(changelist)
|
|
args[(r)--remove]='(1)--remove'
|
|
args+=(
|
|
'(--remove)1:changelist name:_svn_changelists'
|
|
'*:file:_files -g "*(e:_svn_controlled:)"'
|
|
)
|
|
;;
|
|
(commit)
|
|
args=(
|
|
${args/(#b)(*--file*):arg:/$match[1]:file:_files}
|
|
'*:file:_files -g "*(e:_svn_status:)"'
|
|
)
|
|
;;
|
|
(delete)
|
|
args+=(
|
|
'*:file:_files -g ".svn(/e:_svn_deletedfiles:)"'
|
|
)
|
|
;;
|
|
(diff)
|
|
args+=(
|
|
'*: : _alternative "files:file:_files -g \*\(e:_svn_status:\)" "urls:URL:_svn_urls"'
|
|
)
|
|
;;
|
|
(help)
|
|
args+=(
|
|
'*::sub command:_svn_commands'
|
|
)
|
|
;;
|
|
(import)
|
|
args+=(
|
|
'1:project directory or import location: _alternative "files:file:_files" "urls:URL:_svn_urls"'
|
|
'2:import location: _alternative "files:file:_files" "urls:URL:_svn_urls"'
|
|
)
|
|
;;
|
|
(log)
|
|
args+=(
|
|
'1: : _alternative "files:file:_files -g \*\(e:_svn_controlled:\)" "urls:URL:_svn_urls"'
|
|
'*:file:_files -g "*(e:_svn_controlled:)"'
|
|
)
|
|
;;
|
|
(mergeinfo)
|
|
args[(r)--show-revs=:arg:]=( '--show-revs=:revisions:(merged eligible)' )
|
|
;;
|
|
(patch)
|
|
args+=(
|
|
'1:patch file:_files'
|
|
'2::working copy to patch:_files'
|
|
)
|
|
;;
|
|
(propget|propedit|propdel)
|
|
args+=(
|
|
'1:property name:_svn_props'
|
|
'2:target: _alternative "files:file:_files" "urls:URL:_svn_urls"'
|
|
)
|
|
;;
|
|
(propset)
|
|
args=(
|
|
':propname:(svn:ignore svn:keywords svn:executable svn:eol-style svn:mime-type svn:externals svn:needs-lock svn:global-ignores svn:auto-props)'
|
|
':propval:->propset_propval'
|
|
${args/(#b)(*--file*):arg:/$match[1]:file:_files}
|
|
'*:path or url: _alternative "files:file:_files" "urls:URL:_svn_urls"'
|
|
)
|
|
;;
|
|
(resolve|resolved)
|
|
args+=(
|
|
'*:file:_files -g "*(e:_svn_conflicts:)"'
|
|
)
|
|
;;
|
|
(revert)
|
|
args+=(
|
|
'*:file:_files -g "(.svn|*)(/e:_svn_deletedfiles:,e:_svn_status:)"'
|
|
)
|
|
;;
|
|
(x-unshelve)
|
|
args+=( '1:shelf name:compadd - ${(f)"$(_call_program shelves svn x-shelves --quiet)"}' '2::shelf version' )
|
|
;;
|
|
(*)
|
|
case $usage in
|
|
*(SRC|DST|TARGET|URL*PATH)*)
|
|
args+=(
|
|
'*: : _alternative "files:file:_files" "urls:URL:_svn_urls"'
|
|
)
|
|
;;
|
|
*URL*) args+=( ':URL:_svn_urls' ) ;;
|
|
*PATH*) args+=( '*:file:_files' ) ;;
|
|
esac
|
|
;;
|
|
esac
|
|
|
|
_arguments "$args[@]" && ret=0
|
|
case $state in
|
|
(propset_propval)
|
|
case $words[2] in
|
|
(svn:executable|svn:needs-lock) compadd yes;;
|
|
(svn:keywords)
|
|
compset -q
|
|
# '_values -w' only excludes words in argv[1] or later, so
|
|
# install a dummy argv[0]. This affects Foo in [[svn propset
|
|
# svn:keywords 'Foo Bar Baz <TAB>]].
|
|
words=( dummy $words ); (( ++CURRENT ))
|
|
_values -s ' ' -w "keywords (or custom)" \
|
|
'(URL HeadURL)'{URL,HeadURL}'[URL for the head version of the file]' \
|
|
'(Author LastChangedBy)'{Author,LastChangedBy}'[last person to modify the file]' \
|
|
'(Date LastChangedDate)'{Date,LastChangedDate}'[date/time the file was last modified]' \
|
|
'(Rev Revision LastChangedRevision)'{Rev,Revision,LastChangedRevision}'[last revision the file changed]' \
|
|
Id'[compressed summary of URL,Revision,Date,Author]' \
|
|
Header"[like 'Id' but includes the full URL]";;
|
|
(svn:eol-style) compadd - CR LF CRLF native;;
|
|
(svn:mime-type) _mime_types;;
|
|
(*) _message 'property value';;
|
|
esac
|
|
esac
|
|
|
|
else
|
|
_message "unknown svn command: $words[1]"
|
|
fi
|
|
;;
|
|
esac
|
|
|
|
return ret
|
|
}
|
|
|
|
_svnadmin () {
|
|
local curcontext="$curcontext" state line ret=1
|
|
integer NORMARG
|
|
local context state_descr
|
|
typeset -A opt_args
|
|
|
|
_arguments -C \
|
|
'(-)--help[print help information]' \
|
|
'(- *)--version[print client version information]' \
|
|
'1: :->cmds' \
|
|
'*:: :->args' && ret=0
|
|
|
|
if [[ -n $state ]] && (( ! $+_svnadmin_cmds )); then
|
|
typeset -gHA _svnadmin_cmds
|
|
_svnadmin_cmds=(
|
|
${=${(f)${${"$(_call_program commands svnadmin help)"#l#*Available subcommands:}}}/(#s)[[:space:]]#(#b)([-a-z]##)[[:space:]]#(\([a-z, ?]##\))#/$match[1] :$match[1]${match[2]:+:${${match[2]//[(),]}// /:}}:}
|
|
)
|
|
fi
|
|
|
|
case $state in
|
|
cmds)
|
|
_wanted commands expl 'svnadmin command' _svnadmin_commands && ret=0
|
|
;;
|
|
args)
|
|
local cmd args usage
|
|
|
|
cmd="${${(k)_svnadmin_cmds[(R)*:$words[1]:*]}:-${(k)_svnadmin_cmds[(i):$words[1]:]}}"
|
|
if (( $#cmd )); then
|
|
curcontext="${curcontext%:*:*}:svnadmin-${cmd}:"
|
|
|
|
usage=${${(M)${(f)"$(_call_program options svnadmin help $cmd)"}:#$cmd: usage:*}#$cmd: usage: svnadmin $cmd }
|
|
args=(
|
|
${=${${${(M)${(f)"$(_call_program options svnadmin help $cmd)"#(*Valid options:|(#e))}:#* :*}%% #:*}/ (arg|ARG)/:arg:}/(#b)-([[:alpha:]]) \[--([a-z-]##)\](:arg:)#/(--$match[2])-$match[1]$match[3] (-$match[1])--$match[2]$match[3]}
|
|
)
|
|
# All options get {-x+,--long-x=}
|
|
args=( ${args/(#b)(--[A-Za-z0-9-]##):arg:/$match[1]=:arg:} )
|
|
args=( ${args/(#b)([^=]):arg:/$match[1]+:arg:} )
|
|
if [[ $usage == *REPOS_PATH* ]]; then
|
|
args+=( ":repository path:_files -/" )
|
|
case $cmd in
|
|
(freeze)
|
|
args+=( "*:arguments:->normal" )
|
|
;;
|
|
(hotcopy)
|
|
args+=( ":new repository:_files -/" )
|
|
;;
|
|
(setlog)
|
|
args+=( ": :_files" )
|
|
;;
|
|
(setrevprop)
|
|
args+=( ":property name" ":property value file:_files" )
|
|
;;
|
|
(delrevprop)
|
|
args+=( ":property name" )
|
|
;;
|
|
esac
|
|
elif [[ $cmd = help ]]; then
|
|
args+=( "*:subcommand:_svnadmin_commands" )
|
|
fi
|
|
|
|
_arguments -n -s -S : "$args[@]" && ret=0
|
|
|
|
case $state in
|
|
# Test cases:
|
|
# svnadmin freeze . rsync --<TAB> offers --file
|
|
# svnadmin freeze -- . rsync -<TAB> offers rsync's options
|
|
#
|
|
# Note: the NORMARG calculations here include one positional argument
|
|
# (the '.') before the command.
|
|
(normal)
|
|
if (( ${words[(i)--]} < CURRENT )); then
|
|
words[1,NORMARG]=()
|
|
(( CURRENT -= NORMARG ))
|
|
_normal && ret=0
|
|
elif (( NORMARG+1 == CURRENT )); then
|
|
# ### don't allow --options in this case
|
|
# TODO: this should just use '_normal -F "(-*)"', but _normal ignores its arguments.
|
|
_command_names -e && ret=0
|
|
fi
|
|
;;
|
|
esac
|
|
else
|
|
_message "unknown svnadmin command: $words[1]"
|
|
fi
|
|
;;
|
|
esac
|
|
|
|
return ret
|
|
}
|
|
|
|
(( $+functions[_svn_controlled] )) ||
|
|
_svn_controlled() {
|
|
# For svn<=1.6, this was implemented as:
|
|
# [[ -f ${(M)REPLY##*/}.svn/text-base/${REPLY##*/}.svn-base ]]
|
|
# However, because that implementation returns false for all files on svn>=1.7, and
|
|
# because 1.6 has been deprecated for 8 years and EOL for 6 years, we opt to DTRT
|
|
# for >=1.7. Therefore:
|
|
|
|
# TODO: Reimplement this function and _svn_uncontrolled for svn>=1.7.
|
|
# (Use 'svn st' or 'svn info', not 'svn ls')
|
|
return 0
|
|
}
|
|
|
|
|
|
(( $+functions[_svn_uncontrolled] )) ||
|
|
_svn_uncontrolled() {
|
|
# TODO: See comments in _svn_controlled
|
|
return 0
|
|
}
|
|
|
|
(( $+functions[_svn_conflicts] )) ||
|
|
_svn_conflicts() {
|
|
() { (( $# > 0 )) } $REPLY.(mine|r<->)(NY1)
|
|
}
|
|
|
|
(( $+functions[_svn_deletedfiles] )) ||
|
|
_svn_deletedfiles() {
|
|
# Typical usage would be _files -g '.svn(/e:_svn_deletedfiles:)'
|
|
local cont controlled
|
|
reply=( )
|
|
[[ $REPLY = (*/|).svn ]] || return
|
|
controlled=( $REPLY/text-base/*.svn-base(N:r:t) )
|
|
for cont in ${controlled}; do
|
|
[[ -e $REPLY:h/$cont ]] || reply+=( ${REPLY%.svn}$cont )
|
|
done
|
|
}
|
|
|
|
(( $+functions[_svn_status] )) ||
|
|
_svn_status() {
|
|
local dir=$REPLY:h
|
|
local pat="${1:-([ADMR~]|?M)}"
|
|
|
|
zmodload -F zsh/stat b:zstat 2>/dev/null
|
|
local key="$(zstat +device $dir):$(zstat +inode $dir)"
|
|
local mtime="$(zstat +mtime $dir/.svn/entries)"
|
|
|
|
if (( ! $+_cache_svn_status[$key] || _cache_svn_mtime[$key] != mtime )); then
|
|
_cache_svn_status[$key]="$(_call_program files svn status -q -N -- $dir)"
|
|
_cache_svn_mtime[$key]="$mtime"
|
|
fi
|
|
|
|
(( ${(M)#${(f)_cache_svn_status[$key]}:#(#s)${~pat}*$REPLY} ))
|
|
}
|
|
|
|
(( $+functions[_svn_remote_paths] )) ||
|
|
_svn_remote_paths() {
|
|
local expl remfiles remdispf remdispd suf ret=1 pfx='\^/' sub='^/'
|
|
|
|
# prefix must match a valid repository path format, either standard style
|
|
# schema://host/path/.. or ^/path/.. specifying a path relative to the
|
|
# root of the working directory repository. In the second form, allow the
|
|
# leading '^' be escaped in case the user has the extendedglob option set.
|
|
[[ -prefix *://*/ ]] ||
|
|
[[ -f .svn/entries && ( -prefix '^/' || -prefix '\^/' ) ]] ||
|
|
return 1
|
|
|
|
# return if remote access is not permitted
|
|
zstyle -T ":completion:${curcontext}:" remote-access || return 1
|
|
|
|
remfiles=( ${(f)"$(svn list $IPREFIX${${PREFIX%%[^/]#}/#$pfx/$sub} 2>/dev/null)"} )
|
|
(( $? == 0 )) || return 1
|
|
|
|
# you might consider trying to return early if $#remfiles is zero,
|
|
# but for whatever reason remfiles will always contain at least a
|
|
# single empty string; that case is handled correctly below.
|
|
|
|
compset -P '*/'
|
|
compset -S '/*' || suf=file
|
|
remdispf=(${remfiles:#*/})
|
|
remdispd=(${(M)remfiles:#*/})
|
|
_tags files
|
|
while _tags; do
|
|
while _next_label files expl ${suf:-directory}; do
|
|
# add files, unless there is a '/' immediately to the right
|
|
[[ -n $suf ]] &&
|
|
compadd -S ' ' -q "$@" "$expl[@]" -d remdispf $remdispf && ret=0
|
|
# add directories; use empty suffix if there is a '/' immediately to the right
|
|
compadd -S "${suf:+/}" -q "$@" "$expl[@]" -d remdispd ${remdispd%/} && ret=0
|
|
done
|
|
(( ret )) || return 0
|
|
done
|
|
|
|
return 1
|
|
}
|
|
|
|
(( $+functions[_svn_urls] )) ||
|
|
_svn_urls() {
|
|
local urlsch expl ret=1
|
|
|
|
# first try completing a remote path; if successful, we are all done..
|
|
_svn_remote_paths && return 0
|
|
|
|
# allow configuring svn repository locations using the 'urls' zstyle.
|
|
# always attempt completion of these because then matcher-list styles
|
|
# which do substring matching will work correctly.
|
|
_urls -S/ && ret=0
|
|
|
|
if [[ ! -prefix *://? ]] ; then
|
|
zstyle -a ":completion:${curcontext}:" url-schemas urlsch \
|
|
|| urlsch=( file:// http:// https:// svn:// svn+ssh:// )
|
|
|
|
if (( $#urlsch )) ; then
|
|
compset -S '[^:]*'
|
|
_wanted url-schemas expl 'URL schema' compadd -S '' - $urlsch[@] && ret=0
|
|
fi
|
|
fi
|
|
|
|
return ret
|
|
}
|
|
|
|
(( $+functions[_svn_commands] )) ||
|
|
_svn_commands() {
|
|
compadd "$@" -k _svn_cmds || compadd "$@" ${(s.:.)_svn_cmds}
|
|
}
|
|
|
|
(( $+functions[_svnadmin_command] )) ||
|
|
_svnadmin_commands() {
|
|
compadd "$@" -k _svnadmin_cmds || compadd "$@" ${(s.:.)_svnadmin_cmds}
|
|
}
|
|
|
|
(( $+functions[_svn_config_options] )) ||
|
|
_svn_config_options() {
|
|
local -a expl suf
|
|
local cfgfile
|
|
compset -S ':*' || suf=( -qS : )
|
|
if compset -P 2 '*:'; then
|
|
if compset -P '*='; then
|
|
_message -e values 'value'
|
|
else
|
|
_message -e options 'option'
|
|
fi
|
|
elif compset -P 1 '*:'; then
|
|
cfgfile=( ~/.subversion/${(M)${IPREFIX%:}%(config|servers)}(N) /dev/null )
|
|
_description sections expl 'section'
|
|
compadd $suf "$expl[@]" ${${${(M)${(f)"$(<${cfgfile[1]})"}:#\[*\]}#\[}%\]}
|
|
else
|
|
_description config-files expl 'configuration file'
|
|
compadd $suf "$expl[@]" config servers
|
|
fi
|
|
}
|
|
|
|
(( $+functions[_svn_props] )) ||
|
|
_svn_props() {
|
|
local properties
|
|
|
|
properties=( ${${(M)${(f)"$(svn proplist 2>/dev/null)"}:# [^ ]*}# } )
|
|
compadd "$@" -a properties && return 0
|
|
}
|
|
|
|
(( $+functions[_svn_changelists] )) ||
|
|
_svn_changelists() {
|
|
local cls
|
|
|
|
cls=( ${${${(M)${(f)"$(_call_program changelists svn status 2>/dev/null)"}:#--- Changelist*}%??}##*\'} )
|
|
compadd "$@" -a cls && return 0
|
|
}
|
|
|
|
_subversion () {
|
|
case $service in
|
|
(svn) _svn "$@" ;;
|
|
(svnadmin) _svnadmin "$@" ;;
|
|
esac
|
|
}
|
|
|
|
_svn_caching_policy() {
|
|
[[ =$service -nt $1 ]]
|
|
}
|
|
|
|
_subversion "$@"
|