// Copyright (c) 2017, Daniel Martí // See LICENSE for licensing information package interp import ( "fmt" "os" "regexp" "strconv" "strings" "unicode" "unicode/utf8" "mvdan.cc/sh/syntax" ) func anyOfLit(v interface{}, vals ...string) string { word, _ := v.(*syntax.Word) if word == nil || len(word.Parts) != 1 { return "" } lit, ok := word.Parts[0].(*syntax.Lit) if !ok { return "" } for _, val := range vals { if lit.Value == val { return val } } return "" } func (r *Runner) quotedElems(pe *syntax.ParamExp) []string { if pe == nil { return nil } if pe.Param.Value == "@" { return r.Params } if anyOfLit(pe.Index, "@") == "" { return nil } val, _ := r.lookupVar(pe.Param.Value) switch x := val.Value.(type) { case IndexArray: return x } return nil } func (r *Runner) paramExp(pe *syntax.ParamExp) string { name := pe.Param.Value var vr Variable set := false index := pe.Index switch name { case "#": vr.Value = StringVal(strconv.Itoa(len(r.Params))) case "@", "*": vr.Value = IndexArray(r.Params) index = &syntax.Word{Parts: []syntax.WordPart{ &syntax.Lit{Value: name}, }} case "?": vr.Value = StringVal(strconv.Itoa(r.exit)) case "$": vr.Value = StringVal(strconv.Itoa(os.Getpid())) case "PPID": vr.Value = StringVal(strconv.Itoa(os.Getppid())) case "LINENO": line := uint64(pe.Pos().Line()) vr.Value = StringVal(strconv.FormatUint(line, 10)) default: if n, err := strconv.Atoi(name); err == nil { if i := n - 1; i < len(r.Params) { vr.Value, set = StringVal(r.Params[i]), true } } else { vr, set = r.lookupVar(name) } } str := r.varStr(vr, 0) if index != nil { str = r.varInd(vr, index, 0) } slicePos := func(expr syntax.ArithmExpr) int { p := r.arithm(expr) if p < 0 { p = len(str) + p if p < 0 { p = len(str) } } else if p > len(str) { p = len(str) } return p } switch { case pe.Length: n := 1 if anyOfLit(index, "@", "*") != "" { switch x := vr.Value.(type) { case IndexArray: n = len(x) case AssocArray: n = len(x) } } else { n = utf8.RuneCountInString(str) } str = strconv.Itoa(n) case pe.Excl: if pe.Names != 0 { str = strings.Join(r.namesByPrefix(pe.Param.Value), " ") } else if vr.NameRef { str = string(vr.Value.(StringVal)) } else if str != "" { vr, _ = r.lookupVar(str) str = r.varStr(vr, 0) } case pe.Slice != nil: if pe.Slice.Offset != nil { offset := slicePos(pe.Slice.Offset) str = str[offset:] } if pe.Slice.Length != nil { length := slicePos(pe.Slice.Length) str = str[:length] } case pe.Repl != nil: orig := r.lonePattern(pe.Repl.Orig) with := r.loneWord(pe.Repl.With) n := 1 if pe.Repl.All { n = -1 } locs := findAllIndex(orig, str, n) buf := r.strBuilder() last := 0 for _, loc := range locs { buf.WriteString(str[last:loc[0]]) buf.WriteString(with) last = loc[1] } buf.WriteString(str[last:]) str = buf.String() case pe.Exp != nil: arg := r.loneWord(pe.Exp.Word) switch pe.Exp.Op { case syntax.SubstColPlus: if str == "" { break } fallthrough case syntax.SubstPlus: if set { str = arg } case syntax.SubstMinus: if set { break } fallthrough case syntax.SubstColMinus: if str == "" { str = arg } case syntax.SubstQuest: if set { break } fallthrough case syntax.SubstColQuest: if str == "" { r.errf("%s\n", arg) r.exit = 1 r.lastExit() } case syntax.SubstAssgn: if set { break } fallthrough case syntax.SubstColAssgn: if str == "" { r.setVarString(name, arg) str = arg } case syntax.RemSmallPrefix: str = removePattern(str, arg, false, false) case syntax.RemLargePrefix: str = removePattern(str, arg, false, true) case syntax.RemSmallSuffix: str = removePattern(str, arg, true, false) case syntax.RemLargeSuffix: str = removePattern(str, arg, true, true) case syntax.UpperFirst: rs := []rune(str) if len(rs) > 0 { rs[0] = unicode.ToUpper(rs[0]) } str = string(rs) case syntax.UpperAll: str = strings.ToUpper(str) case syntax.LowerFirst: rs := []rune(str) if len(rs) > 0 { rs[0] = unicode.ToLower(rs[0]) } str = string(rs) case syntax.LowerAll: str = strings.ToLower(str) case syntax.OtherParamOps: switch arg { case "Q": str = strconv.Quote(str) case "E": tail := str var rns []rune for tail != "" { var rn rune rn, _, tail, _ = strconv.UnquoteChar(tail, 0) rns = append(rns, rn) } str = string(rns) case "P", "A", "a": panic(fmt.Sprintf("unhandled @%s param expansion", arg)) default: panic(fmt.Sprintf("unexpected @%s param expansion", arg)) } } } return str } func removePattern(str, pattern string, fromEnd, greedy bool) string { expr, err := syntax.TranslatePattern(pattern, greedy) if err != nil { return str } switch { case fromEnd && !greedy: // use .* to get the right-most (shortest) match expr = ".*(" + expr + ")$" case fromEnd: // simple suffix expr = "(" + expr + ")$" default: // simple prefix expr = "^(" + expr + ")" } // no need to check error as TranslatePattern returns one rx := regexp.MustCompile(expr) if loc := rx.FindStringSubmatchIndex(str); loc != nil { // remove the original pattern (the submatch) str = str[:loc[2]] + str[loc[3]:] } return str }