just-bash 2.5.5 → 2.7.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +3 -3
- package/dist/AGENTS.md +3 -3
- package/dist/ast/types.d.ts +100 -11
- package/dist/bin/chunks/alias-EGIS5LUE.js +7 -0
- package/dist/bin/chunks/awk2-GFEJOWML.js +21 -0
- package/dist/bin/chunks/bash-PGDTHIM2.js +6 -0
- package/dist/bin/chunks/chmod-TYLTHXFR.js +9 -0
- package/dist/bin/chunks/chunk-D5WP4CKS.js +3 -0
- package/dist/bin/chunks/chunk-FSAGDARS.js +74 -0
- package/dist/bin/chunks/chunk-FSZWFMB4.js +16 -0
- package/dist/bin/chunks/chunk-IRUD2E3M.js +17 -0
- package/dist/bin/chunks/chunk-K5IXNHO5.js +8 -0
- package/dist/bin/chunks/chunk-KD3EODLB.js +6 -0
- package/dist/bin/{shell/chunks/echo-WSKTON6U.js → chunks/echo-7I42V66Q.js} +2 -2
- package/dist/bin/chunks/env-7A4MH7BJ.js +9 -0
- package/dist/bin/chunks/expansion-BOR3ELLC.js +2 -0
- package/dist/bin/{shell/chunks/find-CBEJ35BR.js → chunks/find-PHDZK64M.js} +1 -1
- package/dist/bin/{shell/chunks/grep-IIVQXFNI.js → chunks/grep-VX7MJMVN.js} +2 -2
- package/dist/bin/chunks/{head-DYK37Z24.js → head-TJHLLIMR.js} +1 -1
- package/dist/bin/{shell/chunks/jq-XXZPU5CA.js → chunks/jq-RGZHJNXC.js} +1 -1
- package/dist/bin/chunks/od-KRKGC2U3.js +5 -0
- package/dist/bin/chunks/printf-YPXD4CRE.js +15 -0
- package/dist/bin/chunks/pwd-L26WH2K4.js +3 -0
- package/dist/bin/chunks/python3-JGT65AEB.js +14 -0
- package/dist/bin/{shell/chunks/rg-Q4OTJOEF.js → chunks/rg-RSDLLECO.js} +1 -1
- package/dist/bin/chunks/rmdir-GOODLY5W.js +14 -0
- package/dist/bin/chunks/sed-JPDTWF4W.js +100 -0
- package/dist/bin/{shell/chunks/tail-BES27CZT.js → chunks/tail-YAUIERGN.js} +1 -1
- package/dist/bin/{shell/chunks/tar-C27YYUAS.js → chunks/tar-LFENC54A.js} +7 -7
- package/dist/bin/chunks/time-37F5EBPK.js +14 -0
- package/dist/bin/chunks/touch-IBPHJBM3.js +7 -0
- package/dist/bin/chunks/wc-SAOHEZYP.js +6 -0
- package/dist/bin/chunks/{which-73KOOLC6.js → which-FCDFBOMN.js} +1 -1
- package/dist/bin/chunks/whoami-JVLUNKSG.js +3 -0
- package/dist/bin/chunks/worker.js +1038 -0
- package/dist/bin/{shell/chunks/xan-6K2NGTHM.js → chunks/xan-5HNHTFMB.js} +19 -19
- package/dist/bin/chunks/{yq-MF2SNFGL.js → yq-PFV4T2PV.js} +1 -1
- package/dist/bin/just-bash.js +659 -135
- package/dist/bin/shell/chunks/alias-EGIS5LUE.js +7 -0
- package/dist/bin/shell/chunks/awk2-GFEJOWML.js +21 -0
- package/dist/bin/shell/chunks/bash-PGDTHIM2.js +6 -0
- package/dist/bin/shell/chunks/chmod-TYLTHXFR.js +9 -0
- package/dist/bin/shell/chunks/chunk-D5WP4CKS.js +3 -0
- package/dist/bin/shell/chunks/chunk-FSAGDARS.js +74 -0
- package/dist/bin/shell/chunks/chunk-FSZWFMB4.js +16 -0
- package/dist/bin/shell/chunks/chunk-IRUD2E3M.js +17 -0
- package/dist/bin/shell/chunks/chunk-K5IXNHO5.js +8 -0
- package/dist/bin/shell/chunks/chunk-KD3EODLB.js +6 -0
- package/dist/bin/{chunks/echo-WSKTON6U.js → shell/chunks/echo-7I42V66Q.js} +2 -2
- package/dist/bin/shell/chunks/env-7A4MH7BJ.js +9 -0
- package/dist/bin/shell/chunks/expansion-BOR3ELLC.js +2 -0
- package/dist/bin/{chunks/find-CBEJ35BR.js → shell/chunks/find-PHDZK64M.js} +1 -1
- package/dist/bin/{chunks/grep-IIVQXFNI.js → shell/chunks/grep-VX7MJMVN.js} +2 -2
- package/dist/bin/shell/chunks/{head-DYK37Z24.js → head-TJHLLIMR.js} +1 -1
- package/dist/bin/{chunks/jq-XXZPU5CA.js → shell/chunks/jq-RGZHJNXC.js} +1 -1
- package/dist/bin/shell/chunks/od-KRKGC2U3.js +5 -0
- package/dist/bin/shell/chunks/printf-YPXD4CRE.js +15 -0
- package/dist/bin/shell/chunks/pwd-L26WH2K4.js +3 -0
- package/dist/bin/shell/chunks/python3-JGT65AEB.js +14 -0
- package/dist/bin/{chunks/rg-Q4OTJOEF.js → shell/chunks/rg-RSDLLECO.js} +1 -1
- package/dist/bin/shell/chunks/rmdir-GOODLY5W.js +14 -0
- package/dist/bin/shell/chunks/sed-JPDTWF4W.js +100 -0
- package/dist/bin/{chunks/tail-BES27CZT.js → shell/chunks/tail-YAUIERGN.js} +1 -1
- package/dist/bin/{chunks/tar-C27YYUAS.js → shell/chunks/tar-LFENC54A.js} +7 -7
- package/dist/bin/shell/chunks/time-37F5EBPK.js +14 -0
- package/dist/bin/shell/chunks/touch-IBPHJBM3.js +7 -0
- package/dist/bin/shell/chunks/wc-SAOHEZYP.js +6 -0
- package/dist/bin/shell/chunks/{which-73KOOLC6.js → which-FCDFBOMN.js} +1 -1
- package/dist/bin/shell/chunks/whoami-JVLUNKSG.js +3 -0
- package/dist/bin/{chunks/xan-6K2NGTHM.js → shell/chunks/xan-5HNHTFMB.js} +19 -19
- package/dist/bin/shell/chunks/{yq-MF2SNFGL.js → yq-PFV4T2PV.js} +1 -1
- package/dist/bin/shell/shell.js +664 -140
- package/dist/bundle/browser.js +1251 -606
- package/dist/bundle/chunks/alias-ATFBB6D2.js +6 -0
- package/dist/bundle/chunks/awk2-6FBZTP57.js +20 -0
- package/dist/bundle/chunks/bash-OLRNM52U.js +5 -0
- package/dist/bundle/chunks/chmod-ODWUR7E6.js +8 -0
- package/dist/bundle/chunks/chunk-3AWP5CWK.js +73 -0
- package/dist/bundle/chunks/chunk-CXEWLFNE.js +16 -0
- package/dist/bundle/chunks/chunk-CZPA5RBA.js +5 -0
- package/dist/bundle/chunks/chunk-EEXR5ZDP.js +2 -0
- package/dist/bundle/chunks/chunk-HDQ56CKY.js +15 -0
- package/dist/bundle/chunks/chunk-PSW6BMXW.js +7 -0
- package/dist/bundle/chunks/{echo-VUHWYV6L.js → echo-6S7WE7XB.js} +2 -2
- package/dist/bundle/chunks/env-2UI6XINU.js +8 -0
- package/dist/bundle/chunks/expansion-RIGCFEMA.js +1 -0
- package/dist/bundle/chunks/{find-ACOAWALE.js → find-YGMSVGUV.js} +1 -1
- package/dist/bundle/chunks/{grep-ACVE42JK.js → grep-NIC6JNLH.js} +2 -2
- package/dist/bundle/chunks/{head-FZ6IQHYW.js → head-SA7P5NJ7.js} +1 -1
- package/dist/bundle/chunks/{jq-3YU5HRKE.js → jq-RLRYRPOJ.js} +1 -1
- package/dist/bundle/chunks/od-3FPDPLWJ.js +4 -0
- package/dist/bundle/chunks/printf-66XGXFCD.js +14 -0
- package/dist/bundle/chunks/pwd-S4NVAMC4.js +2 -0
- package/dist/bundle/chunks/python3-3OP7EKER.js +13 -0
- package/dist/bundle/chunks/{rg-YLZJWCEJ.js → rg-SRMB7L6G.js} +1 -1
- package/dist/bundle/chunks/rmdir-XFQE4ZYV.js +13 -0
- package/dist/bundle/chunks/sed-IV6HLDXU.js +99 -0
- package/dist/bundle/chunks/{tail-PD4RZR6J.js → tail-52LRAWXT.js} +1 -1
- package/dist/bundle/chunks/{tar-QWBXMF7K.js → tar-LWIHPMT6.js} +7 -7
- package/dist/bundle/chunks/time-UWXBG6CS.js +13 -0
- package/dist/bundle/chunks/touch-TDTEBHHI.js +6 -0
- package/dist/bundle/chunks/wc-HE5XARI4.js +5 -0
- package/dist/bundle/chunks/{which-M5MQ6QXQ.js → which-UBLRBDHN.js} +1 -1
- package/dist/bundle/chunks/whoami-YUDAIS32.js +2 -0
- package/dist/bundle/chunks/worker.js +1038 -0
- package/dist/bundle/chunks/{xan-2R2APJJ4.js → xan-A6VPI4HJ.js} +19 -19
- package/dist/bundle/chunks/{yq-KANM4MD2.js → yq-L665QPQU.js} +1 -1
- package/dist/bundle/index.js +657 -133
- package/dist/commands/awk/ast.d.ts +1 -0
- package/dist/commands/awk/interpreter/context.d.ts +14 -0
- package/dist/commands/awk/interpreter/interpreter.d.ts +2 -0
- package/dist/commands/awk/interpreter/{helpers.d.ts → type-coercion.d.ts} +4 -1
- package/dist/commands/awk/parser2-print.d.ts +35 -0
- package/dist/commands/awk/parser2.d.ts +42 -34
- package/dist/commands/printf/strftime.d.ts +9 -0
- package/dist/commands/python3/fs-bridge-handler.d.ts +50 -0
- package/dist/commands/python3/protocol.d.ts +138 -0
- package/dist/commands/python3/python3.d.ts +11 -0
- package/dist/commands/python3/sync-fs-backend.d.ts +59 -0
- package/dist/commands/python3/worker.d.ts +16 -0
- package/dist/commands/query-engine/builtins/array-builtins.d.ts +20 -0
- package/dist/commands/query-engine/builtins/control-builtins.d.ts +18 -0
- package/dist/commands/query-engine/builtins/date-builtins.d.ts +15 -0
- package/dist/commands/query-engine/builtins/format-builtins.d.ts +11 -0
- package/dist/commands/query-engine/builtins/index-builtins.d.ts +16 -0
- package/dist/commands/query-engine/builtins/index.d.ts +17 -0
- package/dist/commands/query-engine/builtins/math-builtins.d.ts +15 -0
- package/dist/commands/query-engine/builtins/navigation-builtins.d.ts +18 -0
- package/dist/commands/query-engine/builtins/object-builtins.d.ts +15 -0
- package/dist/commands/query-engine/builtins/path-builtins.d.ts +20 -0
- package/dist/commands/query-engine/builtins/sql-builtins.d.ts +16 -0
- package/dist/commands/query-engine/builtins/string-builtins.d.ts +15 -0
- package/dist/commands/query-engine/builtins/type-builtins.d.ts +11 -0
- package/dist/commands/query-engine/evaluator.d.ts +10 -2
- package/dist/commands/query-engine/parser-types.d.ts +171 -0
- package/dist/commands/query-engine/parser.d.ts +2 -132
- package/dist/commands/query-engine/path-operations.d.ts +15 -0
- package/dist/commands/query-engine/value-operations.d.ts +39 -0
- package/dist/commands/registry.d.ts +1 -1
- package/dist/commands/rmdir/rmdir.d.ts +2 -0
- package/dist/commands/sed/lexer.d.ts +12 -0
- package/dist/commands/sed/parser.d.ts +9 -0
- package/dist/commands/sed/sed-regex.d.ts +23 -0
- package/dist/commands/sed/types.d.ts +13 -1
- package/dist/commands/tar/tar-options.d.ts +36 -0
- package/dist/commands/time/time.d.ts +25 -0
- package/dist/commands/whoami/whoami.d.ts +9 -0
- package/dist/commands/xan/moonblade-tokenizer.d.ts +25 -0
- package/dist/fs/in-memory-fs/in-memory-fs.d.ts +12 -0
- package/dist/fs/interface.d.ts +16 -0
- package/dist/fs/mountable-fs/mountable-fs.d.ts +12 -0
- package/dist/fs/overlay-fs/overlay-fs.d.ts +13 -1
- package/dist/fs/read-write-fs/read-write-fs.d.ts +13 -1
- package/dist/interpreter/alias-expansion.d.ts +23 -0
- package/dist/interpreter/arithmetic.d.ts +1 -6
- package/dist/interpreter/assignment-expansion.d.ts +24 -0
- package/dist/interpreter/builtin-dispatch.d.ts +39 -0
- package/dist/interpreter/builtins/compgen.d.ts +26 -0
- package/dist/interpreter/builtins/complete.d.ts +17 -0
- package/dist/interpreter/builtins/compopt.d.ts +28 -0
- package/dist/interpreter/builtins/declare-array-parsing.d.ts +14 -0
- package/dist/interpreter/builtins/declare-print.d.ts +39 -0
- package/dist/interpreter/builtins/declare.d.ts +10 -4
- package/dist/interpreter/builtins/dirs.d.ts +29 -0
- package/dist/interpreter/builtins/eval.d.ts +1 -1
- package/dist/interpreter/builtins/export.d.ts +1 -0
- package/dist/interpreter/builtins/getopts.d.ts +18 -0
- package/dist/interpreter/builtins/hash.d.ts +19 -0
- package/dist/interpreter/builtins/help.d.ts +12 -0
- package/dist/interpreter/builtins/index.d.ts +9 -1
- package/dist/interpreter/builtins/local.d.ts +1 -1
- package/dist/interpreter/builtins/read.d.ts +1 -1
- package/dist/interpreter/builtins/set.d.ts +3 -0
- package/dist/interpreter/builtins/shift.d.ts +3 -0
- package/dist/interpreter/builtins/shopt.d.ts +7 -0
- package/dist/interpreter/builtins/unset.d.ts +5 -1
- package/dist/interpreter/builtins/variable-assignment.d.ts +66 -0
- package/dist/interpreter/command-resolution.d.ts +43 -0
- package/dist/interpreter/conditionals.d.ts +1 -1
- package/dist/interpreter/errors.d.ts +36 -1
- package/dist/interpreter/expansion/analysis.d.ts +15 -17
- package/dist/interpreter/expansion/arith-text-expansion.d.ts +20 -0
- package/dist/interpreter/expansion/array-pattern-ops.d.ts +21 -0
- package/dist/interpreter/expansion/array-prefix-suffix.d.ts +46 -0
- package/dist/interpreter/expansion/array-slice-transform.d.ts +36 -0
- package/dist/interpreter/expansion/array-word-expansion.d.ts +39 -0
- package/dist/interpreter/expansion/command-substitution.d.ts +23 -0
- package/dist/interpreter/expansion/glob-escape.d.ts +32 -0
- package/dist/interpreter/expansion/indirect-expansion.d.ts +42 -0
- package/dist/interpreter/expansion/parameter-ops.d.ts +113 -0
- package/dist/interpreter/expansion/pattern-expansion.d.ts +23 -0
- package/dist/interpreter/expansion/pattern-removal.d.ts +18 -0
- package/dist/interpreter/expansion/pattern.d.ts +2 -1
- package/dist/interpreter/expansion/positional-params.d.ts +59 -0
- package/dist/interpreter/expansion/prompt.d.ts +39 -0
- package/dist/interpreter/expansion/quoting.d.ts +13 -0
- package/dist/interpreter/expansion/tilde.d.ts +12 -0
- package/dist/interpreter/expansion/unquoted-expansion.d.ts +76 -0
- package/dist/interpreter/expansion/variable-attrs.d.ts +19 -0
- package/dist/interpreter/expansion/variable.d.ts +10 -1
- package/dist/interpreter/expansion/word-glob-expansion.d.ts +33 -0
- package/dist/interpreter/expansion/word-split.d.ts +11 -6
- package/dist/interpreter/expansion.d.ts +30 -4
- package/dist/interpreter/functions.d.ts +1 -1
- package/dist/interpreter/helpers/array.d.ts +20 -0
- package/dist/interpreter/helpers/ifs.d.ts +66 -5
- package/dist/interpreter/helpers/nameref.d.ts +65 -0
- package/dist/interpreter/helpers/quoting.d.ts +24 -0
- package/dist/interpreter/helpers/readonly.d.ts +28 -4
- package/dist/interpreter/helpers/shell-constants.d.ts +25 -0
- package/dist/interpreter/helpers/shellopts.d.ts +28 -0
- package/dist/interpreter/helpers/string-compare.d.ts +3 -1
- package/dist/interpreter/helpers/tilde.d.ts +13 -0
- package/dist/interpreter/helpers/variable-tests.d.ts +1 -1
- package/dist/interpreter/helpers/word-matching.d.ts +26 -0
- package/dist/interpreter/helpers/xtrace.d.ts +18 -0
- package/dist/interpreter/interpreter.d.ts +13 -14
- package/dist/interpreter/pipeline-execution.d.ts +16 -0
- package/dist/interpreter/redirections.d.ts +38 -1
- package/dist/interpreter/simple-command-assignments.d.ts +29 -0
- package/dist/interpreter/subshell-group.d.ts +32 -0
- package/dist/interpreter/type-command.d.ts +37 -0
- package/dist/interpreter/types.d.ts +255 -21
- package/dist/limits.d.ts +2 -0
- package/dist/parser/arithmetic-parser.d.ts +2 -4
- package/dist/parser/arithmetic-primaries.d.ts +45 -0
- package/dist/parser/compound-parser.d.ts +21 -7
- package/dist/parser/expansion-parser.d.ts +7 -1
- package/dist/parser/lexer.d.ts +57 -0
- package/dist/parser/parser-substitution.d.ts +62 -0
- package/dist/parser/parser.d.ts +39 -2
- package/dist/parser/types.d.ts +1 -0
- package/dist/types.d.ts +17 -0
- package/package.json +10 -5
- package/dist/bin/chunks/alias-PCMLRCRW.js +0 -7
- package/dist/bin/chunks/awk2-D2US2LMM.js +0 -20
- package/dist/bin/chunks/bash-UT3MT5UU.js +0 -7
- package/dist/bin/chunks/chmod-3DIKREF7.js +0 -9
- package/dist/bin/chunks/chunk-ACRTDIBO.js +0 -6
- package/dist/bin/chunks/chunk-DV5HL4K2.js +0 -17
- package/dist/bin/chunks/chunk-J7BCMQDI.js +0 -16
- package/dist/bin/chunks/chunk-NWWB2XRE.js +0 -6
- package/dist/bin/chunks/chunk-PM2DS2YW.js +0 -3
- package/dist/bin/chunks/env-YLLSNZZN.js +0 -9
- package/dist/bin/chunks/od-SLM7SRWU.js +0 -7
- package/dist/bin/chunks/printf-HWUQKYUM.js +0 -14
- package/dist/bin/chunks/pwd-53KG2MCJ.js +0 -3
- package/dist/bin/chunks/sed-5LQMJYRJ.js +0 -80
- package/dist/bin/chunks/touch-TSNAXMZ2.js +0 -4
- package/dist/bin/chunks/wc-QSBRKIF5.js +0 -6
- package/dist/bin/shell/chunks/alias-PCMLRCRW.js +0 -7
- package/dist/bin/shell/chunks/awk2-D2US2LMM.js +0 -20
- package/dist/bin/shell/chunks/bash-UT3MT5UU.js +0 -7
- package/dist/bin/shell/chunks/chmod-3DIKREF7.js +0 -9
- package/dist/bin/shell/chunks/chunk-ACRTDIBO.js +0 -6
- package/dist/bin/shell/chunks/chunk-DV5HL4K2.js +0 -17
- package/dist/bin/shell/chunks/chunk-J7BCMQDI.js +0 -16
- package/dist/bin/shell/chunks/chunk-NWWB2XRE.js +0 -6
- package/dist/bin/shell/chunks/chunk-PM2DS2YW.js +0 -3
- package/dist/bin/shell/chunks/env-YLLSNZZN.js +0 -9
- package/dist/bin/shell/chunks/od-SLM7SRWU.js +0 -7
- package/dist/bin/shell/chunks/printf-HWUQKYUM.js +0 -14
- package/dist/bin/shell/chunks/pwd-53KG2MCJ.js +0 -3
- package/dist/bin/shell/chunks/sed-5LQMJYRJ.js +0 -80
- package/dist/bin/shell/chunks/touch-TSNAXMZ2.js +0 -4
- package/dist/bin/shell/chunks/wc-QSBRKIF5.js +0 -6
- package/dist/bundle/chunks/alias-4UGRF4DM.js +0 -6
- package/dist/bundle/chunks/awk2-46RTIZKB.js +0 -19
- package/dist/bundle/chunks/bash-ZILV3VHA.js +0 -6
- package/dist/bundle/chunks/chmod-FBHLEIY6.js +0 -8
- package/dist/bundle/chunks/chunk-4JO5BKO4.js +0 -5
- package/dist/bundle/chunks/chunk-BOJ3OAZB.js +0 -16
- package/dist/bundle/chunks/chunk-CG2HXOFG.js +0 -5
- package/dist/bundle/chunks/chunk-NWEGHOXL.js +0 -2
- package/dist/bundle/chunks/chunk-W2EKKAIL.js +0 -15
- package/dist/bundle/chunks/env-EG5SQSAQ.js +0 -8
- package/dist/bundle/chunks/od-ECWXNUB4.js +0 -6
- package/dist/bundle/chunks/printf-VG2POOXB.js +0 -13
- package/dist/bundle/chunks/pwd-AC4P3JKI.js +0 -2
- package/dist/bundle/chunks/sed-ZKODWZ6F.js +0 -79
- package/dist/bundle/chunks/touch-MVQSKQKT.js +0 -3
- package/dist/bundle/chunks/wc-DFQKWSIZ.js +0 -5
- package/dist/interpreter/builtins/variable-helpers.d.ts +0 -30
- /package/dist/bin/chunks/{curl-LCMREE7R.js → curl-TH7YRBSA.js} +0 -0
- /package/dist/bin/chunks/{file-LNCFDPQZ.js → file-6PCTL3MH.js} +0 -0
- /package/dist/bin/shell/chunks/{curl-LCMREE7R.js → curl-TH7YRBSA.js} +0 -0
- /package/dist/bin/shell/chunks/{file-LNCFDPQZ.js → file-6PCTL3MH.js} +0 -0
- /package/dist/bundle/chunks/{curl-TVZH24MD.js → curl-XLP4VABU.js} +0 -0
- /package/dist/bundle/chunks/{file-XPA6O6H2.js → file-NQP3CKRV.js} +0 -0
- /package/dist/commands/curl/{utils.d.ts → response-formatting.d.ts} +0 -0
- /package/dist/commands/xan/{xan-utils.d.ts → column-selection.d.ts} +0 -0
- /package/dist/fs/{utils.d.ts → encoding.d.ts} +0 -0
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Positional Parameter Expansion Handlers
|
|
3
|
+
*
|
|
4
|
+
* Handles $@ and $* expansion with various operations:
|
|
5
|
+
* - "${@:offset}" and "${*:offset}" - slicing
|
|
6
|
+
* - "${@/pattern/replacement}" - pattern replacement
|
|
7
|
+
* - "${@#pattern}" - pattern removal (strip)
|
|
8
|
+
* - "$@" and "$*" with adjacent text
|
|
9
|
+
*/
|
|
10
|
+
import type { WordPart } from "../../ast/types.js";
|
|
11
|
+
import type { InterpreterContext } from "../types.js";
|
|
12
|
+
/**
|
|
13
|
+
* Result type for positional parameter expansion handlers.
|
|
14
|
+
* `null` means the handler doesn't apply to this case.
|
|
15
|
+
*/
|
|
16
|
+
export type PositionalExpansionResult = {
|
|
17
|
+
values: string[];
|
|
18
|
+
quoted: boolean;
|
|
19
|
+
} | null;
|
|
20
|
+
import type { ArithExpr } from "../../ast/types.js";
|
|
21
|
+
/**
|
|
22
|
+
* Type for evaluateArithmetic function
|
|
23
|
+
*/
|
|
24
|
+
export type EvaluateArithmeticFn = (ctx: InterpreterContext, expr: ArithExpr, isExpansionContext?: boolean) => Promise<number>;
|
|
25
|
+
/**
|
|
26
|
+
* Type for expandPart function
|
|
27
|
+
*/
|
|
28
|
+
export type ExpandPartFn = (ctx: InterpreterContext, part: WordPart) => Promise<string>;
|
|
29
|
+
/**
|
|
30
|
+
* Type for expandWordPartsAsync function
|
|
31
|
+
*/
|
|
32
|
+
export type ExpandWordPartsAsyncFn = (ctx: InterpreterContext, parts: WordPart[]) => Promise<string>;
|
|
33
|
+
/**
|
|
34
|
+
* Handle "${@:offset}" and "${*:offset}" with Substring operations inside double quotes
|
|
35
|
+
* "${@:offset}": Each sliced positional parameter becomes a separate word
|
|
36
|
+
* "${*:offset}": All sliced params joined with IFS as ONE word
|
|
37
|
+
*/
|
|
38
|
+
export declare function handlePositionalSlicing(ctx: InterpreterContext, wordParts: WordPart[], evaluateArithmetic: EvaluateArithmeticFn, expandPart: ExpandPartFn): Promise<PositionalExpansionResult>;
|
|
39
|
+
/**
|
|
40
|
+
* Handle "${@/pattern/replacement}" and "${* /pattern/replacement}" with PatternReplacement inside double quotes
|
|
41
|
+
* "${@/pattern/replacement}": Each positional parameter has pattern replaced, each becomes a separate word
|
|
42
|
+
* "${* /pattern/replacement}": All params joined with IFS, pattern replaced, becomes ONE word
|
|
43
|
+
*/
|
|
44
|
+
export declare function handlePositionalPatternReplacement(ctx: InterpreterContext, wordParts: WordPart[], expandPart: ExpandPartFn, expandWordPartsAsync: ExpandWordPartsAsyncFn): Promise<PositionalExpansionResult>;
|
|
45
|
+
/**
|
|
46
|
+
* Handle "${@#pattern}" and "${*#pattern}" - positional parameter pattern removal (strip)
|
|
47
|
+
* "${@#pattern}": Remove shortest matching prefix from each parameter, each becomes a separate word
|
|
48
|
+
* "${@##pattern}": Remove longest matching prefix from each parameter
|
|
49
|
+
* "${@%pattern}": Remove shortest matching suffix from each parameter
|
|
50
|
+
* "${@%%pattern}": Remove longest matching suffix from each parameter
|
|
51
|
+
*/
|
|
52
|
+
export declare function handlePositionalPatternRemoval(ctx: InterpreterContext, wordParts: WordPart[], expandPart: ExpandPartFn, expandWordPartsAsync: ExpandWordPartsAsyncFn): Promise<PositionalExpansionResult>;
|
|
53
|
+
/**
|
|
54
|
+
* Handle "$@" and "$*" with adjacent text inside double quotes, e.g., "-$@-"
|
|
55
|
+
* "$@": Each positional parameter becomes a separate word, with prefix joined to first
|
|
56
|
+
* and suffix joined to last. If no params, produces nothing (or just prefix+suffix if present)
|
|
57
|
+
* "$*": All params joined with IFS as ONE word. If no params, produces one empty word.
|
|
58
|
+
*/
|
|
59
|
+
export declare function handleSimplePositionalExpansion(ctx: InterpreterContext, wordParts: WordPart[], expandPart: ExpandPartFn): Promise<PositionalExpansionResult>;
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Prompt expansion
|
|
3
|
+
*
|
|
4
|
+
* Handles prompt escape sequences for ${var@P} transformation and PS1/PS2/PS3/PS4.
|
|
5
|
+
*/
|
|
6
|
+
import type { InterpreterContext } from "../types.js";
|
|
7
|
+
/**
|
|
8
|
+
* Expand prompt escape sequences (${var@P} transformation)
|
|
9
|
+
* Interprets backslash escapes used in PS1, PS2, PS3, PS4 prompt strings.
|
|
10
|
+
*
|
|
11
|
+
* Supported escapes:
|
|
12
|
+
* - \a - bell (ASCII 07)
|
|
13
|
+
* - \e - escape (ASCII 033)
|
|
14
|
+
* - \n - newline
|
|
15
|
+
* - \r - carriage return
|
|
16
|
+
* - \\ - literal backslash
|
|
17
|
+
* - \$ - $ for regular user, # for root (always $ here)
|
|
18
|
+
* - \[ and \] - non-printing sequence delimiters (removed)
|
|
19
|
+
* - \u - username
|
|
20
|
+
* - \h - short hostname (up to first .)
|
|
21
|
+
* - \H - full hostname
|
|
22
|
+
* - \w - current working directory
|
|
23
|
+
* - \W - basename of current working directory
|
|
24
|
+
* - \d - date (Weekday Month Day format)
|
|
25
|
+
* - \t - time HH:MM:SS (24-hour)
|
|
26
|
+
* - \T - time HH:MM:SS (12-hour)
|
|
27
|
+
* - \@ - time HH:MM AM/PM (12-hour)
|
|
28
|
+
* - \A - time HH:MM (24-hour)
|
|
29
|
+
* - \D{format} - strftime format
|
|
30
|
+
* - \s - shell name
|
|
31
|
+
* - \v - bash version (major.minor)
|
|
32
|
+
* - \V - bash version (major.minor.patch)
|
|
33
|
+
* - \j - number of jobs
|
|
34
|
+
* - \l - terminal device basename
|
|
35
|
+
* - \# - command number
|
|
36
|
+
* - \! - history number
|
|
37
|
+
* - \NNN - octal character code
|
|
38
|
+
*/
|
|
39
|
+
export declare function expandPrompt(ctx: InterpreterContext, value: string): string;
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Quoting helpers for word expansion
|
|
3
|
+
*
|
|
4
|
+
* Handles quoting values for shell reuse (${var@Q} transformation).
|
|
5
|
+
*/
|
|
6
|
+
/**
|
|
7
|
+
* Quote a value for safe reuse as shell input (${var@Q} transformation)
|
|
8
|
+
* Uses single quotes with proper escaping for special characters.
|
|
9
|
+
* Follows bash's quoting behavior:
|
|
10
|
+
* - Simple strings without quotes: 'value'
|
|
11
|
+
* - Strings with single quotes or control characters: $'value' with \' escaping
|
|
12
|
+
*/
|
|
13
|
+
export declare function quoteValue(value: string): string;
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Tilde Expansion
|
|
3
|
+
*
|
|
4
|
+
* Functions for handling tilde (~) expansion in word expansion.
|
|
5
|
+
*/
|
|
6
|
+
import type { InterpreterContext } from "../types.js";
|
|
7
|
+
/**
|
|
8
|
+
* Apply tilde expansion to a string.
|
|
9
|
+
* Used after brace expansion to handle cases like ~{/src,root} -> ~/src ~root -> /home/user/src /root
|
|
10
|
+
* Only expands ~ at the start of the string followed by / or end of string.
|
|
11
|
+
*/
|
|
12
|
+
export declare function applyTildeExpansion(ctx: InterpreterContext, value: string): string;
|
|
@@ -0,0 +1,76 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Unquoted Expansion Handlers
|
|
3
|
+
*
|
|
4
|
+
* Handles unquoted positional parameter and array expansions:
|
|
5
|
+
* - Unquoted $@ and $* (with and without prefix/suffix)
|
|
6
|
+
* - Unquoted ${arr[@]} and ${arr[*]}
|
|
7
|
+
* - Unquoted ${@:offset} and ${*:offset} slicing
|
|
8
|
+
* - Unquoted ${@#pattern} and ${*#pattern} pattern removal
|
|
9
|
+
* - Unquoted ${arr[@]/pattern/replacement} pattern replacement
|
|
10
|
+
* - Unquoted ${arr[@]#pattern} pattern removal
|
|
11
|
+
* - Unquoted ${!prefix@} and ${!prefix*} variable name prefix expansion
|
|
12
|
+
* - Unquoted ${!arr[@]} and ${!arr[*]} array keys expansion
|
|
13
|
+
*/
|
|
14
|
+
import type { ArithExpr, WordPart } from "../../ast/types.js";
|
|
15
|
+
import type { InterpreterContext } from "../types.js";
|
|
16
|
+
/**
|
|
17
|
+
* Result type for unquoted expansion handlers.
|
|
18
|
+
* `null` means the handler doesn't apply to this case.
|
|
19
|
+
*/
|
|
20
|
+
export type UnquotedExpansionResult = {
|
|
21
|
+
values: string[];
|
|
22
|
+
quoted: boolean;
|
|
23
|
+
} | null;
|
|
24
|
+
/**
|
|
25
|
+
* Type for expandPart function reference
|
|
26
|
+
*/
|
|
27
|
+
export type ExpandPartFn = (ctx: InterpreterContext, part: WordPart) => Promise<string>;
|
|
28
|
+
/**
|
|
29
|
+
* Type for expandWordPartsAsync function reference
|
|
30
|
+
*/
|
|
31
|
+
export type ExpandWordPartsAsyncFn = (ctx: InterpreterContext, parts: WordPart[]) => Promise<string>;
|
|
32
|
+
/**
|
|
33
|
+
* Type for evaluateArithmetic function
|
|
34
|
+
*/
|
|
35
|
+
export type EvaluateArithmeticFn = (ctx: InterpreterContext, expr: ArithExpr, isExpansionContext?: boolean) => Promise<number>;
|
|
36
|
+
/**
|
|
37
|
+
* Handle unquoted ${array[@]/pattern/replacement} - apply to each element
|
|
38
|
+
* This handles ${array[@]/#/prefix} (prepend) and ${array[@]/%/suffix} (append)
|
|
39
|
+
*/
|
|
40
|
+
export declare function handleUnquotedArrayPatternReplacement(ctx: InterpreterContext, wordParts: WordPart[], expandWordPartsAsync: ExpandWordPartsAsyncFn, expandPart: ExpandPartFn): Promise<UnquotedExpansionResult>;
|
|
41
|
+
/**
|
|
42
|
+
* Handle unquoted ${array[@]#pattern} - apply pattern removal to each element
|
|
43
|
+
* This handles ${array[@]#pattern} (strip shortest prefix), ${array[@]##pattern} (strip longest prefix)
|
|
44
|
+
* ${array[@]%pattern} (strip shortest suffix), ${array[@]%%pattern} (strip longest suffix)
|
|
45
|
+
*/
|
|
46
|
+
export declare function handleUnquotedArrayPatternRemoval(ctx: InterpreterContext, wordParts: WordPart[], expandWordPartsAsync: ExpandWordPartsAsyncFn, expandPart: ExpandPartFn): Promise<UnquotedExpansionResult>;
|
|
47
|
+
/**
|
|
48
|
+
* Handle unquoted ${@#pattern} and ${*#pattern} - apply pattern removal to each positional parameter
|
|
49
|
+
* This handles ${@#pattern} (strip shortest prefix), ${@##pattern} (strip longest prefix)
|
|
50
|
+
* ${@%pattern} (strip shortest suffix), ${@%%pattern} (strip longest suffix)
|
|
51
|
+
*/
|
|
52
|
+
export declare function handleUnquotedPositionalPatternRemoval(ctx: InterpreterContext, wordParts: WordPart[], expandWordPartsAsync: ExpandWordPartsAsyncFn, expandPart: ExpandPartFn): Promise<UnquotedExpansionResult>;
|
|
53
|
+
/**
|
|
54
|
+
* Handle unquoted ${@:offset} and ${*:offset} (with potential prefix/suffix)
|
|
55
|
+
*/
|
|
56
|
+
export declare function handleUnquotedPositionalSlicing(ctx: InterpreterContext, wordParts: WordPart[], evaluateArithmetic: EvaluateArithmeticFn, expandPart: ExpandPartFn): Promise<UnquotedExpansionResult>;
|
|
57
|
+
/**
|
|
58
|
+
* Handle unquoted $@ and $* (simple, without operations)
|
|
59
|
+
*/
|
|
60
|
+
export declare function handleUnquotedSimplePositional(ctx: InterpreterContext, wordParts: WordPart[]): Promise<UnquotedExpansionResult>;
|
|
61
|
+
/**
|
|
62
|
+
* Handle unquoted ${arr[@]} and ${arr[*]} (without operations)
|
|
63
|
+
*/
|
|
64
|
+
export declare function handleUnquotedSimpleArray(ctx: InterpreterContext, wordParts: WordPart[]): Promise<UnquotedExpansionResult>;
|
|
65
|
+
/**
|
|
66
|
+
* Handle unquoted ${!prefix@} and ${!prefix*} (variable name prefix expansion)
|
|
67
|
+
*/
|
|
68
|
+
export declare function handleUnquotedVarNamePrefix(ctx: InterpreterContext, wordParts: WordPart[]): UnquotedExpansionResult;
|
|
69
|
+
/**
|
|
70
|
+
* Handle unquoted ${!arr[@]} and ${!arr[*]} (array keys/indices expansion)
|
|
71
|
+
*/
|
|
72
|
+
export declare function handleUnquotedArrayKeys(ctx: InterpreterContext, wordParts: WordPart[]): UnquotedExpansionResult;
|
|
73
|
+
/**
|
|
74
|
+
* Handle unquoted $@ or $* with prefix/suffix (e.g., =$@= or =$*=)
|
|
75
|
+
*/
|
|
76
|
+
export declare function handleUnquotedPositionalWithPrefixSuffix(ctx: InterpreterContext, wordParts: WordPart[], expandPart: ExpandPartFn): Promise<UnquotedExpansionResult>;
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Variable Attributes
|
|
3
|
+
*
|
|
4
|
+
* Functions for getting variable attributes (${var@a} transformation).
|
|
5
|
+
*/
|
|
6
|
+
import type { InterpreterContext } from "../types.js";
|
|
7
|
+
/**
|
|
8
|
+
* Get the attributes of a variable for ${var@a} transformation.
|
|
9
|
+
* Returns a string with attribute flags (e.g., "ar" for readonly array).
|
|
10
|
+
*
|
|
11
|
+
* Attribute flags (in order):
|
|
12
|
+
* - a: indexed array
|
|
13
|
+
* - A: associative array
|
|
14
|
+
* - i: integer
|
|
15
|
+
* - n: nameref
|
|
16
|
+
* - r: readonly
|
|
17
|
+
* - x: exported
|
|
18
|
+
*/
|
|
19
|
+
export declare function getVariableAttributes(ctx: InterpreterContext, name: string): string;
|
|
@@ -6,12 +6,14 @@
|
|
|
6
6
|
* - Array access (${arr[0]}, ${arr[@]}, ${arr[*]})
|
|
7
7
|
* - Positional parameters ($1, $2, ...)
|
|
8
8
|
* - Regular variables
|
|
9
|
+
* - Nameref resolution
|
|
9
10
|
*/
|
|
10
11
|
import type { InterpreterContext } from "../types.js";
|
|
11
12
|
/**
|
|
12
13
|
* Get all elements of an array stored as arrayName_0, arrayName_1, etc.
|
|
13
14
|
* Returns an array of [index/key, value] tuples, sorted by index/key.
|
|
14
15
|
* For associative arrays, uses string keys.
|
|
16
|
+
* Special arrays FUNCNAME, BASH_LINENO, and BASH_SOURCE are handled dynamically from call stack.
|
|
15
17
|
*/
|
|
16
18
|
export declare function getArrayElements(ctx: InterpreterContext, arrayName: string): Array<[number | string, string]>;
|
|
17
19
|
/**
|
|
@@ -24,4 +26,11 @@ export declare function isArray(ctx: InterpreterContext, name: string): boolean;
|
|
|
24
26
|
* @param name - The variable name
|
|
25
27
|
* @param checkNounset - Whether to check for nounset (default true)
|
|
26
28
|
*/
|
|
27
|
-
export declare function getVariable(ctx: InterpreterContext, name: string, checkNounset?: boolean, _insideDoubleQuotes?: boolean): string
|
|
29
|
+
export declare function getVariable(ctx: InterpreterContext, name: string, checkNounset?: boolean, _insideDoubleQuotes?: boolean): Promise<string>;
|
|
30
|
+
/**
|
|
31
|
+
* Check if a variable is set (exists in the environment).
|
|
32
|
+
* Properly handles array subscripts (e.g., arr[0] -> arr_0).
|
|
33
|
+
* @param ctx - The interpreter context
|
|
34
|
+
* @param name - The variable name (possibly with array subscript)
|
|
35
|
+
*/
|
|
36
|
+
export declare function isVariableSet(ctx: InterpreterContext, name: string): Promise<boolean>;
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Word Expansion with Glob Handling
|
|
3
|
+
*
|
|
4
|
+
* Handles the main word expansion flow including:
|
|
5
|
+
* - Brace expansion
|
|
6
|
+
* - Array and positional parameter expansion
|
|
7
|
+
* - Word splitting
|
|
8
|
+
* - Glob/pathname expansion
|
|
9
|
+
*/
|
|
10
|
+
import type { ArithExpr, ParameterExpansionPart, WordNode, WordPart } from "../../ast/types.js";
|
|
11
|
+
import type { InterpreterContext } from "../types.js";
|
|
12
|
+
/**
|
|
13
|
+
* Dependencies injected to avoid circular imports
|
|
14
|
+
*/
|
|
15
|
+
export interface WordGlobExpansionDeps {
|
|
16
|
+
expandWordAsync: (ctx: InterpreterContext, word: WordNode) => Promise<string>;
|
|
17
|
+
expandWordForGlobbing: (ctx: InterpreterContext, word: WordNode) => Promise<string>;
|
|
18
|
+
expandWordWithBracesAsync: (ctx: InterpreterContext, word: WordNode) => Promise<string[] | null>;
|
|
19
|
+
expandWordPartsAsync: (ctx: InterpreterContext, parts: WordPart[]) => Promise<string>;
|
|
20
|
+
expandPart: (ctx: InterpreterContext, part: WordPart, inDoubleQuotes?: boolean) => Promise<string>;
|
|
21
|
+
expandParameterAsync: (ctx: InterpreterContext, part: ParameterExpansionPart, inDoubleQuotes?: boolean) => Promise<string>;
|
|
22
|
+
hasBraceExpansion: (parts: WordPart[]) => boolean;
|
|
23
|
+
evaluateArithmetic: (ctx: InterpreterContext, expr: ArithExpr, isExpansionContext?: boolean) => Promise<number>;
|
|
24
|
+
buildIfsCharClassPattern: (ifsChars: string) => string;
|
|
25
|
+
smartWordSplit: (ctx: InterpreterContext, wordParts: WordPart[], ifsChars: string, ifsPattern: string, expandPart: (ctx: InterpreterContext, part: WordPart) => Promise<string>) => Promise<string[]>;
|
|
26
|
+
}
|
|
27
|
+
/**
|
|
28
|
+
* Main word expansion function that handles all expansion types and glob matching.
|
|
29
|
+
*/
|
|
30
|
+
export declare function expandWordWithGlobImpl(ctx: InterpreterContext, word: WordNode, deps: WordGlobExpansionDeps): Promise<{
|
|
31
|
+
values: string[];
|
|
32
|
+
quoted: boolean;
|
|
33
|
+
}>;
|
|
@@ -10,15 +10,20 @@ import type { InterpreterContext } from "../types.js";
|
|
|
10
10
|
*/
|
|
11
11
|
export type ExpandPartFn = (ctx: InterpreterContext, part: WordPart) => Promise<string>;
|
|
12
12
|
/**
|
|
13
|
-
* Smart word splitting
|
|
13
|
+
* Smart word splitting for words containing expansions.
|
|
14
14
|
*
|
|
15
|
-
*
|
|
16
|
-
*
|
|
15
|
+
* In bash, word splitting respects quoted parts. When you have:
|
|
16
|
+
* - $a"$b" where a="1 2" and b="3 4"
|
|
17
|
+
* - The unquoted $a gets split by IFS: "1 2" -> ["1", "2"]
|
|
18
|
+
* - The quoted "$b" does NOT get split, it joins with the last field from $a
|
|
19
|
+
* - Result: ["1", "23 4"] (the "2" joins with "3 4")
|
|
20
|
+
*
|
|
21
|
+
* This differs from pure literal words which are never IFS-split.
|
|
17
22
|
*
|
|
18
23
|
* @param ctx - Interpreter context
|
|
19
24
|
* @param wordParts - Word parts to expand and split
|
|
20
|
-
* @param
|
|
21
|
-
* @param ifsPattern - Regex-escaped IFS pattern for splitting
|
|
25
|
+
* @param ifsChars - IFS characters for proper whitespace/non-whitespace handling
|
|
26
|
+
* @param ifsPattern - Regex-escaped IFS pattern for checking if splitting is needed
|
|
22
27
|
* @param expandPartFn - Function to expand individual parts (injected to avoid circular deps)
|
|
23
28
|
*/
|
|
24
|
-
export declare function smartWordSplit(ctx: InterpreterContext, wordParts: WordPart[],
|
|
29
|
+
export declare function smartWordSplit(ctx: InterpreterContext, wordParts: WordPart[], ifsChars: string, _ifsPattern: string, expandPartFn: ExpandPartFn): Promise<string[]>;
|
|
@@ -11,17 +11,43 @@
|
|
|
11
11
|
*/
|
|
12
12
|
import type { WordNode } from "../ast/types.js";
|
|
13
13
|
import type { InterpreterContext } from "./types.js";
|
|
14
|
-
export {
|
|
14
|
+
export { escapeGlobChars, escapeRegexChars } from "./expansion/glob-escape.js";
|
|
15
|
+
export { getArrayElements, getVariable, isArray, } from "./expansion/variable.js";
|
|
15
16
|
/**
|
|
16
17
|
* Check if an entire word is fully quoted
|
|
17
18
|
*/
|
|
18
19
|
export declare function isWordFullyQuoted(word: WordNode): boolean;
|
|
20
|
+
export declare function expandWord(ctx: InterpreterContext, word: WordNode): Promise<string>;
|
|
19
21
|
/**
|
|
20
|
-
*
|
|
22
|
+
* Expand a word for use as a regex pattern (in [[ =~ ]]).
|
|
23
|
+
* Preserves backslash escapes so they're passed to the regex engine.
|
|
24
|
+
* For example, \[\] becomes \[\] in the regex (matching literal [ and ]).
|
|
21
25
|
*/
|
|
22
|
-
export declare function
|
|
23
|
-
|
|
26
|
+
export declare function expandWordForRegex(ctx: InterpreterContext, word: WordNode): Promise<string>;
|
|
27
|
+
/**
|
|
28
|
+
* Expand a word for use as a pattern (e.g., in [[ == ]] or case).
|
|
29
|
+
* Preserves backslash escapes for pattern metacharacters so they're treated literally.
|
|
30
|
+
* This prevents `*\(\)` from being interpreted as an extglob pattern.
|
|
31
|
+
*/
|
|
32
|
+
export declare function expandWordForPattern(ctx: InterpreterContext, word: WordNode): Promise<string>;
|
|
24
33
|
export declare function expandWordWithGlob(ctx: InterpreterContext, word: WordNode): Promise<{
|
|
25
34
|
values: string[];
|
|
26
35
|
quoted: boolean;
|
|
27
36
|
}>;
|
|
37
|
+
export declare function hasQuotedMultiValueAt(ctx: InterpreterContext, word: WordNode): boolean;
|
|
38
|
+
/**
|
|
39
|
+
* Expand a redirect target with glob handling.
|
|
40
|
+
*
|
|
41
|
+
* For redirects:
|
|
42
|
+
* - If glob matches 0 files with failglob → error (returns { error: ... })
|
|
43
|
+
* - If glob matches 0 files without failglob → use literal pattern
|
|
44
|
+
* - If glob matches 1 file → use that file
|
|
45
|
+
* - If glob matches 2+ files → "ambiguous redirect" error
|
|
46
|
+
*
|
|
47
|
+
* Returns { target: string } on success or { error: string } on failure.
|
|
48
|
+
*/
|
|
49
|
+
export declare function expandRedirectTarget(ctx: InterpreterContext, word: WordNode): Promise<{
|
|
50
|
+
target: string;
|
|
51
|
+
} | {
|
|
52
|
+
error: string;
|
|
53
|
+
}>;
|
|
@@ -9,4 +9,4 @@ import type { FunctionDefNode } from "../ast/types.js";
|
|
|
9
9
|
import type { ExecResult } from "../types.js";
|
|
10
10
|
import type { InterpreterContext } from "./types.js";
|
|
11
11
|
export declare function executeFunctionDef(ctx: InterpreterContext, node: FunctionDefNode): ExecResult;
|
|
12
|
-
export declare function callFunction(ctx: InterpreterContext, func: FunctionDefNode, args: string[]): Promise<ExecResult>;
|
|
12
|
+
export declare function callFunction(ctx: InterpreterContext, func: FunctionDefNode, args: string[], stdin?: string, callLine?: number): Promise<ExecResult>;
|
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* Array helper functions for the interpreter.
|
|
3
3
|
*/
|
|
4
|
+
import type { WordNode } from "../../ast/types.js";
|
|
4
5
|
import type { InterpreterContext } from "../types.js";
|
|
5
6
|
/**
|
|
6
7
|
* Get all indices of an array, sorted in ascending order.
|
|
@@ -21,3 +22,22 @@ export declare function getAssocArrayKeys(ctx: InterpreterContext, arrayName: st
|
|
|
21
22
|
* Handles 'key' and "key" → key
|
|
22
23
|
*/
|
|
23
24
|
export declare function unquoteKey(key: string): string;
|
|
25
|
+
/**
|
|
26
|
+
* Parse a keyed array element from an AST WordNode like [key]=value or [key]+=value.
|
|
27
|
+
* Returns { key, valueParts, append } where valueParts are the AST parts for the value.
|
|
28
|
+
* Returns null if not a keyed element pattern.
|
|
29
|
+
*
|
|
30
|
+
* This is used to properly expand variables in the value part of keyed elements.
|
|
31
|
+
*/
|
|
32
|
+
export interface ParsedKeyedElement {
|
|
33
|
+
key: string;
|
|
34
|
+
valueParts: WordNode["parts"];
|
|
35
|
+
append: boolean;
|
|
36
|
+
}
|
|
37
|
+
export declare function parseKeyedElementFromWord(word: WordNode): ParsedKeyedElement | null;
|
|
38
|
+
/**
|
|
39
|
+
* Extract literal string content from a Word node (without expansion).
|
|
40
|
+
* This is used for parsing associative array element syntax like [key]=value
|
|
41
|
+
* where the [key] part may be parsed as a Glob.
|
|
42
|
+
*/
|
|
43
|
+
export declare function wordToLiteralString(word: WordNode): string;
|
|
@@ -15,6 +15,13 @@ export declare function getIfs(env: Record<string, string>): string;
|
|
|
15
15
|
* Check if IFS is set to empty string (disables word splitting).
|
|
16
16
|
*/
|
|
17
17
|
export declare function isIfsEmpty(env: Record<string, string>): boolean;
|
|
18
|
+
/**
|
|
19
|
+
* Check if IFS contains only whitespace characters (space, tab, newline).
|
|
20
|
+
* This affects how empty fields are handled in $@ and $* expansion.
|
|
21
|
+
* When IFS has non-whitespace chars, empty params are preserved.
|
|
22
|
+
* When IFS has only whitespace, empty params are dropped.
|
|
23
|
+
*/
|
|
24
|
+
export declare function isIfsWhitespaceOnly(env: Record<string, string>): boolean;
|
|
18
25
|
/**
|
|
19
26
|
* Build a regex-safe pattern from IFS characters for use in character classes.
|
|
20
27
|
* E.g., for IFS=" \t\n", returns " \\t\\n" (escaped for [pattern] use)
|
|
@@ -26,18 +33,72 @@ export declare function buildIfsCharClassPattern(ifs: string): string;
|
|
|
26
33
|
*/
|
|
27
34
|
export declare function getIfsSeparator(env: Record<string, string>): string;
|
|
28
35
|
/**
|
|
29
|
-
*
|
|
30
|
-
*
|
|
36
|
+
* Advanced IFS splitting for the read builtin with proper whitespace/non-whitespace handling.
|
|
37
|
+
*
|
|
38
|
+
* IFS has two types of characters:
|
|
39
|
+
* - Whitespace (space, tab, newline): Multiple consecutive ones are collapsed,
|
|
40
|
+
* leading/trailing are stripped
|
|
41
|
+
* - Non-whitespace (like 'x', ':'): Create empty fields when consecutive,
|
|
42
|
+
* trailing ones preserved (except the final delimiter)
|
|
31
43
|
*
|
|
32
44
|
* @param value - String to split
|
|
33
45
|
* @param ifs - IFS characters to split on
|
|
46
|
+
* @param maxSplit - Maximum number of splits (for read with multiple vars, the last gets the rest)
|
|
47
|
+
* @param raw - If true, backslash escaping is disabled (like read -r)
|
|
34
48
|
* @returns Object with words array and wordStarts array
|
|
35
49
|
*/
|
|
36
|
-
export declare function
|
|
50
|
+
export declare function splitByIfsForRead(value: string, ifs: string, maxSplit?: number, raw?: boolean): {
|
|
37
51
|
words: string[];
|
|
38
52
|
wordStarts: number[];
|
|
39
53
|
};
|
|
40
54
|
/**
|
|
41
|
-
*
|
|
55
|
+
* IFS splitting for word expansion (unquoted $VAR, $*, etc.).
|
|
56
|
+
*
|
|
57
|
+
* Key differences from splitByIfsForRead:
|
|
58
|
+
* - Trailing non-whitespace delimiter does NOT create an empty field
|
|
59
|
+
* - No maxSplit concept (always splits fully)
|
|
60
|
+
* - No backslash escape handling
|
|
61
|
+
*
|
|
62
|
+
* @param value - String to split
|
|
63
|
+
* @param ifs - IFS characters to split on
|
|
64
|
+
* @returns Array of words after splitting
|
|
65
|
+
*/
|
|
66
|
+
/**
|
|
67
|
+
* Result of splitByIfsForExpansionEx with leading/trailing delimiter info.
|
|
68
|
+
*/
|
|
69
|
+
export interface IfsExpansionSplitResult {
|
|
70
|
+
words: string[];
|
|
71
|
+
/** True if the value started with an IFS whitespace delimiter (affects joining with preceding text) */
|
|
72
|
+
hadLeadingDelimiter: boolean;
|
|
73
|
+
/** True if the value ended with an IFS delimiter (affects joining with subsequent text) */
|
|
74
|
+
hadTrailingDelimiter: boolean;
|
|
75
|
+
}
|
|
76
|
+
/**
|
|
77
|
+
* Extended IFS splitting that tracks trailing delimiters.
|
|
78
|
+
* This is needed for proper word boundary handling when literal text follows an expansion.
|
|
79
|
+
* For example, in `-$x-` where `x='a b c '`, the trailing space means the final `-`
|
|
80
|
+
* should become a separate word, not join with `c`.
|
|
81
|
+
*/
|
|
82
|
+
export declare function splitByIfsForExpansionEx(value: string, ifs: string): IfsExpansionSplitResult;
|
|
83
|
+
export declare function splitByIfsForExpansion(value: string, ifs: string): string[];
|
|
84
|
+
/**
|
|
85
|
+
* Strip trailing IFS from the last variable in read builtin.
|
|
86
|
+
*
|
|
87
|
+
* Bash behavior:
|
|
88
|
+
* 1. Strip trailing IFS whitespace characters (but NOT if they're escaped by backslash)
|
|
89
|
+
* 2. If there's a single trailing IFS non-whitespace character, strip it ONLY IF
|
|
90
|
+
* there are no other non-ws IFS chars in the content (excluding the trailing one)
|
|
91
|
+
*
|
|
92
|
+
* Examples with IFS="x ":
|
|
93
|
+
* - "ax " -> "a" (trailing spaces stripped, then trailing single x stripped because no other x)
|
|
94
|
+
* - "ax" -> "a" (trailing single x stripped because no other x in remaining content)
|
|
95
|
+
* - "axx" -> "axx" (two trailing x's, so don't strip - there's another x)
|
|
96
|
+
* - "ax x" -> "ax x" (trailing x NOT stripped because there's an x earlier)
|
|
97
|
+
* - "bx" -> "b" (trailing x stripped, no other x)
|
|
98
|
+
* - "a\ " -> "a " (backslash-escaped space is NOT stripped)
|
|
99
|
+
*
|
|
100
|
+
* @param value - String to strip (raw, before backslash processing)
|
|
101
|
+
* @param ifs - IFS characters
|
|
102
|
+
* @param raw - If true, backslash escaping is disabled
|
|
42
103
|
*/
|
|
43
|
-
export declare function
|
|
104
|
+
export declare function stripTrailingIfsWhitespace(value: string, ifs: string, raw?: boolean): string;
|
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Nameref (declare -n) support
|
|
3
|
+
*
|
|
4
|
+
* Namerefs are variables that reference other variables by name.
|
|
5
|
+
* When a nameref is accessed, it transparently dereferences to the target variable.
|
|
6
|
+
*/
|
|
7
|
+
import type { InterpreterContext } from "../types.js";
|
|
8
|
+
/**
|
|
9
|
+
* Check if a variable is a nameref
|
|
10
|
+
*/
|
|
11
|
+
export declare function isNameref(ctx: InterpreterContext, name: string): boolean;
|
|
12
|
+
/**
|
|
13
|
+
* Mark a variable as a nameref
|
|
14
|
+
*/
|
|
15
|
+
export declare function markNameref(ctx: InterpreterContext, name: string): void;
|
|
16
|
+
/**
|
|
17
|
+
* Remove the nameref attribute from a variable
|
|
18
|
+
*/
|
|
19
|
+
export declare function unmarkNameref(ctx: InterpreterContext, name: string): void;
|
|
20
|
+
/**
|
|
21
|
+
* Mark a nameref as having an "invalid" target at creation time.
|
|
22
|
+
* Invalid namerefs always read/write their value directly, never resolving.
|
|
23
|
+
*/
|
|
24
|
+
export declare function markNamerefInvalid(ctx: InterpreterContext, name: string): void;
|
|
25
|
+
/**
|
|
26
|
+
* Mark a nameref as "bound" - meaning its target existed at creation time.
|
|
27
|
+
* This is kept for tracking purposes but is currently not used in resolution.
|
|
28
|
+
*/
|
|
29
|
+
export declare function markNamerefBound(ctx: InterpreterContext, name: string): void;
|
|
30
|
+
/**
|
|
31
|
+
* Check if a name refers to a valid, existing variable or array element.
|
|
32
|
+
* Used to determine if a nameref target is "real" or just a stored value.
|
|
33
|
+
*/
|
|
34
|
+
export declare function targetExists(ctx: InterpreterContext, target: string): boolean;
|
|
35
|
+
/**
|
|
36
|
+
* Resolve a nameref chain to the final variable name.
|
|
37
|
+
* Returns the original name if it's not a nameref.
|
|
38
|
+
* Detects circular references and returns undefined.
|
|
39
|
+
*
|
|
40
|
+
* @param ctx - The interpreter context
|
|
41
|
+
* @param name - The variable name to resolve
|
|
42
|
+
* @param maxDepth - Maximum chain depth to prevent infinite loops (default 100)
|
|
43
|
+
* @returns The resolved variable name, or undefined if circular reference detected
|
|
44
|
+
*/
|
|
45
|
+
export declare function resolveNameref(ctx: InterpreterContext, name: string, maxDepth?: number): string | undefined;
|
|
46
|
+
/**
|
|
47
|
+
* Get the target name of a nameref (what it points to).
|
|
48
|
+
* Returns the variable's value if it's a nameref, undefined otherwise.
|
|
49
|
+
*/
|
|
50
|
+
export declare function getNamerefTarget(ctx: InterpreterContext, name: string): string | undefined;
|
|
51
|
+
/**
|
|
52
|
+
* Resolve a nameref for assignment purposes.
|
|
53
|
+
* Unlike resolveNameref, this will resolve to the target variable name
|
|
54
|
+
* even if the target doesn't exist yet (allowing creation).
|
|
55
|
+
*
|
|
56
|
+
* @param ctx - The interpreter context
|
|
57
|
+
* @param name - The variable name to resolve
|
|
58
|
+
* @param valueBeingAssigned - The value being assigned (needed for empty nameref handling)
|
|
59
|
+
* @param maxDepth - Maximum chain depth to prevent infinite loops
|
|
60
|
+
* @returns
|
|
61
|
+
* - undefined if circular reference detected
|
|
62
|
+
* - null if the nameref is empty and value is not an existing variable (skip assignment)
|
|
63
|
+
* - The resolved target name otherwise (may be the nameref itself if target is invalid)
|
|
64
|
+
*/
|
|
65
|
+
export declare function resolveNamerefForAssignment(ctx: InterpreterContext, name: string, valueBeingAssigned?: string, maxDepth?: number): string | null | undefined;
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Shell value quoting utilities
|
|
3
|
+
*
|
|
4
|
+
* Provides functions for quoting values in shell output format,
|
|
5
|
+
* used by both `set` and `declare/typeset` builtins.
|
|
6
|
+
*/
|
|
7
|
+
/**
|
|
8
|
+
* Quote a value for shell output (used by 'set' and 'typeset' with no args)
|
|
9
|
+
* Matches bash's output format:
|
|
10
|
+
* - No quotes for simple alphanumeric values
|
|
11
|
+
* - Single quotes for values with spaces or shell metacharacters
|
|
12
|
+
* - $'...' quoting for values with control characters
|
|
13
|
+
*/
|
|
14
|
+
export declare function quoteValue(value: string): string;
|
|
15
|
+
/**
|
|
16
|
+
* Quote a value for array element output
|
|
17
|
+
* Uses $'...' for control characters, double quotes otherwise
|
|
18
|
+
*/
|
|
19
|
+
export declare function quoteArrayValue(value: string): string;
|
|
20
|
+
/**
|
|
21
|
+
* Quote a value for declare -p output
|
|
22
|
+
* Uses $'...' for control characters, double quotes otherwise
|
|
23
|
+
*/
|
|
24
|
+
export declare function quoteDeclareValue(value: string): string;
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* Readonly variable helpers.
|
|
2
|
+
* Readonly and export variable helpers.
|
|
3
3
|
*
|
|
4
|
-
* Consolidates readonly variable logic used in declare, export, local, etc.
|
|
4
|
+
* Consolidates readonly and export variable logic used in declare, export, local, etc.
|
|
5
5
|
*/
|
|
6
6
|
import type { ExecResult } from "../../types.js";
|
|
7
7
|
import type { InterpreterContext } from "../types.js";
|
|
@@ -10,12 +10,36 @@ import type { InterpreterContext } from "../types.js";
|
|
|
10
10
|
*/
|
|
11
11
|
export declare function markReadonly(ctx: InterpreterContext, name: string): void;
|
|
12
12
|
/**
|
|
13
|
-
* Check if a variable is readonly
|
|
13
|
+
* Check if a variable is readonly.
|
|
14
|
+
*/
|
|
15
|
+
export declare function isReadonly(ctx: InterpreterContext, name: string): boolean;
|
|
16
|
+
/**
|
|
17
|
+
* Check if a variable is readonly and throw an error if so.
|
|
14
18
|
* Returns null if the variable is not readonly (can be modified).
|
|
15
19
|
*
|
|
20
|
+
* Assigning to a readonly variable is a fatal error that stops script execution.
|
|
21
|
+
* This matches the behavior of dash, mksh, ash, and bash in POSIX mode.
|
|
22
|
+
* (Note: bash in non-POSIX mode has a bug where multi-line readonly assignment
|
|
23
|
+
* continues execution, but one-line still stops. We always stop.)
|
|
24
|
+
*
|
|
16
25
|
* @param ctx - Interpreter context
|
|
17
26
|
* @param name - Variable name
|
|
18
27
|
* @param command - Command name for error message (default: "bash")
|
|
19
|
-
* @returns
|
|
28
|
+
* @returns null if variable is not readonly (can be modified)
|
|
29
|
+
* @throws ExitError if variable is readonly
|
|
20
30
|
*/
|
|
21
31
|
export declare function checkReadonlyError(ctx: InterpreterContext, name: string, command?: string): ExecResult | null;
|
|
32
|
+
/**
|
|
33
|
+
* Mark a variable as exported.
|
|
34
|
+
*
|
|
35
|
+
* If we're inside a local scope and the variable is local (exists in the
|
|
36
|
+
* current scope), track it as a locally-exported variable. When the scope
|
|
37
|
+
* is popped, the export attribute will be removed if it wasn't exported
|
|
38
|
+
* before entering the function.
|
|
39
|
+
*/
|
|
40
|
+
export declare function markExported(ctx: InterpreterContext, name: string): void;
|
|
41
|
+
/**
|
|
42
|
+
* Remove the export attribute from a variable.
|
|
43
|
+
* The variable value is preserved, just no longer exported to child processes.
|
|
44
|
+
*/
|
|
45
|
+
export declare function unmarkExported(ctx: InterpreterContext, name: string): void;
|