zsh-workers
 help / color / mirror / code / Atom feed
* [PATCH] better systemctl completion
@ 2011-10-23 16:27 foudfou
  2011-10-24  9:01 ` Peter Stephenson
  0 siblings, 1 reply; 6+ messages in thread
From: foudfou @ 2011-10-23 16:27 UTC (permalink / raw)
  To: zsh-workers

[-- Attachment #1: Type: text/plain, Size: 398 bytes --]

Hi all,

Attached is my proposal of a systemctl completion function. Compared to the existing function shipped with zsh, it brings the following enhancements:
- up-to-date command/arguments definitions
- finer per-command completion of units

Best,
Foudil

---
  Completion/Unix/Command/_systemctl |  339 ++++++++++++++++++++++++++++--------
  1 files changed, 264 insertions(+), 75 deletions(-)



[-- Warning: decoded text below may be mangled, UTF-8 assumed --]
[-- Attachment #2: 0001-better-systemctl-completion.patch --]
[-- Type: text/x-patch; name="0001-better-systemctl-completion.patch", Size: 13038 bytes --]

diff --git a/Completion/Unix/Command/_systemctl b/Completion/Unix/Command/_systemctl
index 5dd35db..801a8e0 100644
--- a/Completion/Unix/Command/_systemctl
+++ b/Completion/Unix/Command/_systemctl
@@ -1,102 +1,291 @@
 #compdef systemctl
 
-# Completion for systemd's systemctl
-# Version 1.0 ARF 3 August 2011
+# Copyright (c) 2011 Foudil Brétel <foudil.newbie+zshsystemctl@gmail.com>
+#
+# This file is released under the GPLv3.
+#
+# inspired from _yum and systemctl-bash-completion.sh (shipped with systemctl)
+#
+# TODO: enable options after commands. Ex: systemctl list-units --all --full
 
-# list the unit completions.
-_systemd_units(){
-  local -a units
-  local expl
-  units=(${(f)"$( systemctl --full list-units | sed -e 's/ .*//' )"})
-  _wanted keys expl 'units' compadd "$units[@]"
-}
-
-# list the job completions.
-_systemd_jobs(){
-  local -a jobs
-  local expl
-  jobs=(${(f)"$( systemctl --full list-jobs | sed -e 's/ .*//' )"})
-  _wanted keys expl 'jobs' compadd "$jobs[@]"
-}
-
-# list the snapshot completions.
-_systemd_snapshots(){
-  local -a ss
-  local expl
-  ss=(${(f)"$(
-  	systemctl dump | sed -n -e 's/^-> Unit \(.*\.snapshot\):/\1/p'
-  )"})
-  _wanted keys expl 'snapshots' compadd "$ss[@]"
-}
-
-_systemctl(){
-  local -a _systemctl_cmds
+# Main dispatcher
+_systemctl()
+{
+  local curcontext="$curcontext" state lstate line
 
-  # Determine the sub commands
-  #$(systemctl --help  | sed -n -e 's/^  \([a-z]\)/\1/p' | sed -e 's/ .*//')
-  _systemctl_cmds=( 
-        list-units start stop reload restart try-restart reload-or-
-        restart reload-or-try-restart isolate kill is-active status show
-        reset-failed enable disable is-enabled load list-jobs cancel
-        monitor dump dot snapshot delete daemon-reload daemon-reexec show-
-        environment set-environment unset-environment default rescue
-        emergency halt poweroff reboot kexec exit
-  )
-
-  local curcontext="$curcontext" state line expl
-  typeset -A opt_args
-
-  # Initial flags
-  _arguments -A '-*' \
-    '--help[Show help]' \
+  # -s for aggregated options like -aP
+  _arguments -s \
+    {-h,--help}'[Show help]' \
     '--version[Show package version]' \
-    '(-a,--all)'{-a,--all}'[Show all units/properties, including dead/empty ones]' \
-    '--full[Dont ellipsize unit names on output]' \
+    {-t,--type=}'[List only units of a particular type]:unit type:(automount device mount path service snapshot socket swap target timer)' \
+    \*{-p,--property=}'[Show only properties by specific name]:unit property:' \
+    {-a,--all}'[Show all units/properties, including dead/empty ones]' \
     '--failed[Show only failed units]' \
+    "--full[Don't ellipsize unit names on output]" \
     '--fail[When queueing a new job, fail if conflicting jobs are pending]' \
-    '--ignore-dependencies[ignore dependencies]' \
-    '(-p,--privileged)'{-p,--privileged}'[Acquire privileges before execution]' \
-    '(-q,--quiet)'{-q,--quiet}'[Suppress output]' \
+    '--ignore-dependencies[When queueing a new job, ignore all its dependencies]' \
+    '--kill-mode=[How to send signal]:killmode:(control-group process)' \
+    '--kill-who=[Who to send signal to]:killwho:(main control all)' \
+    {-s,--signal=}'[Which signal to send]:signal:_signals' \
+    {-H,--host=}'[Show information for remote host]:userathost:_hosts_or_user_at_host' \
+    {-P,--privileged}'[Acquire privileges before execution]' \
+    {-q,--quiet}'[Suppress output]' \
     '--no-block[Do not wait until operation finished]' \
-    '--no-wall[Dont send wall message before halt/power-off/reboot]' \
-    '--no-reload[dont reload daemon configuration]' \
-    '--no-pager[Do use pager]' \
+    "--no-wall[Don't send wall message before halt/power-off/reboot]" \
+    "--no-reload[When enabling/disabling unit files, don't reload daemon configuration]" \
+    '--no-pager[Do not pipe output into a pager]' \
     '--no-ask-password[Do not ask for system passwords]' \
     '--order[When generating graph for dot, show only order]' \
     '--require[When generating graph for dot, show only requirement]' \
     '--system[Connect to system manager]' \
     '--user[Connect to user service manager]' \
     '--global[Enable/disable unit files globally]' \
-    '(-f,--force)'{-f,--force}'[When enabling unit files, override existing symlinks. When shutting down, execute action immediately]' \
+    {-f,--force}'[When enabling unit files, override existing symlinks. When shutting down, execute action immediately]' \
     '--defaults[When disabling unit files, remove default symlinks only]' \
-    '*::command:->subcmd' && return 0
+    '*::systemctl command:_systemctl_command'
+}
+
+_hosts_or_user_at_host()
+{
+  _alternative \
+    'users-hosts:: _user_at_host' \
+    'hosts:: _hosts'
+}
+
+(( $+functions[_systemctl_command] )) || _systemctl_command()
+{
+  local -a _systemctl_cmds
+  _systemctl_cmds=(
+    "list-units:List units"
+    "start:Start (activate) one or more units"
+    "stop:Stop (deactivate) one or more units"
+    "reload:Reload one or more units"
+    "restart:Start or restart one or more units"
+    "try-restart:Restart one or more units if active"
+    "reload-or-restart:Reload one or more units is possible, otherwise start or restart"
+    "reload-or-try-restart:Reload one or more units is possible, otherwise restart if active"
+    "isolate:Start one unit and stop all others"
+    "kill:Send signal to processes of a unit"
+    "is-active:Check whether units are active"
+    "status:Show runtime status of one or more units"
+    "show:Show properties of one or more units/jobs or the manager"
+    "reset-failed:Reset failed state for all, one, or more units"
+    "enable:Enable one or more unit files"
+    "disable:Disable one or more unit files"
+    "is-enabled:Check whether unit files are enabled"
+    "load:Load one or more units"
+    "list-jobs:List jobs"
+    "cancel:Cancel all, one, or more jobs"
+    "monitor:Monitor unit/job changes"
+    "dump:Dump server status"
+    "dot:Dump dependency graph for dot(1)"
+    "snapshot:Create a snapshot"
+    "delete:Remove one or more snapshots"
+    "daemon-reload:Reload systemd manager configuration"
+    "daemon-reexec:Reexecute systemd manager"
+    "show-environment:Dump environment"
+    "set-environment:Set one or more environment variables"
+    "unset-environment:Unset one or more environment variables"
+    "default:Enter system default mode"
+    "rescue:Enter system rescue mode"
+    "emergency:Enter system emergency mode"
+    "halt:Shut down and halt the system"
+    "poweroff:Shut down and power-off the system"
+    "reboot:Shut down and reboot the system"
+    "kexec:Shut down and reboot the system with kexec"
+    "exit:Ask for user instance termination"
+  )
 
-  # Complete subcommands.
   if (( CURRENT == 1 )); then
-    _describe -t commands "command" _systemctl_cmds
-    return
+    _describe -t commands 'systemctl command' _systemctl_cmds || compadd "$@"
+  else
+    local curcontext="$curcontext"
+
+    cmd="${${_systemctl_cmds[(r)$words[1]:*]%%:*}}"
+    # Deal with any aliases
+    case $cmd in
+      condrestart) cmd="try-restart";;
+      force-reload) cmd="reload-or-try-restart";;
+    esac
+
+    if (( $#cmd )); then
+      curcontext="${curcontext%:*:*}:systemctl-${cmd}:"
+
+      local update_policy
+      zstyle -s ":completion:${curcontext}:" cache-policy update_policy
+      if [[ -z "$update_policy" ]]; then
+        zstyle ":completion:${curcontext}:" cache-policy _systemctl_caching_policy
+      fi
+
+      _call_function ret _systemctl_$cmd || _message 'no more arguments'
+    else
+      _message "unknown systemctl command: $words[1]"
+    fi
+    return ret
   fi
+}
 
-  # handle arguments to the subcommands.
-  case $words[1] in
-  	start|restart|reload|stop|status|try-restart|reload-or-restart|reload-or-try-restart|isolate|kill|is-active|show|reset-failed|enable|disable|is-enabled|load)
-		# many units can be listed
-		_arguments "*:key:_systemd_units"
-	;;
+# Fills the unit lists
+_systemctl_all_units()
+{
+  if ( [[ ${+_sys_all_units} -eq 0 ]] || _cache_invalid SYS_ALL_UNITS ) &&
+    ! _retrieve_cache SYS_ALL_UNITS;
+  then
+    _sys_all_units=( $(systemctl list-units --full --all | cut -d' ' -f1 \
+      2>/dev/null) )
+    _store_cache SYS_ALL_UNITS _sys_all_units
+  fi
+}
 
-	cancel) _arguments "*:key:_systemd_jobs" ;;
+_systemctl_inactive_units()
+{
+  _sys_inactive_units=( $(systemctl list-units --full --all \
+    | awk '$3 != "active" {print $1}'  2>/dev/null) )
+}
 
-	# snapshots
-	snapshot) _normal ;;
-	delete) _arguments "*:key:_systemd_snapshots" ;;
+_systemctl_active_units()
+{
+  _sys_active_units=( $(systemctl list-units --full | cut -d' ' -f1 \
+    2>/dev/null) )
+}
 
-	# no arguments
-	dump|dot|monitor|daemon-reload|daemon-reexec|show-environment|default|rescue|emergency|halt|poweroff|reboot|kexec|exit|list-jobs|list-units)
-	;;
+_systemctl_failed_units()
+{
+  _sys_failed_units=( $(systemctl list-units --full --failed | cut -d' ' -f1
+    2>/dev/null) )
+}
+
+_filter_units_by_property () {
+  local property=$1 value=$2 ; shift ; shift
+  local -a units ; units=($*)
+  local -a props ; props=( $(systemctl show --property "$property" -- \
+    ${units[*]} | grep -v '^$') )
+  for ((i=0; $i < ${#units[*]}; i++)); do
+    if [[ "${props[i]}" = "$property=$value" ]]; then
+      echo "${units[i]}"
+    fi
+  done
+}
+
+# Completion functions for ALL_UNITS
+for fun in enable disable is-active is-enabled status show ; do
+  (( $+functions[_systemctl_$fun] )) || _systemctl_$fun()
+  {
+    _systemctl_all_units
+    compadd "$@" -a -- _sys_all_units
+  }
+done
+
+# Completion functions for STARTABLE_UNITS
+(( $+functions[_systemctl_start] )) || _systemctl_start()
+{
+  _systemctl_inactive_units
+  compadd "$@" -- $( _filter_units_by_property CanStart yes \
+    ${_sys_inactive_units[*]} | grep -Ev '\.(device|snapshot)$' )
+}
+
+# Completion functions for RESTARTABLE_UNITS
+for fun in restart reload-or-restart ; do
+  (( $+functions[_systemctl_$fun] )) || _systemctl_$fun()
+  {
+    _systemctl_all_units
+    compadd "$@" -- $( _filter_units_by_property CanStart yes \
+      ${_sys_all_units[*]} | grep -Ev '\.(device|snapshot|socket|timer)$' )
+  }
+done
+
+# Completion functions for STOPPABLE_UNITS
+for fun in stop kill try-restart condrestart ; do
+  (( $+functions[_systemctl_$fun] )) || _systemctl_$fun()
+  {
+    _systemctl_active_units
+    compadd "$@" -- $( _filter_units_by_property CanStop yes \
+      ${_sys_active_units[*]} )
+  }
+done
+
+# Completion functions for RELOADABLE_UNITS
+for fun in reload reload-or-try-restart force-reload ; do
+  (( $+functions[_systemctl_$fun] )) || _systemctl_$fun()
+  {
+    _systemctl_active_units
+    compadd "$@" -- $( _filter_units_by_property CanReload yes \
+      ${_sys_active_units[*]} )
+  }
+done
+
+# Completion functions for ISOLATABLE_UNITS
+(( $+functions[_systemctl_isolate] )) || _systemctl_isolate()
+{
+  _systemctl_all_units
+  compadd "$@" -- $( _filter_units_by_property AllowIsolate yes \
+    ${_sys_all_units[*]} )
+}
+
+# Completion functions for FAILED_UNITS
+(( $+functions[_systemctl_reset-failed] )) || _systemctl_reset-failed()
+{
+  _systemctl_failed_units
+  compadd "$@" -a -- _sys_failed_units || _message "no failed-unit found"
+}
+
+# Completion functions for JOBS
+(( $+functions[_systemctl_cancel] )) || _systemctl_cancel()
+{
+  compadd "$@" -- $( systemctl list-jobs | cut -d' ' -f1  2>/dev/null ) || \
+    _message "no job found"
+}
+
+# Completion functions for SNAPSHOTS
+(( $+functions[_systemctl_delete] )) || _systemctl_delete()
+{
+  compadd "$@" -- $( systemctl list-units --type snapshot --full --all \
+    | cut -d' ' -f1  2>/dev/null ) || _message "no snampshot found"
+}
+
+# Completion functions for ENVS
+(( $+functions[_systemctl_set-environment] )) || _systemctl_set-environment()
+{
+  compadd "$@" -S '' -- $( systemctl show-environment \
+    | sed 's_\([^=]\+=\).*_\1_' )
+}
+(( $+functions[_systemctl_unset-environment] )) || _systemctl_unset-environment()
+{
+  compadd "$@" -S '' -- $( systemctl show-environment \
+    | sed 's_\([^=]\+\)=.*_\1_' )
+}
+
+# no completion for:
+#   [STANDALONE]='daemon-reexec daemon-reload default dot dump emergency exit
+#                 halt kexec list-jobs list-units monitor poweroff reboot
+#                 rescue show-environment'
+#         [NAME]='snapshot load'
+
+_systemctl_caching_policy()
+{
+  local _sysunits
+  local -a oldcache
+
+  # rebuild if cache is more than a day old
+  oldcache=( "$1"(mh+1) )
+  (( $#oldcache )) && return 0
+
+  _sysunits=($(systemctl --full --all | cut -d' ' -f1))
+
+  if (( $#_sysunits )); then
+    for unit in $_sysunits; do
+      [[ "$unit" -nt "$1" ]] && return 0
+    done
+  fi
 
-	*) _message "unknown systemctl command: $words[1]" ;;
-  esac
+  return 1
 }
 
 _systemctl "$@"
 
+# Local Variables:
+# mode: sh
+# sh-indentation: 2
+# indent-tabs-mode: nil
+# sh-basic-offset: 2
+# End:


^ permalink raw reply	[flat|nested] 6+ messages in thread

* Re: [PATCH] better systemctl completion
  2011-10-23 16:27 [PATCH] better systemctl completion foudfou
@ 2011-10-24  9:01 ` Peter Stephenson
  2011-10-24 11:19   ` foudfou
  0 siblings, 1 reply; 6+ messages in thread
From: Peter Stephenson @ 2011-10-24  9:01 UTC (permalink / raw)
  To: foudfou, zsh-workers

On Sun, 23 Oct 2011 18:27:27 +0200
foudfou <foudil.newbie+gmane@gmail.com> wrote:
> Attached is my proposal of a systemctl completion function. Compared
> to the existing function shipped with zsh, it brings the following
> enhancements: 
> - up-to-date command/arguments definitions
> - finer per-command completion of units

Please could you send a context diff ("diff -u") showing what it is
you're changing.

Thanks.

-- 
Peter Stephenson <pws@csr.com>            Software Engineer
Tel: +44 (0)1223 692070                   Cambridge Silicon Radio Limited
Churchill House, Cambridge Business Park, Cowley Road, Cambridge, CB4 0WZ, UK


Member of the CSR plc group of companies. CSR plc registered in England and Wales, registered number 4187346, registered office Churchill House, Cambridge Business Park, Cowley Road, Cambridge, CB4 0WZ, United Kingdom
More information can be found at www.csr.com. Follow CSR on Twitter at http://twitter.com/CSR_PLC and read our blog at www.csr.com/blog


^ permalink raw reply	[flat|nested] 6+ messages in thread

* Re: [PATCH] better systemctl completion
  2011-10-24  9:01 ` Peter Stephenson
