Complgen
Declarative bash/fish/zsh/pwsh completions without writing shell scripts!
Install / Use
/learn @adaszko/ComplgenREADME
Value Proposition
complgen compiles man-page-like grammars into standalone completion scripts. One grammar file compiles into
several different scripts for most popular shells, freeing you from having to reimplement and maintain the
same (state-machine) logic for each shell separately. complgen grammars are declarative and thus much
easier to maintain than long completion scripts. See examples.
Demo

Usage
Bash
<details>$ cat hello.usage
hello --color=(always | never | auto);
$ complgen --bash hello.bash hello.usage
$ source hello.bash
$ hello --color=<TAB>
always auto never
</details>
Fish
<details>$ cat hello.usage
hello --color=(always | never | auto);
$ complgen --fish hello.fish hello.usage
$ source hello.fish
$ hello --color=<TAB>
--color=always --color=auto --color=never
</details>
Zsh
<details>% cat hello.usage
hello --color=(always | never | auto);
% complgen --zsh _hello hello.usage
% source _hello
% hello --color=<TAB>
always
auto
never
💡 Note: Under ZSH, source isn't strictly necessary — it is enough to put the output file in one of the
directories listed in $fpath variable.
PowerShell
<details>PS> Get-Content hello.usage
hello --color=(always | never | auto);
PS> complgen --pwsh hello.ps1 hello.usage
PS> . ./hello.ps1
PS> hello --color=<TAB>
--color=always --color=auto --color=never
💡 Note: Requires PowerShell Core 7+ (pwsh). Add the script to your $PROFILE for persistent completions.
Installation
cargo install --git https://github.com/adaszko/complgen --tag v0.8.3 complgen
Or download a pre-built binary from the latest GitHub release.
Syntax
See the examples subdirectory.
complgen's grammar is based on compleat's one.
A grammar is a series of lines separated by a semicolon (;). Each line either represents a single variant
of invoking the completed command (e.g. git status; git log) or is a nonterminal definition (<FOO> = bar).
a bmatchesafollowed byb.a b | cmatches eithera borc(IOW: sequence binds stronger than alternative).[a]matches zero or one occurrences ofa.a...matches one or more occurrences ofa[a]...matches zero or more occurrences ofa.(aaa | bbb || ccc)showsaaaandbbbas candidates, andccconly when current input matches neitheraaanorbbb.||behaves exactly like|when matching, it differs only when offering completions.
Use parentheses to group patterns:
a (b | c)matchesafollowed by eitherborc.(a | b) ...matchesaorbfollowed by any number of additionalaorb.
<_> Nonterminal
The special nonterminal <_> matches any shell word and doesn't produce the "undefined nonterminal"
compilation warning so it's a good choice to use for program arguments where there isn't enough structure to
produce meaningful completions but matching should still continue at the next shell word.
Filename Completion
There's a small set of predefined nonterminals that are handled specially by complgen:
| Name | bash | fish | zsh | pwsh | Description |
|---------------|------|------|-----|------|-------------|
|<PATH> | ✅ | ✅ | ✅ | ✅ | file or directory path |
|<DIRECTORY> | ✅ | ✅ | ✅ | ✅ | directory path |
These can still be redefined in the grammar (<PATH> = ...), in which case their predefined meaning gets
overridden.
Completion Descriptions (fish/zsh/pwsh)
If a literal is immediately followed with a quoted string, it's going to appear as a hint to the user at completion time. E.g. the grammar:
grep --extended-regexp "PATTERNS are extended regular expressions" | --exclude "skip files that match GLOB";
results in something like this under fish, zsh, and PowerShell:
fish> grep --ex<TAB>
--exclude (skip files that match GLOB) --extended-regexp (PATTERNS are extended regular expressions)
Note that bash does not support showing descriptions. In PowerShell, descriptions appear as tooltips.
Sourcing Completions from External Commands
It is possible to produce completions based on an external command output:
cmd {{{ echo foo; echo bar; echo baz; echo quux }}};
bash$ cmd <TAB>
bar baz foo quux
Descriptions
Externals commands are also assumed to produce descriptions similar to those described in the section above. Their expected stdout format is a sequence of lines of the form
COMPLETION\tDESCRIPTION
For fish and zsh, the DESCRIPTION part will be presented to the user. Under bash, only the COMPLETION
part will be visible. All external commands nonetheless need to take care as to not produce superfluous
\t characters that may confuse the resulting shell scripts.
Target Shell-specific Behavior
External commands quickly degrade into necessitating shell-specific syntax. complgen provides support to conditionally choose the command based on the target shell.
To use an example: all shells are able to complete users present on the system although each has a different function for it:
cmd <USER>;
<USER@bash> = {{{ compgen -A user | sort | uniq }}}; # produce candidates on stdout under bash
<USER@fish> = {{{ __fish_complete_users }}}; # produce candidates on stdout under fish
<USER@zsh> = {{{ _users }}}; # produce candidates via compadd and friends under zsh
<USER@pwsh> = {{{ Get-LocalUser | ForEach-Object { $_.Name } }}}; # PowerShell
complgen will pick the right definition of <USER> depending on what you're compiling the grammar to.
Completion Within Words
It's possible to match not only entire shell words, but also within words, using largely the same grammar
syntax as for matching entire words, barring few spaces here and there. The most common application is to
handle option arguments (e.g. --option=ARGUMENT):
grep --color=(always | never | auto);
The same mechanism works for more complicated things:
strace -e <EXPR>;
<EXPR> = [<qualifier>=][!]<value>[,<value>]...;
<qualifier> = trace | read | write | fault;
<value> = %file | file | all;
The above grammar was pulled straight out of strace man
page, illustrating how complgen follows the de
facto standard man pages notation.
Control What Gets Shown First
If you do git <TAB> under the default completion that comes with git, you get something like:
bash$ git <TAB>
absorb bisect ci credential-gcloud fork help mv reflog revert show submodule whatchanged
ad blame citool describe format-patch init notes refs reword show-branch sw worktree
add br clang-format diff fsck instaweb prune remote rewrite sparse-checkout switch
am branch clean difftool gc lfs pull repack rm stage tag
amend bundle clone dlog gitk log push replace root stash touch
apply checkout co fetch grep maintenance range-diff request-pull scalar stash-all tree
archive cherry commit filter-repo gui merge rebase reset send-email stash-unstaged uncommit
backfill cherry-pick config forgit head mergetool recent restore shortlog status wdiff
So even though git accepts many global options, they don't show up here! If OTOH you do git --<TAB>
instead:
bash$ git --<TAB>
--bare --exec-path= --help --info-path --namespace= --no-replace-objects --version
--exec-path --git-dir= --html-path --man-path --no-pager --paginate --work-tree=
You get presented with options in place of subcommands. It's a useful mechanism that allows to assigning levels of priority to possible choices, surfacing the most frequently used ones.
Under complgen, the same effect is achieved via
fallbacks (||). An abridged version version of the above would look like below:
mygit (<SUBCOMMAND> || <OPTION>);
<SUBCOMMAND> = fetch | add | commit | push;
<OPTION> = --help | --version;
The effect:
bash$ mygit <TAB>
add commit fetch push
bash$ mygit --<TAB>
--help --version
In the 2nd case (mygit --<TAB>), the completion script first tries to match the input (--) against
<SUBCOMMAND> and failing that, proceeds to try matching the part on the right hand side of the fallback
operator ||, i.e. <OPTION>.
Note that || behaves exactly like the regular | in when it comes to matching. Where its behavior diverges
is with respect to completion. Where | variants would offer all alternative branches at once, || tries to
match variants from lef
