Skip to content

Commit

Permalink
syntax: properly support case patterns
Browse files Browse the repository at this point in the history
  • Loading branch information
akinomyoga committed Feb 20, 2021
1 parent ecb8888 commit 64b55b7
Show file tree
Hide file tree
Showing 4 changed files with 322 additions and 51 deletions.
6 changes: 5 additions & 1 deletion lib/core-syntax-ctx.def
Original file line number Diff line number Diff line change
Expand Up @@ -55,7 +55,11 @@ CTX_ARGVR 62 # (コマンド) declare の引数の右辺
CTX_CONDX 32
CTX_CONDI 33
CTX_CONDQ 67
CTX_CASE 34 # case パターン待ち
CTX_CASE 34 # case パターン 開始待ち ("(" を特別扱い)
CTX_CPATX 76 # case パターン 単語待ち ("(" の後)
CTX_CPATI 77 # case パターン 単語内部
CTX_CPATQ 79 # case パターン 変数代入形式(チルダ展開の為に区別)
CTX_CPATX0 78 # case パターン 単語終了後。"|" or ")" 待ち
CTX_PATN 30 # glob 通常文字
ATTR_GLOB 31 # glob 特別文字
CTX_BRAX 54 # [...] (角括弧式)
Expand Down
83 changes: 69 additions & 14 deletions lib/core-syntax.sh
Original file line number Diff line number Diff line change
Expand Up @@ -2323,21 +2323,17 @@ function ble/syntax:bash/ctx-quot {
_BLE_SYNTAX_FCTX[CTX_CASE]=ble/syntax:bash/ctx-case
function ble/syntax:bash/ctx-case {
if [[ $tail =~ ^$_ble_syntax_bash_RexIFSs ]]; then
((_ble_syntax_attr[i]=ctx,
i+=${#BASH_REMATCH}))
((_ble_syntax_attr[i]=ctx,i+=${#BASH_REMATCH}))
return 0
elif [[ $tail == '('* ]]; then
((ctx=CTX_CMDX))
ble/syntax/parse/nest-push "$CTX_PATN"
((_ble_syntax_attr[i++]=ATTR_GLOB))
((_ble_syntax_attr[i++]=ATTR_GLOB,ctx=CTX_CPATX))
return 0
elif [[ $tail == 'esac'$_ble_syntax_bash_RexDelimiter* || $tail == 'esac' ]]; then
((ctx=CTX_CMDX))
ble/syntax:bash/ctx-command
else
((ctx=CTX_CMDX))
ble/syntax/parse/nest-push "$CTX_PATN"
ble/syntax:bash/ctx-globpat
((ctx=CTX_CPATX))
ble/syntax:bash/ctx-command-case-pattern-expect
fi
}

Expand Down Expand Up @@ -3035,13 +3031,16 @@ function ble/syntax:bash/check-tilde-expansion {
# 変数代入の形式の引数の右辺ではチルダ展開が有効である。
#

# 変数代入形式の時に文脈を切り替える文脈値。実際に変数代入でなくても変数代入形
# 式によるチルダ展開が有効である時には区別する必要がある。
_ble_syntax_bash_command_CtxAssign[CTX_CMDI]=$CTX_VRHS
_ble_syntax_bash_command_CtxAssign[CTX_COARGI]=$CTX_VRHS
_ble_syntax_bash_command_CtxAssign[CTX_ARGVI]=$CTX_ARGVR
_ble_syntax_bash_command_CtxAssign[CTX_ARGEI]=$CTX_ARGER
_ble_syntax_bash_command_CtxAssign[CTX_ARGI]=$CTX_ARGQ
_ble_syntax_bash_command_CtxAssign[CTX_FARGI3]=$CTX_FARGQ3
_ble_syntax_bash_command_CtxAssign[CTX_CARGI1]=$CTX_CARGQ1
_ble_syntax_bash_command_CtxAssign[CTX_CPATI]=$CTX_CPATQ
_ble_syntax_bash_command_CtxAssign[CTX_VALI]=$CTX_VALQ
_ble_syntax_bash_command_CtxAssign[CTX_CONDI]=$CTX_CONDQ

Expand All @@ -3052,6 +3051,7 @@ _ble_syntax_bash_command_IsAssign[CTX_ARGER]=$CTX_ARGEI
_ble_syntax_bash_command_IsAssign[CTX_ARGQ]=$CTX_ARGI
_ble_syntax_bash_command_IsAssign[CTX_FARGQ3]=$CTX_FARGI3
_ble_syntax_bash_command_IsAssign[CTX_CARGQ1]=$CTX_CARGI1
_ble_syntax_bash_command_IsAssign[CTX_CPATQ]=$CTX_CPATI
_ble_syntax_bash_command_IsAssign[CTX_VALR]=$CTX_VALI
_ble_syntax_bash_command_IsAssign[CTX_VALQ]=$CTX_VALI
_ble_syntax_bash_command_IsAssign[CTX_CONDQ]=$CTX_CONDI
Expand Down Expand Up @@ -3192,12 +3192,18 @@ _BLE_SYNTAX_FEND[CTX_FARGI3]=ble/syntax:bash/ctx-command/check-word-end
_BLE_SYNTAX_FEND[CTX_FARGQ3]=ble/syntax:bash/ctx-command/check-word-end
_BLE_SYNTAX_FCTX[CTX_CARGX1]=ble/syntax:bash/ctx-command-compound-expect
_BLE_SYNTAX_FCTX[CTX_CARGX2]=ble/syntax:bash/ctx-command-compound-expect
_BLE_SYNTAX_FCTX[CTX_CPATX]=ble/syntax:bash/ctx-command-case-pattern-expect
_BLE_SYNTAX_FCTX[CTX_CPATX0]=ble/syntax:bash/ctx-command-case-pattern-expect
_BLE_SYNTAX_FCTX[CTX_CARGI1]=ble/syntax:bash/ctx-command
_BLE_SYNTAX_FCTX[CTX_CARGQ1]=ble/syntax:bash/ctx-command
_BLE_SYNTAX_FCTX[CTX_CARGI2]=ble/syntax:bash/ctx-command
_BLE_SYNTAX_FCTX[CTX_CPATI]=ble/syntax:bash/ctx-command
_BLE_SYNTAX_FCTX[CTX_CPATQ]=ble/syntax:bash/ctx-command
_BLE_SYNTAX_FEND[CTX_CARGI1]=ble/syntax:bash/ctx-command/check-word-end
_BLE_SYNTAX_FEND[CTX_CARGQ1]=ble/syntax:bash/ctx-command/check-word-end
_BLE_SYNTAX_FEND[CTX_CARGI2]=ble/syntax:bash/ctx-command/check-word-end
_BLE_SYNTAX_FEND[CTX_CPATI]=ble/syntax:bash/ctx-command/check-word-end
_BLE_SYNTAX_FEND[CTX_CPATQ]=ble/syntax:bash/ctx-command/check-word-end
_BLE_SYNTAX_FCTX[CTX_TARGX1]=ble/syntax:bash/ctx-command-time-expect
_BLE_SYNTAX_FCTX[CTX_TARGX2]=ble/syntax:bash/ctx-command-time-expect
_BLE_SYNTAX_FCTX[CTX_TARGI1]=ble/syntax:bash/ctx-command
Expand Down Expand Up @@ -3339,6 +3345,8 @@ _ble_syntax_bash_command_EndCtx[CTX_FARGQ3]=$CTX_FARGX3
_ble_syntax_bash_command_EndCtx[CTX_CARGI1]=$CTX_CARGX2
_ble_syntax_bash_command_EndCtx[CTX_CARGQ1]=$CTX_CARGX2
_ble_syntax_bash_command_EndCtx[CTX_CARGI2]=$CTX_CASE
_ble_syntax_bash_command_EndCtx[CTX_CPATI]=$CTX_CPATX0
_ble_syntax_bash_command_EndCtx[CTX_CPATQ]=$CTX_CPATX0
_ble_syntax_bash_command_EndCtx[CTX_TARGI1]=$((_ble_bash>=40200?CTX_TARGX2:CTX_CMDXT)) #1
_ble_syntax_bash_command_EndCtx[CTX_TARGI2]=$CTX_CMDXT

Expand All @@ -3363,6 +3371,8 @@ _ble_syntax_bash_command_EndWtype[CTX_FARGX2]=$CTX_FARGI2 # in
_ble_syntax_bash_command_EndWtype[CTX_FARGX3]=$CTX_ARGI # in
_ble_syntax_bash_command_EndWtype[CTX_CARGX1]=$CTX_ARGI
_ble_syntax_bash_command_EndWtype[CTX_CARGX2]=$CTX_CARGI2 # in
_ble_syntax_bash_command_EndWtype[CTX_CPATX]=$CTX_CPATI
_ble_syntax_bash_command_EndWtype[CTX_CPATX0]=$CTX_CPATI
_ble_syntax_bash_command_EndWtype[CTX_TARGX1]=$CTX_ARGI # -p
_ble_syntax_bash_command_EndWtype[CTX_TARGX2]=$CTX_ARGI # --

Expand Down Expand Up @@ -3748,15 +3758,17 @@ _ble_syntax_bash_command_BeginCtx[CTX_FARGX2]=$CTX_FARGI2
_ble_syntax_bash_command_BeginCtx[CTX_FARGX3]=$CTX_FARGI3
_ble_syntax_bash_command_BeginCtx[CTX_CARGX1]=$CTX_CARGI1
_ble_syntax_bash_command_BeginCtx[CTX_CARGX2]=$CTX_CARGI2
_ble_syntax_bash_command_BeginCtx[CTX_CPATX]=$CTX_CPATI
_ble_syntax_bash_command_BeginCtx[CTX_CPATX0]=$CTX_CPATI
_ble_syntax_bash_command_BeginCtx[CTX_TARGX1]=$CTX_TARGI1
_ble_syntax_bash_command_BeginCtx[CTX_TARGX2]=$CTX_TARGI2
_ble_syntax_bash_command_BeginCtx[CTX_COARGX]=$CTX_COARGI

#%if !release
## @arr _ble_syntax_bash_command_isARGI[ctx]
##
## この配列要素が非空文字列のとき、
## その文脈はシェル単語を解析中に用いられることを表す
## assert 用の配列。シェル単語の解析中に現れても良い文脈値を管理する。この配
## 列要素が非空文字列のとき、その文脈はシェル単語の解析中に現れても良い
##
_ble_syntax_bash_command_isARGI[CTX_CMDI]=1
_ble_syntax_bash_command_isARGI[CTX_VRHS]=1
Expand All @@ -3773,6 +3785,8 @@ _ble_syntax_bash_command_isARGI[CTX_FARGQ3]=1 # args... (= の後)
_ble_syntax_bash_command_isARGI[CTX_CARGI1]=1 # value
_ble_syntax_bash_command_isARGI[CTX_CARGQ1]=1 # value (= の後)
_ble_syntax_bash_command_isARGI[CTX_CARGI2]=1 # in
_ble_syntax_bash_command_isARGI[CTX_CPATI]=1 # pattern
_ble_syntax_bash_command_isARGI[CTX_CPATQ]=1 # pattern
_ble_syntax_bash_command_isARGI[CTX_TARGI1]=1 # -p
_ble_syntax_bash_command_isARGI[CTX_TARGI2]=1 # --
_ble_syntax_bash_command_isARGI[CTX_COARGI]=1 # var (coproc の後)
Expand All @@ -3792,7 +3806,7 @@ function ble/syntax:bash/ctx-command/.check-word-begin {
# 配列 _ble_syntax_bash_command_EndWtype により変換されてから tree に登録される。
ble/syntax/parse/word-push "$wtype" "$i"

((octx!=CTX_ARGX0)); return "$?" # return unexpectedWbegin
((octx!=CTX_ARGX0&&octx!=CTX_CPATX0)); return "$?" # return unexpectedWbegin
fi

#%if !release
Expand All @@ -3806,7 +3820,8 @@ function ble/syntax:bash/ctx-command {
#%if !release
if ble/syntax:bash/starts-with-delimiter-or-redirect; then
ble/util/assert '
((ctx==CTX_ARGX||ctx==CTX_ARGX0||ctx==CTX_ARGVX||ctx==CTX_ARGEX||ctx==CTX_FARGX2||ctx==CTX_FARGX3||ctx==CTX_COARGX||
((ctx==CTX_ARGX||ctx==CTX_ARGX0||ctx==CTX_ARGVX||ctx==CTX_ARGEX||
ctx==CTX_FARGX2||ctx==CTX_FARGX3||ctx==CTX_COARGX||
ctx==CTX_CMDX||ctx==CTX_CMDX1||ctx==CTX_CMDXT||ctx==CTX_CMDXC||
ctx==CTX_CMDXE||ctx==CTX_CMDXD||ctx==CTX_CMDXD0||ctx==CTX_CMDXV))' "invalid ctx=$ctx @ i=$i"
ble/util/assert '((wbegin<0&&wtype<0))' "invalid word-context (wtype=$wtype wbegin=$wbegin) on non-word char."
Expand Down Expand Up @@ -3963,6 +3978,38 @@ function ble/syntax:bash/ctx-command-time-expect {
ble/syntax:bash/ctx-command
}

function ble/syntax:bash/ctx-command-case-pattern-expect {
ble/util/assert '((ctx==CTX_CPATX||ctx==CTX_CPATX0))'

if ble/syntax:bash/starts-with-delimiter-or-redirect; then
local delimiter=$BASH_REMATCH
if [[ $tail =~ ^$_ble_syntax_bash_RexSpaces ]]; then
((_ble_syntax_attr[i]=ctx,i+=${#BASH_REMATCH}))
elif [[ $tail == '|'* ]]; then
((_ble_syntax_attr[i++]=ctx==CTX_CPATX?ATTR_ERR:ATTR_GLOB,ctx=CTX_CPATX))
elif [[ $tail == ')'* ]]; then
((_ble_syntax_attr[i++]=ctx==CTX_CPATX?ATTR_ERR:ATTR_GLOB,ctx=CTX_CMDX))
elif [[ $tail == '('* ]]; then
# ctx-command と同様にエラーにして @() を始める。
ble/syntax:bash/ctx-command/.check-delimiter-or-redirect
else
# 改行、リダイレクト、; & はエラー
((_ble_syntax_attr[i]=ATTR_ERR,i+=${#delimiter}))
fi
return "$?"
fi

# コメント禁止
local i0=$i
if ble/syntax:bash/check-comment; then
((_ble_syntax_attr[i0]=ATTR_ERR))
return 0
fi

# 他は同じ
ble/syntax:bash/ctx-command
}

#------------------------------------------------------------------------------
# 文脈: 配列値リスト
#
Expand Down Expand Up @@ -5138,6 +5185,8 @@ _ble_syntax_bash_complete_check_prefix[CTX_FARGI3]='inside-argument argument'
_ble_syntax_bash_complete_check_prefix[CTX_FARGQ3]='inside-argument argument'
_ble_syntax_bash_complete_check_prefix[CTX_CARGI1]='inside-argument argument'
_ble_syntax_bash_complete_check_prefix[CTX_CARGQ1]='inside-argument argument'
_ble_syntax_bash_complete_check_prefix[CTX_CPATI]='inside-argument argument'
_ble_syntax_bash_complete_check_prefix[CTX_CPATQ]='inside-argument argument'
_ble_syntax_bash_complete_check_prefix[CTX_COARGI]='inside-argument variable command'
_ble_syntax_bash_complete_check_prefix[CTX_VALI]='inside-argument file'
_ble_syntax_bash_complete_check_prefix[CTX_VALQ]='inside-argument file'
Expand Down Expand Up @@ -5203,6 +5252,7 @@ function ble/syntax/completion-context/.check-prefix/ctx:next-command {
## ARGX 系統の文脈に対する補完文脈の生成
_ble_syntax_bash_complete_check_prefix[CTX_ARGX]=next-argument
_ble_syntax_bash_complete_check_prefix[CTX_CARGX1]=next-argument
_ble_syntax_bash_complete_check_prefix[CTX_CPATX]=next-argument
_ble_syntax_bash_complete_check_prefix[CTX_FARGX3]=next-argument
_ble_syntax_bash_complete_check_prefix[CTX_COARGX]=next-argument
_ble_syntax_bash_complete_check_prefix[CTX_ARGVX]=next-argument
Expand Down Expand Up @@ -5273,6 +5323,7 @@ function ble/syntax/completion-context/.check-prefix/ctx:next-identifier {
fi
}
_ble_syntax_bash_complete_check_prefix[CTX_ARGX0]="next-word sabbrev"
_ble_syntax_bash_complete_check_prefix[CTX_CPATX0]="next-word sabbrev"
_ble_syntax_bash_complete_check_prefix[CTX_CMDXD0]="next-word wordlist:-rs:';:{:do'"
_ble_syntax_bash_complete_check_prefix[CTX_CMDXD]="next-word wordlist:-rs:'{:do'"
_ble_syntax_bash_complete_check_prefix[CTX_CMDXE]="next-word wordlist:-rs:'}:fi:done:esac:then:elif:else:do'"
Expand Down Expand Up @@ -5528,7 +5579,7 @@ function ble/syntax/completion-context/.check-here {
ble/syntax/completion-context/.add wordlist:-rs:';:{:do' "$index"
elif ((ctx==CTX_CMDXD)); then
ble/syntax/completion-context/.add wordlist:-rs:'{:do' "$index"
elif ((ctx==CTX_ARGX0)); then
elif ((ctx==CTX_ARGX0||ctx==CTX_CPATX0)); then
ble/syntax/completion-context/.add sabbrev "$index"
elif ((ctx==CTX_ARGX||ctx==CTX_CARGX1||ctx==CTX_FARGX3)); then
ble/syntax/completion-context/.add argument "$index"
Expand All @@ -5550,7 +5601,7 @@ function ble/syntax/completion-context/.check-here {
elif ((ctx==CTX_COARGX)); then
ble/syntax/completion-context/.add variable:w "$index"
ble/syntax/completion-context/.add command "$index"
elif ((ctx==CTX_RDRF||ctx==CTX_RDRS)); then
elif ((ctx==CTX_CPATI||ctx==CTX_RDRF||ctx==CTX_RDRS)); then
ble/syntax/completion-context/.add file "$index"
elif ((ctx==CTX_VRHS||ctx==CTX_ARGVR||ctx==CTX_ARGER||ctx==CTX_VALR)); then
ble/syntax/completion-context/.add rhs "$index"
Expand Down Expand Up @@ -5965,6 +6016,10 @@ function ble/syntax/faces-onload-hook {
ble/syntax/attr2iface/.define CTX_CARGI1 syntax_default
ble/syntax/attr2iface/.define CTX_CARGQ1 syntax_default
ble/syntax/attr2iface/.define CTX_CARGI2 command_keyword
ble/syntax/attr2iface/.define CTX_CPATX syntax_default
ble/syntax/attr2iface/.define CTX_CPATI syntax_default
ble/syntax/attr2iface/.define CTX_CPATQ syntax_default
ble/syntax/attr2iface/.define CTX_CPATX0 syntax_default

ble/syntax/attr2iface/.define CTX_TARGX1 syntax_default
ble/syntax/attr2iface/.define CTX_TARGX2 syntax_default
Expand Down
5 changes: 3 additions & 2 deletions memo/ChangeLog.md
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@
- canvas, edit: support `bleopt info_display` (suggested by 0neGuyDev) `#D1458` 69228fa
- canvas (panel): always call `panel::render` to update height `#D1472` 51d2c05
- prompt: support `bleopt prompt_status_{line,align}` and `face prompt_status_line` `#D1462` cca1cbc
- syntax: properly support case patterns `#D1474` `#D1475` `#D1476` 0000000

## Changes

Expand All @@ -29,7 +30,7 @@
- complete: perform pathname expansions in subshells (motivated by 3ximus) `#D1450` d511896
- complete: support `bleopt complete_timeout_compvar` to time out pathname expansions for `COMP_WORDS` / `COMP_LINE` `#D1457` cc2881a
- prompt: rename `bleopt prompt_{status_line => term_status}` `#D1462` cca1cbc
- edit (`ble/builtin/read`): cancel by <kbd>C-d</kbd> on an empty line `#D1473` 0000000
- edit (`ble/builtin/read`): cancel by <kbd>C-d</kbd> on an empty line `#D1473` ecb8888

## Fixes

Expand All @@ -47,7 +48,7 @@
- syntax: fix broken AST with `[[` keyword `#D1454` 69658ef
- benchmark (`ble-measure`): work around a locale-dependent decimal point of `EPOCHREALTIME` (reported by 3ximus) `#D1460` 1aa471b
- global:work around bash-4.2 bug of `declare -gA` (reported by 0xC0ncord) `#D1470` 8856a04
- global: fix declaration of associative arrays for `ble-reload` `#D1471` 3cae6e4
- global: fix declaration of associative arrays for `ble-reload` (reported by 0xC0ncord) `#D1471` 3cae6e4

## Internal changes and fixes

Expand Down
Loading

0 comments on commit 64b55b7

Please sign in to comment.