Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 1 | #------------------------------------------------------------------------------ |
| 2 | # Detect broken &&-chains in tests. |
| 3 | # |
| 4 | # At present, only &&-chains in subshells are examined by this linter; |
| 5 | # top-level &&-chains are instead checked directly by the test framework. Like |
| 6 | # the top-level &&-chain linter, the subshell linter (intentionally) does not |
| 7 | # check &&-chains within {...} blocks. |
| 8 | # |
| 9 | # Checking for &&-chain breakage is done line-by-line by pure textual |
| 10 | # inspection. |
| 11 | # |
| 12 | # Incomplete lines (those ending with "\") are stitched together with following |
| 13 | # lines to simplify processing, particularly of "one-liner" statements. |
| 14 | # Top-level here-docs are swallowed to avoid false positives within the |
| 15 | # here-doc body, although the statement to which the here-doc is attached is |
| 16 | # retained. |
| 17 | # |
| 18 | # Heuristics are used to detect end-of-subshell when the closing ")" is cuddled |
| 19 | # with the final subshell statement on the same line: |
| 20 | # |
| 21 | # (cd foo && |
| 22 | # bar) |
| 23 | # |
| 24 | # in order to avoid misinterpreting the ")" in constructs such as "x=$(...)" |
| 25 | # and "case $x in *)" as ending the subshell. |
| 26 | # |
Eric Sunshine | 0d71317 | 2021-12-13 01:30:53 -0500 | [diff] [blame] | 27 | # Lines missing a final "&&" are flagged with "?!AMP?!", as are lines which |
| 28 | # chain commands with ";" internally rather than "&&". A line may be flagged |
| 29 | # for both violations. |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 30 | # |
| 31 | # Detection of a missing &&-link in a multi-line subshell is complicated by the |
| 32 | # fact that the last statement before the closing ")" must not end with "&&". |
| 33 | # Since processing is line-by-line, it is not known whether a missing "&&" is |
| 34 | # legitimate or not until the _next_ line is seen. To accommodate this, within |
| 35 | # multi-line subshells, each line is stored in sed's "hold" area until after |
| 36 | # the next line is seen and processed. If the next line is a stand-alone ")", |
| 37 | # then a missing "&&" on the previous line is legitimate; otherwise a missing |
| 38 | # "&&" is a break in the &&-chain. |
| 39 | # |
| 40 | # ( |
| 41 | # cd foo && |
| 42 | # bar |
| 43 | # ) |
| 44 | # |
| 45 | # In practical terms, when "bar" is encountered, it is flagged with "?!AMP?!", |
| 46 | # but when the stand-alone ")" line is seen which closes the subshell, the |
| 47 | # "?!AMP?!" violation is removed from the "bar" line (retrieved from the "hold" |
| 48 | # area) since the final statement of a subshell must not end with "&&". The |
| 49 | # final line of a subshell may still break the &&-chain by using ";" internally |
Eric Sunshine | 0d71317 | 2021-12-13 01:30:53 -0500 | [diff] [blame] | 50 | # to chain commands together rather than "&&", but an internal "?!AMP?!" is |
| 51 | # never removed from a line even though a line-ending "?!AMP?!" might be. |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 52 | # |
| 53 | # Care is taken to recognize the last _statement_ of a multi-line subshell, not |
| 54 | # necessarily the last textual _line_ within the subshell, since &&-chaining |
| 55 | # applies to statements, not to lines. Consequently, blank lines, comment |
| 56 | # lines, and here-docs are swallowed (but not the command to which the here-doc |
| 57 | # is attached), leaving the last statement in the "hold" area, not the last |
| 58 | # line, thus simplifying &&-link checking. |
| 59 | # |
| 60 | # The final statement before "done" in for- and while-loops, and before "elif", |
| 61 | # "else", and "fi" in if-then-else likewise must not end with "&&", thus |
| 62 | # receives similar treatment. |
| 63 | # |
Eric Sunshine | c2c29cc | 2018-08-13 04:47:34 -0400 | [diff] [blame] | 64 | # Swallowing here-docs with arbitrary tags requires a bit of finesse. When a |
Eric Sunshine | 34ba05c | 2021-12-13 01:30:57 -0500 | [diff] [blame] | 65 | # line such as "cat <<EOF" is seen, the here-doc tag is copied to the front of |
| 66 | # the line enclosed in angle brackets as a sentinel, giving "<EOF>cat <<EOF". |
Eric Sunshine | c2c29cc | 2018-08-13 04:47:34 -0400 | [diff] [blame] | 67 | # As each subsequent line is read, it is appended to the target line and a |
| 68 | # (whitespace-loose) back-reference match /^<(.*)>\n\1$/ is attempted to see if |
| 69 | # the content inside "<...>" matches the entirety of the newly-read line. For |
| 70 | # instance, if the next line read is "some data", when concatenated with the |
Eric Sunshine | 34ba05c | 2021-12-13 01:30:57 -0500 | [diff] [blame] | 71 | # target line, it becomes "<EOF>cat <<EOF\nsome data", and a match is attempted |
Eric Sunshine | c2c29cc | 2018-08-13 04:47:34 -0400 | [diff] [blame] | 72 | # to see if "EOF" matches "some data". Since it doesn't, the next line is |
| 73 | # attempted. When a line consisting of only "EOF" (and possible whitespace) is |
Eric Sunshine | 34ba05c | 2021-12-13 01:30:57 -0500 | [diff] [blame] | 74 | # encountered, it is appended to the target line giving "<EOF>cat <<EOF\nEOF", |
Eric Sunshine | c2c29cc | 2018-08-13 04:47:34 -0400 | [diff] [blame] | 75 | # in which case the "EOF" inside "<...>" does match the text following the |
| 76 | # newline, thus the closing here-doc tag has been found. The closing tag line |
| 77 | # and the "<...>" prefix on the target line are then discarded, leaving just |
Eric Sunshine | 34ba05c | 2021-12-13 01:30:57 -0500 | [diff] [blame] | 78 | # the target line "cat <<EOF". |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 79 | #------------------------------------------------------------------------------ |
| 80 | |
| 81 | # incomplete line -- slurp up next line |
| 82 | :squash |
| 83 | /\\$/ { |
Eric Sunshine | ace64e5 | 2018-07-31 01:03:20 -0400 | [diff] [blame] | 84 | N |
| 85 | s/\\\n// |
| 86 | bsquash |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 87 | } |
| 88 | |
| 89 | # here-doc -- swallow it to avoid false hits within its body (but keep the |
| 90 | # command to which it was attached) |
Eric Sunshine | 2d53614 | 2021-12-13 01:30:55 -0500 | [diff] [blame] | 91 | /<<-*[ ]*[\\'"]*[A-Za-z0-9_]/ { |
Eric Sunshine | 22597af | 2021-12-13 01:30:56 -0500 | [diff] [blame] | 92 | /"[^"]*<<[^"]*"/bnotdoc |
Eric Sunshine | 34ba05c | 2021-12-13 01:30:57 -0500 | [diff] [blame] | 93 | s/^\(.*<<-*[ ]*\)[\\'"]*\([A-Za-z0-9_][A-Za-z0-9_]*\)['"]*/<\2>\1\2/ |
Ævar Arnfjörð Bjarmason | a3c4c88 | 2018-08-24 15:20:14 +0000 | [diff] [blame] | 94 | :hered |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 95 | N |
Eric Sunshine | c2c29cc | 2018-08-13 04:47:34 -0400 | [diff] [blame] | 96 | /^<\([^>]*\)>.*\n[ ]*\1[ ]*$/!{ |
| 97 | s/\n.*$// |
Ævar Arnfjörð Bjarmason | a3c4c88 | 2018-08-24 15:20:14 +0000 | [diff] [blame] | 98 | bhered |
Eric Sunshine | c2c29cc | 2018-08-13 04:47:34 -0400 | [diff] [blame] | 99 | } |
| 100 | s/^<[^>]*>// |
| 101 | s/\n.*$// |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 102 | } |
Eric Sunshine | 22597af | 2021-12-13 01:30:56 -0500 | [diff] [blame] | 103 | :notdoc |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 104 | |
| 105 | # one-liner "(...) &&" |
| 106 | /^[ ]*!*[ ]*(..*)[ ]*&&[ ]*$/boneline |
| 107 | |
| 108 | # same as above but without trailing "&&" |
| 109 | /^[ ]*!*[ ]*(..*)[ ]*$/boneline |
| 110 | |
| 111 | # one-liner "(...) >x" (or "2>x" or "<x" or "|x" or "&" |
| 112 | /^[ ]*!*[ ]*(..*)[ ]*[0-9]*[<>|&]/boneline |
| 113 | |
| 114 | # multi-line "(...\n...)" |
Kyohei Kadota | b3b753b | 2020-09-10 02:17:41 +0000 | [diff] [blame] | 115 | /^[ ]*(/bsubsh |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 116 | |
| 117 | # innocuous line -- print it and advance to next line |
| 118 | b |
| 119 | |
| 120 | # found one-liner "(...)" -- mark suspect if it uses ";" internally rather than |
| 121 | # "&&" (but not ";" in a string) |
| 122 | :oneline |
| 123 | /;/{ |
Eric Sunshine | 0d71317 | 2021-12-13 01:30:53 -0500 | [diff] [blame] | 124 | /"[^"]*;[^"]*"/!s/;/; ?!AMP?!/ |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 125 | } |
| 126 | b |
| 127 | |
Kyohei Kadota | b3b753b | 2020-09-10 02:17:41 +0000 | [diff] [blame] | 128 | :subsh |
Ævar Arnfjörð Bjarmason | 2d9ded8 | 2018-08-24 15:20:13 +0000 | [diff] [blame] | 129 | # bare "(" line? -- stash for later printing |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 130 | /^[ ]*([ ]*$/ { |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 131 | h |
Kyohei Kadota | b3b753b | 2020-09-10 02:17:41 +0000 | [diff] [blame] | 132 | bnextln |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 133 | } |
Eric Sunshine | d73f5cf | 2021-12-13 01:30:59 -0500 | [diff] [blame] | 134 | # "(..." line -- "(" opening subshell cuddled with command; temporarily replace |
| 135 | # "(" with sentinel "^" and process the line as if "(" had been seen solo on |
| 136 | # the preceding line; this temporary replacement prevents several rules from |
| 137 | # accidentally thinking "(" introduces a nested subshell; "^" is changed back |
| 138 | # to "(" at output time |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 139 | x |
Eric Sunshine | d73f5cf | 2021-12-13 01:30:59 -0500 | [diff] [blame] | 140 | s/.*// |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 141 | x |
Eric Sunshine | d73f5cf | 2021-12-13 01:30:59 -0500 | [diff] [blame] | 142 | s/(/^/ |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 143 | bslurp |
| 144 | |
Kyohei Kadota | b3b753b | 2020-09-10 02:17:41 +0000 | [diff] [blame] | 145 | :nextln |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 146 | N |
| 147 | s/.*\n// |
| 148 | |
| 149 | :slurp |
| 150 | # incomplete line "...\" |
Ævar Arnfjörð Bjarmason | a3c4c88 | 2018-08-24 15:20:14 +0000 | [diff] [blame] | 151 | /\\$/bicmplte |
Eric Sunshine | 22e3e02 | 2018-08-13 04:47:38 -0400 | [diff] [blame] | 152 | # multi-line quoted string "...\n..."? |
Kyohei Kadota | b3b753b | 2020-09-10 02:17:41 +0000 | [diff] [blame] | 153 | /"/bdqstr |
Eric Sunshine | 22e3e02 | 2018-08-13 04:47:38 -0400 | [diff] [blame] | 154 | # multi-line quoted string '...\n...'? (but not contraction in string "it's") |
| 155 | /'/{ |
Kyohei Kadota | b3b753b | 2020-09-10 02:17:41 +0000 | [diff] [blame] | 156 | /"[^'"]*'[^'"]*"/!bsqstr |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 157 | } |
Eric Sunshine | d938711 | 2018-08-13 04:47:37 -0400 | [diff] [blame] | 158 | :folded |
Eric Sunshine | 22597af | 2021-12-13 01:30:56 -0500 | [diff] [blame] | 159 | # here-doc -- swallow it (but not "<<" in a string) |
| 160 | /<<-*[ ]*[\\'"]*[A-Za-z0-9_]/{ |
| 161 | /"[^"]*<<[^"]*"/!bheredoc |
| 162 | } |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 163 | # comment or empty line -- discard since final non-comment, non-empty line |
| 164 | # before closing ")", "done", "elsif", "else", or "fi" will need to be |
| 165 | # re-visited to drop "suspect" marking since final line of those constructs |
| 166 | # legitimately lacks "&&", so "suspect" mark must be removed |
Kyohei Kadota | b3b753b | 2020-09-10 02:17:41 +0000 | [diff] [blame] | 167 | /^[ ]*#/bnextln |
| 168 | /^[ ]*$/bnextln |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 169 | # in-line comment -- strip it (but not "#" in a string, Bash ${#...} array |
| 170 | # length, or Perforce "//depot/path#42" revision in filespec) |
| 171 | /[ ]#/{ |
| 172 | /"[^"]*#[^"]*"/!s/[ ]#.*$// |
| 173 | } |
| 174 | # one-liner "case ... esac" |
Eric Sunshine | d73f5cf | 2021-12-13 01:30:59 -0500 | [diff] [blame] | 175 | /^[ ^]*case[ ]*..*esac/bchkchn |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 176 | # multi-line "case ... esac" |
Eric Sunshine | d73f5cf | 2021-12-13 01:30:59 -0500 | [diff] [blame] | 177 | /^[ ^]*case[ ]..*[ ]in/bcase |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 178 | # multi-line "for ... done" or "while ... done" |
Eric Sunshine | d73f5cf | 2021-12-13 01:30:59 -0500 | [diff] [blame] | 179 | /^[ ^]*for[ ]..*[ ]in/bcont |
| 180 | /^[ ^]*while[ ]/bcont |
Kyohei Kadota | b3b753b | 2020-09-10 02:17:41 +0000 | [diff] [blame] | 181 | /^[ ]*do[ ]/bcont |
| 182 | /^[ ]*do[ ]*$/bcont |
| 183 | /;[ ]*do/bcont |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 184 | /^[ ]*done[ ]*&&[ ]*$/bdone |
| 185 | /^[ ]*done[ ]*$/bdone |
| 186 | /^[ ]*done[ ]*[<>|]/bdone |
| 187 | /^[ ]*done[ ]*)/bdone |
Kyohei Kadota | b3b753b | 2020-09-10 02:17:41 +0000 | [diff] [blame] | 188 | /||[ ]*exit[ ]/bcont |
| 189 | /||[ ]*exit[ ]*$/bcont |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 190 | # multi-line "if...elsif...else...fi" |
Eric Sunshine | d73f5cf | 2021-12-13 01:30:59 -0500 | [diff] [blame] | 191 | /^[ ^]*if[ ]/bcont |
Kyohei Kadota | b3b753b | 2020-09-10 02:17:41 +0000 | [diff] [blame] | 192 | /^[ ]*then[ ]/bcont |
| 193 | /^[ ]*then[ ]*$/bcont |
| 194 | /;[ ]*then/bcont |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 195 | /^[ ]*elif[ ]/belse |
| 196 | /^[ ]*elif[ ]*$/belse |
| 197 | /^[ ]*else[ ]/belse |
| 198 | /^[ ]*else[ ]*$/belse |
| 199 | /^[ ]*fi[ ]*&&[ ]*$/bdone |
| 200 | /^[ ]*fi[ ]*$/bdone |
| 201 | /^[ ]*fi[ ]*[<>|]/bdone |
| 202 | /^[ ]*fi[ ]*)/bdone |
| 203 | # nested one-liner "(...) &&" |
Eric Sunshine | d73f5cf | 2021-12-13 01:30:59 -0500 | [diff] [blame] | 204 | /^[ ^]*(.*)[ ]*&&[ ]*$/bchkchn |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 205 | # nested one-liner "(...)" |
Eric Sunshine | d73f5cf | 2021-12-13 01:30:59 -0500 | [diff] [blame] | 206 | /^[ ^]*(.*)[ ]*$/bchkchn |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 207 | # nested one-liner "(...) >x" (or "2>x" or "<x" or "|x") |
Eric Sunshine | d73f5cf | 2021-12-13 01:30:59 -0500 | [diff] [blame] | 208 | /^[ ^]*(.*)[ ]*[0-9]*[<>|]/bchkchn |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 209 | # nested multi-line "(...\n...)" |
Eric Sunshine | d73f5cf | 2021-12-13 01:30:59 -0500 | [diff] [blame] | 210 | /^[ ^]*(/bnest |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 211 | # multi-line "{...\n...}" |
Eric Sunshine | d73f5cf | 2021-12-13 01:30:59 -0500 | [diff] [blame] | 212 | /^[ ^]*{/bblock |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 213 | # closing ")" on own line -- exit subshell |
Ævar Arnfjörð Bjarmason | a3c4c88 | 2018-08-24 15:20:14 +0000 | [diff] [blame] | 214 | /^[ ]*)/bclssolo |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 215 | # "$((...))" -- arithmetic expansion; not closing ")" |
Ævar Arnfjörð Bjarmason | a3c4c88 | 2018-08-24 15:20:14 +0000 | [diff] [blame] | 216 | /\$(([^)][^)]*))[^)]*$/bchkchn |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 217 | # "$(...)" -- command substitution; not closing ")" |
Ævar Arnfjörð Bjarmason | a3c4c88 | 2018-08-24 15:20:14 +0000 | [diff] [blame] | 218 | /\$([^)][^)]*)[^)]*$/bchkchn |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 219 | # multi-line "$(...\n...)" -- command substitution; treat as nested subshell |
Eric Sunshine | 06fc5c9 | 2018-08-13 04:47:36 -0400 | [diff] [blame] | 220 | /\$([^)]*$/bnest |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 221 | # "=(...)" -- Bash array assignment; not closing ")" |
Ævar Arnfjörð Bjarmason | a3c4c88 | 2018-08-24 15:20:14 +0000 | [diff] [blame] | 222 | /=(/bchkchn |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 223 | # closing "...) &&" |
| 224 | /)[ ]*&&[ ]*$/bclose |
| 225 | # closing "...)" |
| 226 | /)[ ]*$/bclose |
| 227 | # closing "...) >x" (or "2>x" or "<x" or "|x") |
| 228 | /)[ ]*[<>|]/bclose |
Ævar Arnfjörð Bjarmason | a3c4c88 | 2018-08-24 15:20:14 +0000 | [diff] [blame] | 229 | :chkchn |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 230 | # mark suspect if line uses ";" internally rather than "&&" (but not ";" in a |
| 231 | # string and not ";;" in one-liner "case...esac") |
| 232 | /;/{ |
| 233 | /;;/!{ |
Eric Sunshine | 0d71317 | 2021-12-13 01:30:53 -0500 | [diff] [blame] | 234 | /"[^"]*;[^"]*"/!s/;/; ?!AMP?!/ |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 235 | } |
| 236 | } |
| 237 | # line ends with pipe "...|" -- valid; not missing "&&" |
Kyohei Kadota | b3b753b | 2020-09-10 02:17:41 +0000 | [diff] [blame] | 238 | /|[ ]*$/bcont |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 239 | # missing end-of-line "&&" -- mark suspect |
Eric Sunshine | db8c7a1 | 2021-12-13 01:30:50 -0500 | [diff] [blame] | 240 | /&&[ ]*$/!s/$/ ?!AMP?!/ |
Kyohei Kadota | b3b753b | 2020-09-10 02:17:41 +0000 | [diff] [blame] | 241 | :cont |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 242 | # retrieve and print previous line |
| 243 | x |
Eric Sunshine | d73f5cf | 2021-12-13 01:30:59 -0500 | [diff] [blame] | 244 | s/^\([ ]*\)^/\1(/ |
Eric Sunshine | 34ba05c | 2021-12-13 01:30:57 -0500 | [diff] [blame] | 245 | s/?!HERE?!/<</g |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 246 | n |
| 247 | bslurp |
| 248 | |
| 249 | # found incomplete line "...\" -- slurp up next line |
Ævar Arnfjörð Bjarmason | a3c4c88 | 2018-08-24 15:20:14 +0000 | [diff] [blame] | 250 | :icmplte |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 251 | N |
| 252 | s/\\\n// |
| 253 | bslurp |
| 254 | |
Eric Sunshine | 22e3e02 | 2018-08-13 04:47:38 -0400 | [diff] [blame] | 255 | # check for multi-line double-quoted string "...\n..." -- fold to one line |
Kyohei Kadota | b3b753b | 2020-09-10 02:17:41 +0000 | [diff] [blame] | 256 | :dqstr |
Eric Sunshine | 22e3e02 | 2018-08-13 04:47:38 -0400 | [diff] [blame] | 257 | # remove all quote pairs |
| 258 | s/"\([^"]*\)"/@!\1@!/g |
| 259 | # done if no dangling quote |
| 260 | /"/!bdqdone |
| 261 | # otherwise, slurp next line and try again |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 262 | N |
| 263 | s/\n// |
Kyohei Kadota | b3b753b | 2020-09-10 02:17:41 +0000 | [diff] [blame] | 264 | bdqstr |
Eric Sunshine | 22e3e02 | 2018-08-13 04:47:38 -0400 | [diff] [blame] | 265 | :dqdone |
| 266 | s/@!/"/g |
Eric Sunshine | d938711 | 2018-08-13 04:47:37 -0400 | [diff] [blame] | 267 | bfolded |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 268 | |
Eric Sunshine | 22e3e02 | 2018-08-13 04:47:38 -0400 | [diff] [blame] | 269 | # check for multi-line single-quoted string '...\n...' -- fold to one line |
Kyohei Kadota | b3b753b | 2020-09-10 02:17:41 +0000 | [diff] [blame] | 270 | :sqstr |
Eric Sunshine | 22e3e02 | 2018-08-13 04:47:38 -0400 | [diff] [blame] | 271 | # remove all quote pairs |
| 272 | s/'\([^']*\)'/@!\1@!/g |
| 273 | # done if no dangling quote |
| 274 | /'/!bsqdone |
| 275 | # otherwise, slurp next line and try again |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 276 | N |
| 277 | s/\n// |
Kyohei Kadota | b3b753b | 2020-09-10 02:17:41 +0000 | [diff] [blame] | 278 | bsqstr |
Eric Sunshine | 22e3e02 | 2018-08-13 04:47:38 -0400 | [diff] [blame] | 279 | :sqdone |
| 280 | s/@!/'/g |
Eric Sunshine | d938711 | 2018-08-13 04:47:37 -0400 | [diff] [blame] | 281 | bfolded |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 282 | |
| 283 | # found here-doc -- swallow it to avoid false hits within its body (but keep |
Eric Sunshine | c2c29cc | 2018-08-13 04:47:34 -0400 | [diff] [blame] | 284 | # the command to which it was attached) |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 285 | :heredoc |
Eric Sunshine | 34ba05c | 2021-12-13 01:30:57 -0500 | [diff] [blame] | 286 | s/^\(.*\)<<\(-*[ ]*\)[\\'"]*\([A-Za-z0-9_][A-Za-z0-9_]*\)['"]*/<\3>\1?!HERE?!\2\3/ |
Kyohei Kadota | b3b753b | 2020-09-10 02:17:41 +0000 | [diff] [blame] | 287 | :hdocsub |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 288 | N |
Eric Sunshine | c2c29cc | 2018-08-13 04:47:34 -0400 | [diff] [blame] | 289 | /^<\([^>]*\)>.*\n[ ]*\1[ ]*$/!{ |
| 290 | s/\n.*$// |
Kyohei Kadota | b3b753b | 2020-09-10 02:17:41 +0000 | [diff] [blame] | 291 | bhdocsub |
Eric Sunshine | c2c29cc | 2018-08-13 04:47:34 -0400 | [diff] [blame] | 292 | } |
| 293 | s/^<[^>]*>// |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 294 | s/\n.*$// |
Eric Sunshine | d938711 | 2018-08-13 04:47:37 -0400 | [diff] [blame] | 295 | bfolded |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 296 | |
| 297 | # found "case ... in" -- pass through untouched |
| 298 | :case |
| 299 | x |
Eric Sunshine | d73f5cf | 2021-12-13 01:30:59 -0500 | [diff] [blame] | 300 | s/^\([ ]*\)^/\1(/ |
Eric Sunshine | 34ba05c | 2021-12-13 01:30:57 -0500 | [diff] [blame] | 301 | s/?!HERE?!/<</g |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 302 | n |
Eric Sunshine | 31da22d | 2021-12-13 01:30:58 -0500 | [diff] [blame] | 303 | :cascom |
| 304 | /^[ ]*#/{ |
| 305 | N |
| 306 | s/.*\n// |
| 307 | bcascom |
| 308 | } |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 309 | /^[ ]*esac/bslurp |
| 310 | bcase |
| 311 | |
| 312 | # found "else" or "elif" -- drop "suspect" from final line before "else" since |
| 313 | # that line legitimately lacks "&&" |
| 314 | :else |
| 315 | x |
Eric Sunshine | 0d71317 | 2021-12-13 01:30:53 -0500 | [diff] [blame] | 316 | s/\( ?!AMP?!\)* ?!AMP?!$// |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 317 | x |
Kyohei Kadota | b3b753b | 2020-09-10 02:17:41 +0000 | [diff] [blame] | 318 | bcont |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 319 | |
| 320 | # found "done" closing for-loop or while-loop, or "fi" closing if-then -- drop |
| 321 | # "suspect" from final contained line since that line legitimately lacks "&&" |
| 322 | :done |
| 323 | x |
Eric Sunshine | 0d71317 | 2021-12-13 01:30:53 -0500 | [diff] [blame] | 324 | s/\( ?!AMP?!\)* ?!AMP?!$// |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 325 | x |
| 326 | # is 'done' or 'fi' cuddled with ")" to close subshell? |
| 327 | /done.*)/bclose |
| 328 | /fi.*)/bclose |
Ævar Arnfjörð Bjarmason | a3c4c88 | 2018-08-24 15:20:14 +0000 | [diff] [blame] | 329 | bchkchn |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 330 | |
| 331 | # found nested multi-line "(...\n...)" -- pass through untouched |
| 332 | :nest |
| 333 | x |
Kyohei Kadota | b3b753b | 2020-09-10 02:17:41 +0000 | [diff] [blame] | 334 | :nstslrp |
Eric Sunshine | d73f5cf | 2021-12-13 01:30:59 -0500 | [diff] [blame] | 335 | s/^\([ ]*\)^/\1(/ |
Eric Sunshine | 34ba05c | 2021-12-13 01:30:57 -0500 | [diff] [blame] | 336 | s/?!HERE?!/<</g |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 337 | n |
Eric Sunshine | 31da22d | 2021-12-13 01:30:58 -0500 | [diff] [blame] | 338 | :nstcom |
| 339 | # comment -- not closing ")" if in comment |
| 340 | /^[ ]*#/{ |
| 341 | N |
| 342 | s/.*\n// |
| 343 | bnstcom |
| 344 | } |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 345 | # closing ")" on own line -- stop nested slurp |
Kyohei Kadota | b3b753b | 2020-09-10 02:17:41 +0000 | [diff] [blame] | 346 | /^[ ]*)/bnstcl |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 347 | # "$((...))" -- arithmetic expansion; not closing ")" |
Ævar Arnfjörð Bjarmason | a3c4c88 | 2018-08-24 15:20:14 +0000 | [diff] [blame] | 348 | /\$(([^)][^)]*))[^)]*$/bnstcnt |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 349 | # "$(...)" -- command substitution; not closing ")" |
Ævar Arnfjörð Bjarmason | a3c4c88 | 2018-08-24 15:20:14 +0000 | [diff] [blame] | 350 | /\$([^)][^)]*)[^)]*$/bnstcnt |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 351 | # closing "...)" -- stop nested slurp |
Kyohei Kadota | b3b753b | 2020-09-10 02:17:41 +0000 | [diff] [blame] | 352 | /)/bnstcl |
Ævar Arnfjörð Bjarmason | a3c4c88 | 2018-08-24 15:20:14 +0000 | [diff] [blame] | 353 | :nstcnt |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 354 | x |
Kyohei Kadota | b3b753b | 2020-09-10 02:17:41 +0000 | [diff] [blame] | 355 | bnstslrp |
| 356 | :nstcl |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 357 | # is it "))" which closes nested and parent subshells? |
| 358 | /)[ ]*)/bslurp |
Ævar Arnfjörð Bjarmason | a3c4c88 | 2018-08-24 15:20:14 +0000 | [diff] [blame] | 359 | bchkchn |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 360 | |
| 361 | # found multi-line "{...\n...}" block -- pass through untouched |
| 362 | :block |
| 363 | x |
Eric Sunshine | d73f5cf | 2021-12-13 01:30:59 -0500 | [diff] [blame] | 364 | s/^\([ ]*\)^/\1(/ |
Eric Sunshine | 34ba05c | 2021-12-13 01:30:57 -0500 | [diff] [blame] | 365 | s/?!HERE?!/<</g |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 366 | n |
Eric Sunshine | 31da22d | 2021-12-13 01:30:58 -0500 | [diff] [blame] | 367 | :blkcom |
| 368 | /^[ ]*#/{ |
| 369 | N |
| 370 | s/.*\n// |
| 371 | bblkcom |
| 372 | } |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 373 | # closing "}" -- stop block slurp |
Ævar Arnfjörð Bjarmason | a3c4c88 | 2018-08-24 15:20:14 +0000 | [diff] [blame] | 374 | /}/bchkchn |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 375 | bblock |
| 376 | |
| 377 | # found closing ")" on own line -- drop "suspect" from final line of subshell |
| 378 | # since that line legitimately lacks "&&" and exit subshell loop |
Ævar Arnfjörð Bjarmason | a3c4c88 | 2018-08-24 15:20:14 +0000 | [diff] [blame] | 379 | :clssolo |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 380 | x |
Eric Sunshine | 0d71317 | 2021-12-13 01:30:53 -0500 | [diff] [blame] | 381 | s/\( ?!AMP?!\)* ?!AMP?!$// |
Eric Sunshine | d73f5cf | 2021-12-13 01:30:59 -0500 | [diff] [blame] | 382 | s/^\([ ]*\)^/\1(/ |
Eric Sunshine | 34ba05c | 2021-12-13 01:30:57 -0500 | [diff] [blame] | 383 | s/?!HERE?!/<</g |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 384 | p |
| 385 | x |
Eric Sunshine | d73f5cf | 2021-12-13 01:30:59 -0500 | [diff] [blame] | 386 | s/^\([ ]*\)^/\1(/ |
Eric Sunshine | 34ba05c | 2021-12-13 01:30:57 -0500 | [diff] [blame] | 387 | s/?!HERE?!/<</g |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 388 | b |
| 389 | |
| 390 | # found closing "...)" -- exit subshell loop |
| 391 | :close |
| 392 | x |
Eric Sunshine | d73f5cf | 2021-12-13 01:30:59 -0500 | [diff] [blame] | 393 | s/^\([ ]*\)^/\1(/ |
Eric Sunshine | 34ba05c | 2021-12-13 01:30:57 -0500 | [diff] [blame] | 394 | s/?!HERE?!/<</g |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 395 | p |
| 396 | x |
Eric Sunshine | d73f5cf | 2021-12-13 01:30:59 -0500 | [diff] [blame] | 397 | s/^\([ ]*\)^/\1(/ |
Eric Sunshine | 34ba05c | 2021-12-13 01:30:57 -0500 | [diff] [blame] | 398 | s/?!HERE?!/<</g |
Eric Sunshine | 878f988 | 2018-07-11 02:46:33 -0400 | [diff] [blame] | 399 | b |