@ 2011-10-24 11:19   ` foudfou
  2011-10-24 11:27     ` Peter Stephenson
  0 siblings, 1 reply; 6+ messages in thread
From: foudfou @ 2011-10-24 11:19 UTC (permalink / raw)
  To: zsh-workers

On 24/10/2011 11:01, Peter Stephenson wrote:
> On Sun, 23 Oct 2011 18:27:27 +0200
> foudfou<foudil.newbie+gmane@gmail.com>  wrote:
>> Attached is my proposal of a systemctl completion function. Compared
>> to the existing function shipped with zsh, it brings the following
>> enhancements:
>> - up-to-date command/arguments definitions
>> - finer per-command completion of units
>
> Please could you send a context diff ("diff -u") showing what it is
> you're changing.

?... I believe the attachment is already a patch produced by "diff -u" (git-format-patch actually).

What may not be clear is that my proposal is a replacement of the existing completion function.

Best,
Foudil


^ permalink raw reply	[flat|nested] 6+ messages in thread

* Re: [PATCH] better systemctl completion
  2011-10-24 11:19   ` foudfou
@ 2011-10-24 11:27     ` Peter Stephenson
  2011-10-24 18:45       ` Frank Terbeck
  0 siblings, 1 reply; 6+ messages in thread
From: Peter Stephenson @ 2011-10-24 11:27 UTC (permalink / raw)
  To: foudfou, zsh-workers

On Mon, 24 Oct 2011 13:19:45 +0200
foudfou <foudil.newbie+gmane@gmail.com> wrote:
> ?... I believe the attachment is already a patch produced by "diff -u"
> (git-format-patch actually).

Sorry, my mistake.  I'll apply the patch.

> What may not be clear is that my proposal is a replacement of the
> existing completion function.

Actually, that's what confused me... you said it was a replacement
rather than an update.

-- 
Peter Stephenson <pws@csr.com>            Software Engineer
Tel: +44 (0)1223 692070                   Cambridge Silicon Radio Limited
Churchill House, Cambridge Business Park, Cowley Road, Cambridge, CB4 0WZ, UK


Member of the CSR plc group of companies. CSR plc registered in England and Wales, registered number 4187346, registered office Churchill House, Cambridge Business Park, Cowley Road, Cambridge, CB4 0WZ, United Kingdom
More information can be found at www.csr.com. Follow CSR on Twitter at http://twitter.com/CSR_PLC and read our blog at www.csr.com/blog


^ permalink raw reply	[flat|nested] 6+ messages in thread

* Re: [PATCH] better systemctl completion
  2011-10-24 11:27     ` Peter Stephenson
