Make tests usable with path with spaces
This is somewhat subtle:
The #RUN line in a littlecheck file will be run by a posix shell,
which means the substitutions will also be mangled by it.
Now, we *have* shell-quoted them, but unfortunately what we need is to
quote them for inside a pre-existing layer of quotes, e.g.
# RUN: fish -C 'set -g fish %fish'
here, %fish can't be replaced with `'path with spaces/fish'`, because
that ends up as
# RUN: fish -C 'set -g fish 'path with spaces/fish''
which is just broken.
So instead, we pass it as a variable to that fish:
# RUN: fish=%fish fish...
In addition, we need to not mangle the arguments in our test_driver.
For that, because we insist on posix shell, which has only one array,
and we source a file, we *need* to stop having that file use
arguments.
Which is okay - test_env.sh could previously be used to start a test,
and now it no longer can because that is test_*driver*.sh's job.
For the interactive tests, it's slightly different:
pexpect.spawn(foo) is sensitive to shell metacharacters like space.
So we shell-quote it.
But if you pass any args to pexpect.spawn, it no longer uses a shell,
and so we cannot shell-quote it.
There could be a better way to fix this?
2024-12-27 19:54:43 +00:00
#RUN: fish=%fish %fish %s
2020-01-13 19:34:22 +00:00
function complete_test_alpha1
echo $argv
end
2018-02-27 22:06:39 +00:00
2018-02-27 03:21:46 +00:00
complete -c complete_test_alpha1 --no-files -a '(commandline)'
complete -c complete_test_alpha2 --no-files -w 'complete_test_alpha1 extra1'
complete -c complete_test_alpha3 --no-files -w 'complete_test_alpha2 extra2'
complete -C 'complete_test_alpha1 arg1 '
2021-03-23 07:12:06 +00:00
# CHECK: complete_test_alpha1 arg1
2022-01-15 14:57:29 +00:00
complete --escape -C 'complete_test_alpha1 arg1 '
2024-04-12 23:59:07 +00:00
# CHECK: 'complete_test_alpha1 arg1 '
2018-02-27 03:21:46 +00:00
complete -C 'complete_test_alpha2 arg2 '
2021-03-23 07:12:06 +00:00
# CHECK: complete_test_alpha1 extra1 arg2
2018-02-27 03:21:46 +00:00
complete -C 'complete_test_alpha3 arg3 '
2021-03-23 07:12:06 +00:00
# CHECK: complete_test_alpha1 extra1 extra2 arg3
2019-04-26 12:54:51 +00:00
# Works even with the argument as a separate token.
complete -C 'complete_test_alpha3 arg3 '
2021-03-23 07:12:06 +00:00
# CHECK: complete_test_alpha1 extra1 extra2 arg3
2018-02-27 22:06:39 +00:00
alias myalias1 'complete_test_alpha1 arg1'
alias myalias2 = 'complete_test_alpha1 arg2'
myalias1 call1
myalias2 call2
2019-07-03 10:57:23 +00:00
# CHECK: arg1 call1
# CHECK: arg2 call2
2018-02-27 22:06:39 +00:00
complete -C 'myalias1 call3 '
2019-01-17 15:43:24 +00:00
complete -C 'myalias2 call3 '
2021-03-23 07:12:06 +00:00
# CHECK: complete_test_alpha1 arg1 call3
# CHECK: complete_test_alpha1 arg2 call3
2019-01-17 15:43:24 +00:00
# Ensure that commands can't wrap themselves - if this did,
# the completion would be executed a bunch of times.
2020-01-13 19:34:22 +00:00
function t --wraps t
echo t
end
2019-01-17 15:43:24 +00:00
complete -c t -fa '(t)'
complete -C 't '
2019-07-03 10:57:23 +00:00
# CHECK: t
2019-05-30 17:12:49 +00:00
2024-12-27 19:02:11 +00:00
touch test.fish
2019-05-30 17:12:49 +00:00
# Ensure file completion happens even though it was disabled above.
complete -c t -l fileoption -rF
2019-07-03 10:57:23 +00:00
# Only match one file because I don't want to touch this any time we add a test file.
2020-03-22 00:46:56 +00:00
complete -C 't --fileoption ' | string match test.fish
# CHECK: test.fish
2019-09-19 22:01:09 +00:00
2023-02-14 16:06:11 +00:00
# See that an empty command gets files
complete -C '"" t' | string match test.fish
# CHECK: test.fish
2019-09-19 22:01:09 +00:00
# Make sure bare `complete` is reasonable,
complete -p '/complete test/beta1' -d 'desc, desc' -sZ
complete -c 'complete test beta2' -r -d 'desc \' desc2 [ ' -a ' foo bar'
2020-03-09 18:36:12 +00:00
complete -c complete_test_beta2 -x -n false -A -o test
2019-09-19 22:01:09 +00:00
complete
2020-10-10 09:54:52 +00:00
# CHECK: complete --no-files complete_test_alpha1 -a '(commandline)'
# CHECK: complete --no-files complete_test_alpha2
# CHECK: complete --no-files complete_test_alpha3
# CHECK: complete --force-files t -l fileoption
# CHECK: complete --no-files t -a '(t)'
2019-09-19 22:01:09 +00:00
# CHECK: complete -p '/complete test/beta1' -s Z -d 'desc, desc'
2024-04-12 23:00:44 +00:00
# CHECK: complete --require-parameter 'complete test beta2' -d "desc ' desc2 [" -a 'foo bar'
2020-10-10 09:54:52 +00:00
# CHECK: complete --exclusive complete_test_beta2 -o test -n false
2019-09-19 22:44:15 +00:00
# CHECK: complete {{.*}}
# CHECK: complete {{.*}}
# CHECK: complete {{.*}}
# CHECK: complete {{.*}}
2019-10-03 05:55:52 +00:00
# Recursive calls to complete (see #3474)
complete -c complete_test_recurse1 -xa '(echo recursing 1>&2; complete -C"complete_test_recurse1 ")'
2020-09-26 12:13:56 +00:00
LANG = C complete -C 'complete_test_recurse1 '
2019-10-03 05:55:52 +00:00
# CHECKERR: recursing
2019-10-03 05:58:04 +00:00
# CHECKERR: recursing
# CHECKERR: recursing
# CHECKERR: recursing
# CHECKERR: recursing
# CHECKERR: recursing
# CHECKERR: recursing
# CHECKERR: recursing
# CHECKERR: recursing
# CHECKERR: recursing
# CHECKERR: recursing
# CHECKERR: recursing
# CHECKERR: recursing
# CHECKERR: recursing
# CHECKERR: recursing
# CHECKERR: recursing
# CHECKERR: recursing
# CHECKERR: recursing
# CHECKERR: recursing
# CHECKERR: recursing
# CHECKERR: recursing
# CHECKERR: recursing
# CHECKERR: recursing
# CHECKERR: recursing
2020-11-07 21:48:13 +00:00
# CHECKERR: error: completion reached maximum recursion depth, possible cycle?
Completion: complete argument to last of a group of short options
Consider a group of short options, like -xzPARAM, where x and z are options and z takes an argument.
This commit enables completion of the argument to the last option (z), both within the same
token (-xzP) or in the next one (-xz P).
complete -C'-xz' will complete only parameters to z.
complete -C'-xz ' will complete only parameters to z if z requires a parameter
otherwise, it will also complete non-option parameters
To do so this implements a heuristic to differentiate such strings from single long options. To
detect whether our token contains some short options, we only require the first character after the
dash (here x) to be an option. Previously, all characters had to be short options. The last option
in our example is z. Everything after the last option is assumed to be a parameter to the last
option.
Assume there is also a single long option -x-foo, then complete -C'-x' will suggest both -x-foo and
-xy. However, when the single option x requires an argument, this will not suggest -x-foo.
However, I assume this will almost never happen in practise since completions very rarely mix
short and single long options.
Fixes #332
2019-09-10 12:12:56 +00:00
# short options
complete -c foo -f -a non-option-argument
complete -c foo -f --short-option x
2020-03-09 18:36:12 +00:00
complete -c foo -f --short-option y -a ARGY
complete -c foo -f --short-option z -a ARGZ -r
Completion: complete argument to last of a group of short options
Consider a group of short options, like -xzPARAM, where x and z are options and z takes an argument.
This commit enables completion of the argument to the last option (z), both within the same
token (-xzP) or in the next one (-xz P).
complete -C'-xz' will complete only parameters to z.
complete -C'-xz ' will complete only parameters to z if z requires a parameter
otherwise, it will also complete non-option parameters
To do so this implements a heuristic to differentiate such strings from single long options. To
detect whether our token contains some short options, we only require the first character after the
dash (here x) to be an option. Previously, all characters had to be short options. The last option
in our example is z. Everything after the last option is assumed to be a parameter to the last
option.
Assume there is also a single long option -x-foo, then complete -C'-x' will suggest both -x-foo and
-xy. However, when the single option x requires an argument, this will not suggest -x-foo.
However, I assume this will almost never happen in practise since completions very rarely mix
short and single long options.
Fixes #332
2019-09-10 12:12:56 +00:00
complete -c foo -f --old-option single-long-ending-in-z
complete -c foo -f --old-option x-single-long
complete -c foo -f --old-option y-single-long
complete -c foo -f --old-option z-single-long
2020-03-09 18:36:12 +00:00
complete -c foo -f --long-option x-long -a ARGLONG
Completion: complete argument to last of a group of short options
Consider a group of short options, like -xzPARAM, where x and z are options and z takes an argument.
This commit enables completion of the argument to the last option (z), both within the same
token (-xzP) or in the next one (-xz P).
complete -C'-xz' will complete only parameters to z.
complete -C'-xz ' will complete only parameters to z if z requires a parameter
otherwise, it will also complete non-option parameters
To do so this implements a heuristic to differentiate such strings from single long options. To
detect whether our token contains some short options, we only require the first character after the
dash (here x) to be an option. Previously, all characters had to be short options. The last option
in our example is z. Everything after the last option is assumed to be a parameter to the last
option.
Assume there is also a single long option -x-foo, then complete -C'-x' will suggest both -x-foo and
-xy. However, when the single option x requires an argument, this will not suggest -x-foo.
However, I assume this will almost never happen in practise since completions very rarely mix
short and single long options.
Fixes #332
2019-09-10 12:12:56 +00:00
# Make sure that arguments of concatenated short options are expanded (#332)
complete -C 'foo -xy'
# CHECK: -xyARGY
# CHECK: -xyz
# A required parameter means we don't want more short options.
complete -C 'foo -yz'
# CHECK: -yzARGZ
# Required parameter with space: complete only the parameter (no non-option arguments).
complete -C 'foo -xz '
# CHECK: ARGZ
# Optional parameter with space: complete only non-option arguments.
complete -C 'foo -xy '
# CHECK: non-option-argument
complete -C 'foo -single-long-ending-in-z'
# CHECK: -single-long-ending-in-z
complete -C 'foo -single-long-ending-in-z '
# CHECK: non-option-argument
# CHECK: -x-single-long
complete -C 'foo -x' | string match -- -x-single-long
# CHECK: -y-single-long
complete -C 'foo -y' | string match -- -y-single-long
# This does NOT suggest -z-single-long, but will probably not occur in practise.
# CHECK: -zARGZ
complete -C 'foo -z'
2024-12-14 07:30:34 +00:00
function foo2; end
complete -c foo2 -s s -l long -xa "hello-world goodbye-friend"
complete -C "foo2 -sfrie"
# CHECK: -sgoodbye-friend
complete -C "foo2 --long=frien"
# CHECK: --long=goodbye-friend
2019-10-29 12:32:26 +00:00
# Builtins (with subcommands; #2705)
2024-01-22 06:42:45 +00:00
complete -c complete_test_subcommand -n 'test (commandline -xp)[1] = complete_test_subcommand' -xa ok
2019-10-29 12:32:26 +00:00
complete -C 'not complete_test_subcommand '
# CHECK: ok
complete -C 'echo; and complete_test_subcommand '
# CHECK: ok
complete -C 'or complete_test_subcommand '
# CHECK: ok
complete -C 'echo && command complete_test_subcommand '
# CHECK: ok
complete -C 'echo || exec complete_test_subcommand '
# CHECK: ok
complete -C 'echo | builtin complete_test_subcommand '
# CHECK: ok
complete -C 'echo & complete_test_subcommand '
# CHECK: ok
complete -C 'if while begin begin complete_test_subcommand '
# CHECK: ok
complete -C 'for _ in ' | string collect > & - && echo completed some files
# CHECK: completed some files
# function; #5415
complete -C 'function : --arg'
# CHECK: --argument-names {{.*}}
2019-12-16 11:37:41 +00:00
complete -C 'echo > /' | string length -q && echo ok
# CHECK: ok
2020-01-08 16:53:46 +00:00
function some_function
2020-01-13 19:34:22 +00:00
echo line1
echo line2
2020-01-08 16:53:46 +00:00
end
complete -c complete_test_function_desc -xa '(complete -Csome_function)'
complete -C 'complete_test_function_desc ' | count
# CHECK: 1
2020-01-18 19:20:22 +00:00
complete -c prev-arg-variable -f
complete -C 'prev-arg-variable $HOME '
2020-02-08 09:55:59 +00:00
# Regression test for issue #3129. In previous versions these statements would
# cause an `assert()` to fire thus killing the shell.
complete -c pkill -o ''
#CHECKERR: complete: -o requires a non-empty string
complete -c pkill -l ''
#CHECKERR: complete: -l requires a non-empty string
complete -c pkill -s ''
#CHECKERR: complete: -s requires a non-empty string
# Test that conditions that add or remove completions don't deadlock, etc.
# We actually encountered some case that was effectively like this (Issue 2 in github)
complete --command AAAA -l abcd --condition 'complete -c AAAA -l efgh'
echo "AAAA:"
complete -C 'AAAA -' | sort
#CHECK: AAAA:
#CHECK: --abcd
echo "AAAA:"
complete -C 'AAAA -' | sort
#CHECK: AAAA:
#CHECK: --abcd
#CHECK: --efgh
complete --command BBBB -l abcd --condition 'complete -e --command BBBB -l abcd'
echo "BBBB:"
complete -C 'BBBB -'
#CHECK: BBBB:
#CHECK: --abcd
echo "BBBB:"
complete -C 'BBBB -'
#CHECK: BBBB:
2021-03-23 07:12:06 +00:00
#CHECK:
2020-02-08 09:55:59 +00:00
# Test that erasing completions works correctly
echo
function sort
# GNU sort is really stupid, a non-C locale seems to make it assume --dictionary-order
# If I wanted --dictionary-order, I would have specified --dictionary-order!
env LC_ALL = C sort $argv
end
complete -c CCCC -l bar
complete -c CCCC -l baz
complete -c CCCC -o bar
complete -c CCCC -o foo
complete -c CCCC -s a
complete -c CCCC -s b
echo "CCCC:"
complete -C 'CCCC -' | sort
complete -c CCCC -l bar -e
#CHECK: CCCC:
#CHECK: --bar
#CHECK: --baz
#CHECK: -a
#CHECK: -b
#CHECK: -bar
#CHECK: -foo
echo "CCCC:"
complete -C 'CCCC -' | sort
complete -c CCCC -o foo -e
#CHECK: CCCC:
#CHECK: --baz
#CHECK: -a
#CHECK: -b
#CHECK: -bar
#CHECK: -foo
echo "CCCC:"
complete -C 'CCCC -' | sort
complete -c CCCC -s a -e
#CHECK: CCCC:
#CHECK: --baz
#CHECK: -a
#CHECK: -b
#CHECK: -bar
echo "CCCC:"
complete -C 'CCCC -' | sort
complete -c CCCC -e
#CHECK: CCCC:
#CHECK: --baz
#CHECK: -b
#CHECK: -bar
echo "CCCC:"
complete -C 'CCCC -' | sort
#CHECK: CCCC:
echo "Test that -- suppresses option completions"
complete -c TestDoubleDash -l TestDoubleDashOption
complete -C 'TestDoubleDash -' | sort
#CHECK: Test that -- suppresses option completions
#CHECK: --TestDoubleDashOption
echo "Expect no output:" ( complete -C 'TestDoubleDash -- -' | sort )
#CHECK: Expect no output:
# fish seems to have always handled "exclusive" options strangely
# It seems to treat them the same as "old-style" (single-dash) long options
echo "Testing exclusive options"
#CHECK: Testing exclusive options
complete -c TestExclusive -x -s Q
complete -c TestExclusive -x -s W
complete -c TestExclusive -s A
echo "Expect -A -Q -W:" ( complete -C 'TestExclusive -' | sort | string join ' ' )
#CHECK: Expect -A -Q -W: -A -Q -W
echo "Expect -AQ -AW:" ( complete -C 'TestExclusive -A' | sort | string join ' ' )
#CHECK: Expect -AQ -AW: -AQ -AW
echo "Expect no output 1:" ( complete -C 'TestExclusive -Q' )
#CHECK: Expect no output 1:
echo "Expect no output 2:" ( complete -C 'TestExclusive -W' )
#CHECK: Expect no output 2:
# Test for optional arguments, like cp's --backup
complete -c TestOptionalArgument -l backup -f -a 'none all simple'
echo "Expect --backup --backup=:" ( complete -C 'TestOptionalArgument -' | sort | string join ' ' )
#CHECK: Expect --backup --backup=: --backup --backup=
echo "Expect --backup=all --backup=none --backup=simple:" ( complete -C 'TestOptionalArgument --backup=' | sort | string join ' ' )
#CHECK: Expect --backup=all --backup=none --backup=simple: --backup=all --backup=none --backup=simple
# Test that directory completions work correctly
2020-03-09 18:36:12 +00:00
if begin
rm -rf test6.tmp.dir; and mkdir test6.tmp.dir
end
2020-02-08 09:55:59 +00:00
pushd test6.tmp.dir
2021-03-23 07:12:06 +00:00
# The "incorrect implicit cd from PATH" fails if mktemp returns an absolute path and
# `realpath --relative-to` is not available on macOS.
# set dir (realpath --relative-to="$PWD" (mktemp -d XXXXXXXX))
set dir ( basename ( mktemp -d XXXXXXXX) )
mkdir -p $dir
2021-11-04 11:18:54 +00:00
if complete -C $dir | string match -r " ^ $dir /.*dir " > /dev/null
2020-02-08 09:55:59 +00:00
echo "implicit cd complete works"
else
echo "no implicit cd complete"
end
#CHECK: implicit cd complete works
2021-11-04 11:18:54 +00:00
if complete -C " command $dir " | string match -r " ^ $dir /.*dir " > /dev/null
2020-03-24 20:55:38 +00:00
echo "implicit cd complete after 'command'"
2020-02-08 09:55:59 +00:00
else
echo "no implicit cd complete after 'command'"
end
2020-03-24 20:55:38 +00:00
#CHECK: implicit cd complete after 'command'
2020-02-08 09:55:59 +00:00
popd
if begin
set -l PATH $PWD /test6.tmp.dir $PATH 2 > /dev/null
2021-11-04 11:18:54 +00:00
complete -C $dir | string match -r " ^ $dir /.*dir " > /dev/null
2020-02-08 09:55:59 +00:00
end
echo "incorrect implicit cd from PATH"
else
echo "PATH does not cause incorrect implicit cd"
end
#CHECK: PATH does not cause incorrect implicit cd
rm -rf test6.tmp.dir
else
echo "error: could not create temp environment" > & 2
end
# Test command expansion with parened PATHs (#952)
begin
2020-03-09 18:36:12 +00:00
set -l parened_path $PWD /'test6.tmp2.(paren).dir'
set -l parened_subpath $parened_path /subdir
if not begin
rm -rf $parened_path
and mkdir $parened_path
and mkdir $parened_subpath
and ln -s /bin/ls $parened_path /'__test6_(paren)_command'
and ln -s /bin/ls $parened_subpath /'__test6_subdir_(paren)_command'
end
echo "error: could not create command expansion temp environment" > & 2
end
# Verify that we can expand commands when PATH has parens
set -l PATH $parened_path $PATH
set -l completed ( complete -C__test6_ | cut -f 1 -d \t )
if test " $completed " = '__test6_(paren)_command'
echo "Command completion with parened PATHs test passed"
else
echo " Command completion with parened PATHs test failed. Expected __test6_(paren)_command, got $completed " > & 2
end
#CHECK: Command completion with parened PATHs test passed
# Verify that commands with intermediate slashes do NOT expand with respect to PATH
set -l completed ( complete -Csubdir /__test6_subdir)
if test -z " $completed "
echo "Command completion with intermediate slashes passed"
else
echo " Command completion with intermediate slashes: should output nothing, instead got $completed " > & 2
end
#CHECK: Command completion with intermediate slashes passed
rm -rf $parened_path
2020-02-08 09:55:59 +00:00
end
2020-09-09 16:03:39 +00:00
# This should only list the completions for `banana`
complete -c banana -a '1 2 3'
complete -c banana
2020-10-10 09:54:52 +00:00
#CHECK: complete banana -a '1 2 3'
2020-09-09 16:37:46 +00:00
# "-c" is optional
complete banana -a bar
complete banana
2020-10-10 09:54:52 +00:00
#CHECK: complete banana -a bar
#CHECK: complete banana -a '1 2 3'
2020-09-09 16:37:46 +00:00
# "-a" ain't
complete banana bar
2021-11-04 05:52:17 +00:00
#CHECKERR: complete: too many arguments
2020-09-09 16:37:46 +00:00
#CHECKERR:
#CHECKERR: {{.*}}checks/complete.fish (line {{\d+}}):
#CHECKERR: complete banana bar
#CHECKERR: ^
#CHECKERR:
#CHECKERR: (Type 'help complete' for related documentation)
2020-10-10 09:54:52 +00:00
# Multiple commands can be specified, in that case "-c" (or "-p") is mandatory.
complete -c kapstachelbeere -c physalis -a arg
complete -c kapstachelbeere -c physalis
# CHECK: complete kapstachelbeere -a arg
# CHECK: complete physalis -a arg
2020-10-10 10:35:20 +00:00
set -l dir ( mktemp -d )
echo > $dir /target
complete -C ': $dir/'
# CHECK: $dir/target
rm $dir /target
2021-04-10 19:56:17 +00:00
cd $dir
touch yummyinmytummy
complete -c fudge -f
complete -c fudge -n '__fish_seen_subcommand_from eat' -F
complete -C 'fudge eat yummyin'
# CHECK: yummyinmytummy
2022-11-12 21:00:36 +00:00
complete -C "echo no commpletion inside comment # "
2021-04-10 19:56:17 +00:00
cd -
rm -r $dir
2021-05-11 21:02:15 +00:00
set -l dir ( mktemp -d )
cd $dir
: > command -not-in-path
chmod +x command -not-in-path
complete -p $PWD /command -not-in-path -xa relative-path
complete -C './command-not-in-path '
# CHECK: relative-path
2021-11-27 08:56:35 +00:00
# Expand variables and tildes in command.
complete -C '$PWD/command-not-in-path '
# CHECK: relative-path
HOME = $PWD complete -C '~/command-not-in-path '
# CHECK: relative-path
2021-05-11 21:02:15 +00:00
# Non-canonical command path
mkdir -p subdir
: > subdir/command -in-subdir
chmod +x subdir/command -in-subdir
complete -p " $PWD /subdir/command-in-subdir " -xa custom-completions
complete -C './subdir/../subdir/command-in-subdir '
# CHECK: custom-completions
# Relative $PATH
begin
set -lx PATH subdir $PATH
complete -C 'command-in-subdir '
# CHECK: custom-completions
end
2021-08-30 15:07:11 +00:00
cd -
2021-05-11 21:02:15 +00:00
rm -r $dir
2021-11-27 08:56:35 +00:00
# Expand variables and tildes in command.
complete cat -xa +pet
set -l path_to_cat ( command -v cat)
complete -C '$path_to_cat '
# CHECK: +pet
HOME = $path_to_cat /.. complete -C '~/cat '
# CHECK: +pet
# Do not expand command substitutions.
complete -C '(echo cat) ' | string match +pet
# Give up if we expand to multiple arguments (we'd need to handle the arguments).
complete -C '{cat,arg1,arg2} ' | string match +pet
# Don't expand wildcards though we could.
complete -C '$path_to_cat* ' | string match +pet
# Also expand wrap targets.
function crookshanks --wraps '$path_to_cat'
end
complete -C 'crookshanks '
# CHECK: +pet
2022-01-26 14:22:18 +00:00
# Custom completion works with variable overrides.
2024-01-22 06:42:45 +00:00
complete cmd_with_fancy_completion -xa '(commandline -xpc | count)'
2022-01-26 14:22:18 +00:00
complete -C "a=1 b=2 cmd_with_fancy_completion "
# CHECK: 1
complete -C "a=1 b=2 cmd_with_fancy_completion 1 "
# CHECK: 2
2022-05-26 12:15:28 +00:00
complete -c thing -x -F
# CHECKERR: complete: invalid option combination, '--exclusive' and '--force-files'
2022-05-20 17:02:42 +00:00
# Multiple conditions
complete -f -c shot
2024-01-22 06:42:45 +00:00
complete -fc shot -n 'test (count (commandline -xpc) -eq 1' -n 'test (commandline -xpc)[-1] = shot' -a 'through'
# CHECKERR: complete: -n 'test (count (commandline -xpc) -eq 1': Unexpected end of string, expecting ')'
# CHECKERR: test (count (commandline -xpc) -eq 1
2022-05-20 17:02:42 +00:00
# CHECKERR: ^
2024-01-22 06:42:45 +00:00
complete -fc shot -n 'test (count (commandline -xpc)) -eq 1' -n 'test (commandline -xpc)[-1] = shot' -a 'through'
complete -fc shot -n 'test (count (commandline -xpc)) -eq 2' -n 'test (commandline -xpc)[-1] = through' -a 'the'
complete -fc shot -n 'test (count (commandline -xpc)) -eq 3' -n 'test (commandline -xpc)[-1] = the' -a 'heart'
complete -fc shot -n 'test (count (commandline -xpc)) -eq 4' -n 'test (commandline -xpc)[-1] = heart' -a 'and'
complete -fc shot -n 'test (count (commandline -xpc)) -eq 5' -n 'test (commandline -xpc)[-1] = and' -a "you\'re"
complete -fc shot -n 'test (count (commandline -xpc)) -eq 6' -n 'test (commandline -xpc)[-1] = "you\' re" ' -a 'to'
complete -fc shot -n 'test (count (commandline -xpc)) -eq 7' -n 'test (commandline -xpc)[-1] = to' -a 'blame'
2022-05-20 17:02:42 +00:00
complete -C "shot "
# CHECK: through
complete -C "shot through "
# CHECK: the
# See that conditions after a failing one aren't executed.
set -g oops 0
complete -fc oooops
complete -fc oooops -n true -n true -n true -n 'false' -n 'set -g oops 1' -a oops
complete -C 'oooops '
echo $oops
# CHECK: 0
complete -fc oooops -n 'true' -n 'set -g oops 1' -a oops
complete -C 'oooops '
# CHECK: oops
echo $oops
# CHECK: 1
2022-08-11 15:05:32 +00:00
# See that we load completions only if the command exists in $PATH,
# as a workaround for #3117.
# We need a completion script that runs the command,
# and prints something simple, and isn't already used above.
#
# Currently, tr fits the bill (it does `tr --version` to check for GNUisms)
begin
$fish -c "complete -C'tr -'" | string match -- '-d*'
# CHECK: -d{{\t.*}}
end
set -l tmpdir ( mktemp -d )
cd $tmpdir
begin
touch tr
chmod +x tr
set -l PATH
complete -C './tr -' | string match -- -d
or echo No match for relative
# CHECK: No match for relative
complete -c tr | string length -q
or echo Empty completions
# CHECK: Empty completions
end
2022-09-12 22:33:29 +00:00
rm -$f $tmpdir /*
# Leading dots are not completed for default file completion,
# but may be for custom command (e.g. git add).
function dotty
end
function notty
end
complete -c dotty --no-files -a '(echo .a*)'
touch .abc .def
complete -C 'notty '
echo "Should be nothing"
# CHECK: Should be nothing
complete -C 'dotty '
# CHECK: .abc
2022-08-11 15:05:32 +00:00
rm -r $tmpdir
2023-09-10 16:11:23 +00:00
2023-09-11 14:52:38 +00:00
complete -C 'complete --command=mktemp' | string replace -rf '=mktemp\t.*' '=mktemp'
2023-09-10 16:11:23 +00:00
# (one "--command=" is okay, we used to get "--command=--command="
2023-09-11 14:52:38 +00:00
# CHECK: --command=mktemp
2024-01-06 07:45:33 +00:00
## Test token expansion in commandline -x
complete complete_make -f -a '( argparse C/directory = -- ( commandline -xpc ) [ 2 .. ] ;
echo Completing targets in directory $_flag_C ) '
var = path/to complete -C 'complete_make -C "$var/build-directory" '
# CHECK: Completing targets in directory path/to/build-directory
var1 = path complete -C 'var2=to complete_make -C "$var1/$var2/other-build-directory" '
# CHECK: Completing targets in directory path/to/other-build-directory
complete complete_existing_argument -f -a '(commandline -xpc)[2..]'
var = a_value complete -C 'complete_existing_argument "1 2" $var \' quoted ( foo bar) \' unquoted( baz qux) '
# CHECK: 1 2
# CHECK: a_value
# CHECK: quoted (foo bar)
# CHECK: unquoted(baz qux)
complete complete_first_argument_and_count -f -a '( set -l args ( commandline -xpc ) [ 2 .. ]
echo ( count $args ) arguments, first argument is $args [ 1 ] ) '
list = arg( seq 10 ) begin
complete -C 'complete_first_argument_and_count $list$list '
# CHECK: 100 arguments, first argument is arg1arg1
complete -C 'complete_first_argument_and_count $list$list$list '
# CHECK: 1 arguments, first argument is $list$list$list
end
## Test commandline --tokens-raw
complete complete_raw_tokens -f -ka '(commandline --tokens-raw)'
complete -C 'complete_raw_tokens "foo" bar\\ baz (qux) '
# CHECK: complete_raw_tokens
# CHECK: "foo"
# CHECK: bar\ baz
# CHECK: (qux)
## Test deprecated commandline -o
complete complete_unescaped_tokens -f -ka '(commandline -o)'
complete -C 'complete_unescaped_tokens "foo" bar\\ baz (qux) '
# CHECK: complete_unescaped_tokens
# CHECK: foo
# CHECK: bar baz
# CHECK: (qux)
2024-01-27 16:06:10 +00:00
## Fuzzy completion of options
complete complete_long_option -f -l some-long-option
complete -C 'complete_long_option --slo'
# CHECK: --some-long-option
complete complete_long_option -f -o an-old-option
complete -C 'complete_long_option -ao'
# CHECK: -an-old-option
2024-01-30 06:58:47 +00:00
# But only if the user typed a dash
complete -C 'complete_long_option lo'
2024-06-28 01:45:38 +00:00
# Check that descriptions are correctly generated for commands.
2024-06-28 02:32:11 +00:00
# Override __fish_describe_command to prevent missing man pages or broken __fish_apropos on macOS
# from failing this test. (TODO: Test the latter separately.)
function __fish_describe_command
echo -e "whereis\twhere is it"
echo -e "whoami\twho am i"
echo -e "which\which is it"
end
2024-06-28 01:45:38 +00:00
test ( count ( complete -C "wh" | string match -rv "\tcommand|^while" ) ) -gt 0 && echo "found" || echo "fail"
# CHECK: found
2024-08-08 19:15:32 +00:00
set -l commands check search show
complete -c testcommand -n " not __fish_seen_subcommand_from $commands " -a 'check\t"Check the frobnicator" search\t"Search for frobs" show\t"Show all frobs"'
complete -C 'testcommand '
# CHECK: check{{\t}}Check the frobnicator
# CHECK: search{{\t}}Search for frobs
# CHECK: show{{\t}}Show all frobs