| 1 | #!/usr/bin/env bash |
| 2 | |
| 3 | #### Eval |
| 4 | eval "a=3" |
| 5 | echo $a |
| 6 | ## stdout: 3 |
| 7 | |
| 8 | #### Source |
| 9 | lib=$TMP/spec-test-lib.sh |
| 10 | echo 'LIBVAR=libvar' > $lib |
| 11 | . $lib # dash doesn't have source |
| 12 | echo $LIBVAR |
| 13 | ## stdout: libvar |
| 14 | |
| 15 | #### Source nonexistent |
| 16 | source /nonexistent/path |
| 17 | echo status=$? |
| 18 | ## stdout: status=1 |
| 19 | ## OK dash/zsh stdout: status=127 |
| 20 | |
| 21 | #### Source with no arguments |
| 22 | source |
| 23 | echo status=$? |
| 24 | ## stdout: status=2 |
| 25 | ## OK mksh/zsh stdout: status=1 |
| 26 | ## N-I dash stdout: status=127 |
| 27 | |
| 28 | #### Source with arguments |
| 29 | . spec/testdata/show-argv.sh foo bar # dash doesn't have source |
| 30 | ## STDOUT: |
| 31 | show-argv: foo bar |
| 32 | ## END |
| 33 | ## N-I dash STDOUT: |
| 34 | show-argv: |
| 35 | ## END |
| 36 | |
| 37 | #### Source from a function, mutating argv and defining a local var |
| 38 | f() { |
| 39 | . spec/testdata/source-argv.sh # no argv |
| 40 | . spec/testdata/source-argv.sh args to src # new argv |
| 41 | echo $@ |
| 42 | echo foo=$foo # defined in source-argv.sh |
| 43 | } |
| 44 | f args to func |
| 45 | echo foo=$foo # not defined |
| 46 | ## STDOUT: |
| 47 | source-argv: args to func |
| 48 | source-argv: args to src |
| 49 | to func |
| 50 | foo=foo_val |
| 51 | foo= |
| 52 | ## END |
| 53 | ## N-I dash STDOUT: |
| 54 | source-argv: args to func |
| 55 | source-argv: to func |
| 56 | func |
| 57 | foo=foo_val |
| 58 | foo= |
| 59 | ## END |
| 60 | |
| 61 | #### Source with syntax error |
| 62 | # TODO: We should probably use dash behavior of a fatal error. |
| 63 | # Although set-o errexit handles this. We don't want to break the invariant |
| 64 | # that a builtin like 'source' behaves like an external program. An external |
| 65 | # program can't halt the shell! |
| 66 | echo 'echo >' > $TMP/syntax-error.sh |
| 67 | . $TMP/syntax-error.sh |
| 68 | echo status=$? |
| 69 | ## stdout: status=2 |
| 70 | ## OK bash/mksh stdout: status=1 |
| 71 | ## OK zsh stdout: status=126 |
| 72 | ## OK dash stdout-json: "" |
| 73 | ## OK dash status: 2 |
| 74 | |
| 75 | #### Eval with syntax error |
| 76 | eval 'echo >' |
| 77 | echo status=$? |
| 78 | ## stdout: status=2 |
| 79 | ## OK bash/zsh stdout: status=1 |
| 80 | ## OK dash stdout-json: "" |
| 81 | ## OK dash status: 2 |
| 82 | ## OK mksh stdout-json: "" |
| 83 | ## OK mksh status: 1 |
| 84 | |
| 85 | #### Eval in does tilde expansion |
| 86 | |
| 87 | x="~" |
| 88 | eval y="$x" # scalar |
| 89 | test "$x" = "$y" || echo FALSE |
| 90 | [[ $x == /* ]] || echo FALSE # doesn't start with / |
| 91 | [[ $y == /* ]] && echo TRUE |
| 92 | |
| 93 | #argv "$x" "$y" |
| 94 | |
| 95 | ## STDOUT: |
| 96 | FALSE |
| 97 | FALSE |
| 98 | TRUE |
| 99 | ## END |
| 100 | ## BUG dash status: 127 |
| 101 | ## BUG dash stdout-json: "FALSE\n" |
| 102 | ## BUG mksh status: 1 |
| 103 | ## BUG mksh stdout-json: "FALSE\n" |
| 104 | |
| 105 | #### Eval in bash does tilde expansion in array |
| 106 | |
| 107 | # the "make" plugin in bash-completion relies on this? wtf? |
| 108 | x="~" |
| 109 | |
| 110 | # UPSTREAM CODE |
| 111 | |
| 112 | #eval array=( "$x" ) |
| 113 | |
| 114 | # FIXED CODE -- proper quoting. |
| 115 | |
| 116 | eval 'array=(' "$x" ')' # array |
| 117 | |
| 118 | test "$x" = "${array[0]}" || echo FALSE |
| 119 | [[ $x == /* ]] || echo FALSE # doesn't start with / |
| 120 | [[ "${array[0]}" == /* ]] && echo TRUE |
| 121 | ## STDOUT: |
| 122 | FALSE |
| 123 | FALSE |
| 124 | TRUE |
| 125 | ## END |
| 126 | ## N-I dash status: 2 |
| 127 | ## N-I dash stdout-json: "" |
| 128 | ## BUG mksh status: 1 |
| 129 | ## BUG mksh STDOUT: |
| 130 | FALSE |
| 131 | ## END |
| 132 | ## BUG zsh status: 1 |
| 133 | ## BUG zsh STDOUT: |
| 134 | FALSE |
| 135 | FALSE |
| 136 | ## END |
| 137 | |
| 138 | #### source works for files in current directory (bash only) |
| 139 | cd $TMP |
| 140 | echo "echo current dir" > cmd |
| 141 | . cmd |
| 142 | echo status=$? |
| 143 | ## STDOUT: |
| 144 | current dir |
| 145 | status=0 |
| 146 | ## END |
| 147 | ## N-I zsh STDOUT: |
| 148 | status=127 |
| 149 | ## END |
| 150 | |
| 151 | # This is a special builtin so failure is fatal. |
| 152 | |
| 153 | ## N-I dash stdout-json: "" |
| 154 | ## N-I dash status: 2 |
| 155 | ## N-I mksh stdout-json: "" |
| 156 | ## N-I mksh status: 1 |
| 157 | |
| 158 | #### source looks in PATH for files |
| 159 | mkdir -p dir |
| 160 | echo "echo hi" > dir/cmd |
| 161 | PATH="dir:$PATH" |
| 162 | . cmd |
| 163 | rm dir/cmd |
| 164 | ## STDOUT: |
| 165 | hi |
| 166 | ## END |
| 167 | |
| 168 | #### source finds files in PATH before current dir |
| 169 | cd $TMP |
| 170 | mkdir -p dir |
| 171 | echo "echo path" > dir/cmd |
| 172 | echo "echo current dir" > cmd |
| 173 | PATH="dir:$PATH" |
| 174 | . cmd |
| 175 | echo status=$? |
| 176 | ## STDOUT: |
| 177 | path |
| 178 | status=0 |
| 179 | ## END |
| 180 | |
| 181 | #### source works for files in subdirectory |
| 182 | mkdir -p dir |
| 183 | echo "echo path" > dir/cmd |
| 184 | . dir/cmd |
| 185 | rm dir/cmd |
| 186 | ## STDOUT: |
| 187 | path |
| 188 | ## END |
| 189 | |
| 190 | #### exit within eval (regression) |
| 191 | eval 'exit 42' |
| 192 | echo 'should not get here' |
| 193 | ## stdout-json: "" |
| 194 | ## status: 42 |
| 195 | |
| 196 | |
| 197 | #### exit within source (regression) |
| 198 | cd $TMP |
| 199 | echo 'exit 42' > lib.sh |
| 200 | . ./lib.sh |
| 201 | echo 'should not get here' |
| 202 | ## stdout-json: "" |
| 203 | ## status: 42 |