@ 2011-10-24 18:45       ` Frank Terbeck
  2011-10-25  9:00         ` Peter Stephenson
  0 siblings, 1 reply; 6+ messages in thread
From: Frank Terbeck @ 2011-10-24 18:45 UTC (permalink / raw)
  To: zsh-workers

Peter Stephenson wrote:
[...]
> Sorry, my mistake.  I'll apply the patch.
[...]

Um. The file got a new licence as GPLv3 ("# This file is released under
the GPLv3."). I am obviously not a lawyer, but I know that we requested
a dual/relicencing before including an elaborate update for the `_hg'
completion function:

  <http://www.zsh.org/mla/workers//2007/msg01022.html>

I've seen that there a few files that mention the GPL in our repository
right now: Unix/Command/_darcs, Unix/Command/_osc,
openSUSE/Command/_osc, openSUSE/Command/_zypper.

I'd like to know what the policy WRT to licences of completion files
etc. is and/or if we are allowed to do that (I really have no idea).

Regards, Frank


^ permalink raw reply	[flat|nested] 6+ messages in thread

* Re: [PATCH] better systemctl completion
  2011-10-24 18:45       ` Frank Terbeck
@ 2011-10-25  9:00         ` Peter Stephenson
  0 siblings, 0 replies; 6+ messages in thread
From: Peter Stephenson @ 2011-10-25  9:00 UTC (permalink / raw)
  To: zsh-workers

On Mon, 24 Oct 2011 20:45:58 +0200
Frank Terbeck <ft@bewatermyfriend.org> wrote:
> Peter Stephenson wrote:
> [...]
> > Sorry, my mistake.  I'll apply the patch.
> [...]
> 
> Um. The file got a new licence as GPLv3 ("# This file is released under
> the GPLv3.").

Certainly that's better avoided, it does complicate matters.  We
wouldn't be able to do that if it were part of the main shell without
repercussions (we already have to be careful if the shell gets linked
against GDBM).  I don't think it has particular issues in the case
of individual completion functions --- though if they became part of the
library rather than the top level that would then affect the rest of the
completion functions.

Given that it's partly based on the bash systemctl completion you can
argue it's inevitable in this case.

Note the zsh licence already says "any provisions made in individual
files take precedence" so it does provide for the general case (though
this case is particularly complicated).

-- 
Peter Stephenson <pws@csr.com>            Software Engineer
Tel: +44 (0)1223 692070                   Cambridge Silicon Radio Limited
Churchill House, Cambridge Business Park, Cowley Road, Cambridge, CB4 0WZ, UK



Member of the CSR plc group of companies. CSR plc registered in England and Wales, registered number 4187346, registered office Churchill House, Cambridge Business Park, Cowley Road, Cambridge, CB4 0WZ, United Kingdom
More information can be found at www.csr.com. Follow CSR on Twitter at http://twitter.com/CSR_PLC and read our blog at www.csr.com/blog


^ permalink raw reply	[flat|nested] 6+ messages in thread

end of thread, other threads:[~2011-10-25  9:39 UTC | newest]

Thread overview: 6+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2011-10-23 16:27 [PATCH] better systemctl completion foudfou
2011-10-24  9:01 ` Peter Stephenson
2011-10-24 11:19   ` foudfou
2011-10-24 11:27     ` Peter Stephenson
2011-10-24 18:45       ` Frank Terbeck
2011-10-25  9:00         ` Peter Stephenson

Code repositories for project(s) associated with this public inbox

	https://git.vuxu.org/mirror/zsh/

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).