aboutsummaryrefslogtreecommitdiffhomepage
path: root/share/functions/eval.fish
blob: 1545540286bca06c2be656eb09deba3b45e1b21c (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
function eval -S -d "Evaluate parameters as a command"
	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 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

        # rfish: To eval 'foo', we construct a block "begin ; foo; end <&3 3<&-"
        # The 'eval2_inner' is a param to 'begin' itself; I believe it does nothing.
        # Note the redirections are also within the quotes.
        #
        # We then pipe this to 'source 3<&0' which dup2's 3 to stdin.
        #
        # 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.

	echo "begin; $argv "\n" ;end eval2_inner <&3 3<&-" | source 3<&0
	set -l res $status

	status --job-control $mode
	return $res
end