1 | #!/usr/bin/env python2
|
2 | from __future__ import print_function
|
3 |
|
4 | from typing import List, Dict, Optional, Any
|
5 |
|
6 |
|
7 | class Option(object):
|
8 |
|
9 | def __init__(self,
|
10 | index,
|
11 | name,
|
12 | short_flag=None,
|
13 | builtin='shopt',
|
14 | default=False,
|
15 | implemented=True,
|
16 | groups=None):
|
17 | # type: (int, str, str, Optional[str], bool, bool, List[str]) -> None
|
18 | self.index = index
|
19 | self.name = name # e.g. 'errexit'
|
20 | self.short_flag = short_flag # 'e' for -e
|
21 |
|
22 | if short_flag:
|
23 | self.builtin = 'set'
|
24 | else:
|
25 | # The 'interactive' option is the only one where builtin is None. It has
|
26 | # a cell but you can't change it. Only the shell can.
|
27 | self.builtin = builtin
|
28 |
|
29 | self.default = default # default value is True in some cases
|
30 | self.implemented = implemented
|
31 | self.groups = groups or [] # list of groups
|
32 |
|
33 | # for optview
|
34 | self.is_parse = (name.startswith('parse_') or
|
35 | name.startswith('strict_parse_') or
|
36 | name == 'expand_aliases')
|
37 | # interactive() is an accessor
|
38 | self.is_exec = implemented and not self.is_parse
|
39 |
|
40 |
|
41 | class _OptionDef(object):
|
42 | """Description of all shell options.
|
43 |
|
44 | Similar to id_kind_def.IdSpec
|
45 | """
|
46 |
|
47 | def __init__(self):
|
48 | # type: () -> None
|
49 | self.opts = [] # type: List[Option]
|
50 | self.index = 1 # start with 1
|
51 | self.array_size = -1
|
52 |
|
53 | def Add(self, *args, **kwargs):
|
54 | # type: (Any, Any) -> None
|
55 | self.opts.append(Option(self.index, *args, **kwargs))
|
56 | self.index += 1
|
57 |
|
58 | def DoneWithImplementedOptions(self):
|
59 | # type: () -> None
|
60 | self.array_size = self.index
|
61 |
|
62 |
|
63 | # Used by builtin
|
64 | _OTHER_SET_OPTIONS = [
|
65 | # NOTE: set -i and +i is explicitly disallowed. Only osh -i or +i is valid
|
66 | # https://unix.stackexchange.com/questions/339506/can-an-interactive-shell-become-non-interactive-or-vice-versa
|
67 | ('n', 'noexec'),
|
68 | ('x', 'xtrace'),
|
69 | ('v', 'verbose'), # like xtrace, but prints unevaluated commands
|
70 | ('f', 'noglob'),
|
71 | ('C', 'noclobber'),
|
72 | ('E', 'errtrace'),
|
73 |
|
74 | # A no-op for modernish.
|
75 | (None, 'posix'),
|
76 | (None, 'vi'),
|
77 | (None, 'emacs'),
|
78 | ]
|
79 |
|
80 | _STRICT_OPTS = [
|
81 | # $a{[@]::} is not allowed, you need ${a[@]::0} or ${a[@]::n}
|
82 | 'strict_parse_slice',
|
83 |
|
84 | # These are RUNTIME strict options.
|
85 | 'strict_argv', # empty argv not allowed
|
86 | 'strict_arith', # string to integer conversions, e.g. x=foo; echo $(( x ))
|
87 |
|
88 | # No implicit conversions between string and array.
|
89 | # - foo="$@" not allowed because it decays. Should be foo=( "$@" ).
|
90 | # - ${a} not ${a[0]} (not implemented)
|
91 | # sane-array? compare arrays like [[ "$@" == "${a[@]}" ]], which is
|
92 | # incompatible because bash coerces
|
93 | # default: do not allow
|
94 | 'strict_array',
|
95 | 'strict_control_flow', # break/continue at top level is fatal
|
96 | # 'return $empty' and return "" are NOT accepted
|
97 | 'strict_errexit', # errexit can't be disabled in compound commands
|
98 | 'strict_nameref', # trap invalid variable names
|
99 | 'strict_word_eval', # negative slices, unicode
|
100 | 'strict_tilde', # ~nonexistent is an error (like zsh)
|
101 |
|
102 | # Not implemented
|
103 | 'strict_glob', # glob_.py GlobParser has warnings
|
104 | ]
|
105 |
|
106 | # These will break some programs, but the fix should be simple.
|
107 |
|
108 | # command_sub_errexit makes 'local foo=$(false)' and echo $(false) fail.
|
109 | # By default, we have mimic bash's undesirable behavior of ignoring
|
110 | # these failures, since ash copied it, and Alpine's abuild relies on it.
|
111 | #
|
112 | # Note that inherit_errexit is a strict option.
|
113 |
|
114 | _UPGRADE_RUNTIME_OPTS = [
|
115 | ('simple_word_eval', False), # No splitting; arity isn't data-dependent
|
116 | # Don't reparse program data as globs
|
117 | ('dashglob', True), # do globs return files starting with - ?
|
118 |
|
119 | # TODO: Should these be in strict mode?
|
120 | # The logic was that strict_errexit improves your bash programs, but these
|
121 | # would lead you to remove error handling. But the same could be said for
|
122 | # strict_array?
|
123 | ('command_sub_errexit', False), # check after command sub
|
124 | ('process_sub_fail', False), # like pipefail, but for <(sort foo.txt)
|
125 | ('xtrace_rich', False), # Hierarchical trace with PIDs
|
126 | ('xtrace_details', True), # Legacy set -x stuff
|
127 |
|
128 | # Whether status 141 in pipelines is turned into 0
|
129 | ('sigpipe_status_ok', False),
|
130 |
|
131 | # This applies to shell functions too
|
132 | # It's also turned on in interactive mode
|
133 | ('redefine_proc_func', True),
|
134 | ]
|
135 |
|
136 | # TODO: Add strict_arg_parse? For example, 'trap 1 2 3' shouldn't be
|
137 | # valid, because it has an extra argument. Builtins are inconsistent about
|
138 | # checking this.
|
139 |
|
140 | _YSH_RUNTIME_OPTS = [
|
141 | ('simple_echo', False), # echo takes 0 or 1 arguments
|
142 | ('simple_eval_builtin', False), # eval takes exactly 1 argument
|
143 |
|
144 | # only file tests (no strings), remove [, status 2
|
145 | ('simple_test_builtin', False),
|
146 |
|
147 | # TODO: simple_trap
|
148 |
|
149 | # Turn aliases off so we can statically parse. bash has it off
|
150 | # non-interactively, sothis shouldn't break much.
|
151 | ('expand_aliases', True),
|
152 | ]
|
153 |
|
154 | # Stuff that doesn't break too many programs.
|
155 | _UPGRADE_PARSE_OPTS = [
|
156 | 'parse_at', # @array, @[expr]
|
157 | 'parse_proc', # proc p { ... }
|
158 | 'parse_func', # func f(x) { ... }
|
159 | 'parse_brace', # cd /bin { ... }
|
160 | 'parse_bracket', # assert [42 === x]
|
161 |
|
162 | # bare assignment 'x = 42' is allowed in Hay { } blocks, but disallowed
|
163 | # everywhere else. It's not a command 'x' with arg '='.
|
164 | 'parse_equals',
|
165 | 'parse_paren', # if (x > 0) ...
|
166 | 'parse_ysh_string', # r'' u'' b'' and multi-line versions
|
167 | 'parse_triple_quote', # for ''' and """
|
168 | ]
|
169 |
|
170 | # Extra stuff that breaks too many programs.
|
171 | _YSH_PARSE_OPTS = [
|
172 | ('parse_at_all', False), # @ starting any word, e.g. @[] @{} @@ @_ @-
|
173 |
|
174 | # Legacy syntax that is removed. These options are distinct from strict_*
|
175 | # because they don't help you avoid bugs in bash programs. They just makes
|
176 | # the language more consistent.
|
177 | ('parse_backslash', True),
|
178 | ('parse_backticks', True),
|
179 | ('parse_dollar', True),
|
180 | ('parse_ignored', True),
|
181 | ('parse_sh_arith', True), # disallow all shell arithmetic, $(( )) etc.
|
182 | ('parse_dparen', True), # disallow bash's ((
|
183 | ('parse_dbracket', True), # disallow bash's [[
|
184 | ('parse_bare_word', True), # 'case bare' and 'for x in bare'
|
185 | ]
|
186 |
|
187 | # No-ops for bash compatibility
|
188 | _NO_OPS = [
|
189 | 'lastpipe', # this feature is always on
|
190 |
|
191 | # Handled one by one
|
192 | 'progcomp',
|
193 | 'histappend', # stubbed out for issue #218
|
194 | 'hostcomplete', # complete words with '@' ?
|
195 | 'cmdhist', # multi-line commands in history
|
196 |
|
197 | # Copied from https://www.gnu.org/software/bash/manual/bash.txt
|
198 | # except 'compat*' because they were deemed too ugly
|
199 | 'assoc_expand_once',
|
200 | 'autocd',
|
201 | 'cdable_vars',
|
202 | 'cdspell',
|
203 | 'checkhash',
|
204 | 'checkjobs',
|
205 | 'checkwinsize',
|
206 | 'complete_fullquote', # Set by default
|
207 | # If set, Bash quotes all shell metacharacters in filenames and
|
208 | # directory names when performing completion. If not set, Bash
|
209 | # removes metacharacters such as the dollar sign from the set of
|
210 | # characters that will be quoted in completed filenames when
|
211 | # these metacharacters appear in shell variable references in
|
212 | # words to be completed. This means that dollar signs in
|
213 | # variable names that expand to directories will not be quoted;
|
214 | # however, any dollar signs appearing in filenames will not be
|
215 | # quoted, either. This is active only when bash is using
|
216 | # backslashes to quote completed filenames. This variable is
|
217 | # set by default, which is the default Bash behavior in versions
|
218 | # through 4.2.
|
219 | 'direxpand',
|
220 | 'dirspell',
|
221 | 'dotglob',
|
222 | 'execfail',
|
223 | 'extdebug', # for --debugger?
|
224 | 'extquote',
|
225 | 'force_fignore',
|
226 | 'globasciiranges',
|
227 | 'globstar', # TODO: implement **
|
228 | 'gnu_errfmt',
|
229 | 'histreedit',
|
230 | 'histverify',
|
231 | 'huponexit',
|
232 | 'interactive_comments',
|
233 | 'lithist',
|
234 | 'localvar_inherit',
|
235 | 'localvar_unset',
|
236 | 'login_shell',
|
237 | 'mailwarn',
|
238 | 'no_empty_cmd_completion',
|
239 | 'nocaseglob',
|
240 | 'progcomp_alias',
|
241 | 'promptvars',
|
242 | 'restricted_shell',
|
243 | 'shift_verbose',
|
244 | 'sourcepath',
|
245 | 'xpg_echo',
|
246 | ]
|
247 |
|
248 |
|
249 | def _Init(opt_def):
|
250 | # type: (_OptionDef) -> None
|
251 |
|
252 | opt_def.Add('errexit',
|
253 | short_flag='e',
|
254 | builtin='set',
|
255 | groups=['ysh:upgrade', 'ysh:all'])
|
256 | opt_def.Add('nounset',
|
257 | short_flag='u',
|
258 | builtin='set',
|
259 | groups=['ysh:upgrade', 'ysh:all'])
|
260 | opt_def.Add('pipefail', builtin='set', groups=['ysh:upgrade', 'ysh:all'])
|
261 |
|
262 | opt_def.Add('inherit_errexit', groups=['ysh:upgrade', 'ysh:all'])
|
263 | # Hm is this subsumed by simple_word_eval?
|
264 | opt_def.Add('nullglob', groups=['ysh:upgrade', 'ysh:all'])
|
265 | opt_def.Add('verbose_errexit', groups=['ysh:upgrade', 'ysh:all'])
|
266 |
|
267 | # set -o noclobber, etc.
|
268 | for short_flag, name in _OTHER_SET_OPTIONS:
|
269 | opt_def.Add(name, short_flag=short_flag, builtin='set')
|
270 |
|
271 | # The only one where builtin=None. Only the shell can change it.
|
272 | opt_def.Add('interactive', builtin=None)
|
273 |
|
274 | # bash --norc -c 'set -o' shows this is on by default
|
275 | opt_def.Add('hashall', short_flag='h', builtin='set', default=True)
|
276 |
|
277 | #
|
278 | # shopt
|
279 | # (bash uses $BASHOPTS rather than $SHELLOPTS)
|
280 | #
|
281 |
|
282 | # shopt options that aren't in any groups.
|
283 | opt_def.Add('failglob')
|
284 | opt_def.Add('extglob')
|
285 | opt_def.Add('nocasematch')
|
286 |
|
287 | # recursive parsing and evaluation - for compatibility, ble.sh, etc.
|
288 | opt_def.Add('eval_unsafe_arith')
|
289 |
|
290 | opt_def.Add('ignore_flags_not_impl')
|
291 | opt_def.Add('ignore_opts_not_impl')
|
292 |
|
293 | # For implementing strict_errexit
|
294 | # TODO: could be _no_command_sub / _no_process_sub, if we had to discourage
|
295 | # "default True" options
|
296 | opt_def.Add('_allow_command_sub', default=True)
|
297 | opt_def.Add('_allow_process_sub', default=True)
|
298 |
|
299 | # For implementing 'proc'
|
300 | opt_def.Add('dynamic_scope', default=True)
|
301 |
|
302 | # On in interactive shell
|
303 | opt_def.Add('redefine_module', default=False)
|
304 | # Hm these aren't the same?
|
305 | #opt_def.Add('redefine_proc_func', default=False),
|
306 |
|
307 | # For disabling strict_errexit while running traps. Because we run in the
|
308 | # main loop, the value can be "off". Prefix with _ because it's undocumented
|
309 | # and users shouldn't fiddle with it. We need a stack so this is a
|
310 | # convenient place.
|
311 | opt_def.Add('_running_trap')
|
312 | opt_def.Add('_running_hay')
|
313 |
|
314 | # For fixing lastpipe / job control / DEBUG trap interaction
|
315 | opt_def.Add('_no_debug_trap')
|
316 | # To implement ERR trap semantics - it's only run for the WHOLE pipeline,
|
317 | # not each part (even the last part)
|
318 | opt_def.Add('_no_err_trap')
|
319 |
|
320 | # shopt -s strict_arith, etc.
|
321 | for name in _STRICT_OPTS:
|
322 | opt_def.Add(name, groups=['strict:all', 'ysh:all'])
|
323 |
|
324 | #
|
325 | # Options that enable YSH features
|
326 | #
|
327 |
|
328 | for name in _UPGRADE_PARSE_OPTS:
|
329 | opt_def.Add(name, groups=['ysh:upgrade', 'ysh:all'])
|
330 | # shopt -s simple_word_eval, etc.
|
331 | for name, default in _UPGRADE_RUNTIME_OPTS:
|
332 | opt_def.Add(name, default=default, groups=['ysh:upgrade', 'ysh:all'])
|
333 |
|
334 | for name, default in _YSH_PARSE_OPTS:
|
335 | opt_def.Add(name, default=default, groups=['ysh:all'])
|
336 | for name, default in _YSH_RUNTIME_OPTS:
|
337 | opt_def.Add(name, default=default, groups=['ysh:all'])
|
338 |
|
339 | opt_def.DoneWithImplementedOptions()
|
340 |
|
341 | # NO_OPS
|
342 |
|
343 | # Stubs for shopt -s xpg_echo, etc.
|
344 | for name in _NO_OPS:
|
345 | opt_def.Add(name, implemented=False)
|
346 |
|
347 |
|
348 | def All():
|
349 | # type: () -> List[Option]
|
350 | """Return a list of options with metadata.
|
351 |
|
352 | - Used by osh/builtin_pure.py to construct the arg spec.
|
353 | - Used by frontend/lexer_gen.py to construct the lexer/matcher
|
354 | """
|
355 | return _OPTION_DEF.opts
|
356 |
|
357 |
|
358 | def ArraySize():
|
359 | # type: () -> int
|
360 | """Unused now, since we use opt_num::ARRAY_SIZE.
|
361 |
|
362 | We could get rid of unimplemented options and shrink the array.
|
363 | """
|
364 | return _OPTION_DEF.array_size
|
365 |
|
366 |
|
367 | def OptionDict():
|
368 | # type: () -> Dict[str, int]
|
369 | """For the slow path in frontend/match.py."""
|
370 | return dict((opt.name, opt.index) for opt in _OPTION_DEF.opts)
|
371 |
|
372 |
|
373 | def ParseOptNames():
|
374 | # type: () -> List[str]
|
375 | """Used by core/optview*.py."""
|
376 | return [opt.name for opt in _OPTION_DEF.opts if opt.is_parse]
|
377 |
|
378 |
|
379 | def ExecOptNames():
|
380 | # type: () -> List[str]
|
381 | """Used by core/optview*.py."""
|
382 | return [opt.name for opt in _OPTION_DEF.opts if opt.is_exec]
|
383 |
|
384 |
|
385 | _OPTION_DEF = _OptionDef()
|
386 |
|
387 | _Init(_OPTION_DEF)
|
388 |
|
389 | # Sort by name because we print options.
|
390 | # TODO: for MEMBERSHIP queries, we could sort by the most common? errexit
|
391 | # first?
|
392 | _SORTED = sorted(_OPTION_DEF.opts, key=lambda opt: opt.name)
|
393 |
|
394 | PARSE_OPTION_NUMS = [opt.index for opt in _SORTED if opt.is_parse]
|
395 |
|
396 | # Sorted because 'shopt -o -p' should be sorted, etc.
|
397 | VISIBLE_SHOPT_NUMS = [
|
398 | opt.index for opt in _SORTED if opt.builtin == 'shopt' and opt.implemented
|
399 | ]
|
400 |
|
401 | YSH_UPGRADE = [opt.index for opt in _SORTED if 'ysh:upgrade' in opt.groups]
|
402 | YSH_ALL = [opt.index for opt in _SORTED if 'ysh:all' in opt.groups]
|
403 | STRICT_ALL = [opt.index for opt in _SORTED if 'strict:all' in opt.groups]
|
404 | DEFAULT_TRUE = [opt.index for opt in _SORTED if opt.default]
|
405 | #print([opt.name for opt in _SORTED if opt.default])
|
406 |
|
407 | META_OPTIONS = ['strict:all', 'ysh:upgrade',
|
408 | 'ysh:all'] # Passed to flag parser
|
409 |
|
410 | # For printing option names to stdout. Wrapped by frontend/consts.
|
411 | OPTION_NAMES = dict((opt.index, opt.name) for opt in _SORTED)
|