mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-15 06:24:01 +00:00
60ce10ad84
Fixes #5692.
71 lines
2.5 KiB
Fish
71 lines
2.5 KiB
Fish
function eval -S -d "Evaluate parameters as a command"
|
||
# keep a copy of the previous $status and use restore_status
|
||
# to preserve the status in case the block that is evaluated
|
||
# does not modify the status itself.
|
||
set -l status_copy $status
|
||
function __fish_restore_status
|
||
return $argv[1]
|
||
end
|
||
|
||
if not set -q argv[2]
|
||
# like most builtins, we only check for -h/--help
|
||
# if we only have a single argument
|
||
switch "$argv[1]"
|
||
case -h --help
|
||
__fish_print_help eval
|
||
return 0
|
||
end
|
||
end
|
||
|
||
if not string length -q -- $argv
|
||
# If the argument is empty, eval should return 0 for compatibility with other shells.
|
||
# See #5692.
|
||
return 0
|
||
end
|
||
|
||
# If we are in an interactive shell, eval should enable full
|
||
# job control since it should behave like the real code was
|
||
# executed. If we don't do this, commands that expect to be
|
||
# used interactively, like less, wont work using eval.
|
||
|
||
set -l mode
|
||
if status --is-interactive-job-control
|
||
set mode interactive
|
||
else
|
||
if status --is-full-job-control
|
||
set mode full
|
||
else
|
||
set mode none
|
||
end
|
||
end
|
||
if status --is-interactive
|
||
status --job-control full
|
||
end
|
||
__fish_restore_status $status_copy
|
||
|
||
# To eval 'foo', we construct a block "begin ; foo; end <&3 3<&-"
|
||
# Note the redirections are also within the quotes.
|
||
#
|
||
# We then pipe this to 'source 3<&0’.
|
||
#
|
||
# You might expect that the dup2(3, stdin) should overwrite stdin,
|
||
# and therefore prevent 'source' from reading the piped-in block. This doesn't happen
|
||
# because when you pipe to a builtin, we don't overwrite stdin with the read end
|
||
# of the block; instead we set a separate fd in a variable 'builtin_stdin', which is
|
||
# what it reads from. So builtins are magic in that, in pipes, their stdin
|
||
# is not fd 0.
|
||
#
|
||
# ‘source’ does not apply the redirections to itself. Instead it saves them and passes
|
||
# them as block-level redirections to parser.eval(). Ultimately the eval’d code sees
|
||
# the following redirections (in the following order):
|
||
# dup2 0 -> 3
|
||
# dup2 pipe -> 0
|
||
# dup2 3 -> 0
|
||
# where the pipe is the pipe we get from piping ‘echo’ to ‘source’. Thus the redirection
|
||
# effectively makes stdin fd0, instead of the thing that was piped to ‘source’
|
||
echo "begin; $argv "\n" ;end <&3 3<&-" | source 3<&0
|
||
set -l res $status
|
||
|
||
status --job-control $mode
|
||
return $res
|
||
end
|