1
0
mirror of https://github.com/go-task/task.git synced 2025-01-12 04:34:11 +02:00

Update github.com/mvdan/sh

This commit is contained in:
Andrey Nering 2017-05-17 14:49:27 -03:00
parent b590e74ce6
commit 504723bc19
11 changed files with 474 additions and 409 deletions

View File

@ -70,7 +70,7 @@ func (r *Runner) arithm(expr syntax.ArithmExpr) int {
}
return binArit(x.Op, r.arithm(x.X), r.arithm(x.Y))
default:
r.errf("unexpected arithm expr: %T", x)
r.runErr(expr.Pos(), "unexpected arithm expr: %T", x)
return 0
}
}

View File

@ -8,6 +8,7 @@ import (
"os/exec"
"path/filepath"
"strconv"
"strings"
"github.com/mvdan/sh/syntax"
)
@ -18,32 +19,31 @@ func isBuiltin(name string) bool {
"echo", "printf", "break", "continue", "pwd", "cd",
"wait", "builtin", "trap", "type", "source", "command",
"pushd", "popd", "umask", "alias", "unalias", "fg", "bg",
"getopts":
"getopts", "eval":
return true
}
return false
}
func (r *Runner) builtin(pos syntax.Pos, name string, args []string) {
exit := 0
func (r *Runner) builtinCode(pos syntax.Pos, name string, args []string) int {
switch name {
case "true", ":":
case "false":
exit = 1
return 1
case "exit":
switch len(args) {
case 0:
r.lastExit()
case 1:
if n, err := strconv.Atoi(args[0]); err != nil {
r.runErr(pos, "invalid exit code: %q", args[0])
} else {
exit = n
r.err = ExitCode(n)
r.exit = n
}
default:
r.runErr(pos, "exit cannot take multiple arguments")
}
r.lastExit()
return r.exit
case "set":
r.args = args
case "shift":
@ -58,8 +58,7 @@ func (r *Runner) builtin(pos syntax.Pos, name string, args []string) {
fallthrough
default:
r.errf("usage: shift [n]\n")
exit = 2
break
return 2
}
if len(r.args) < n {
n = len(r.args)
@ -97,8 +96,7 @@ func (r *Runner) builtin(pos syntax.Pos, name string, args []string) {
case "printf":
if len(args) == 0 {
r.errf("usage: printf format [arguments]\n")
exit = 2
break
return 2
}
var a []interface{}
for _, arg := range args[1:] {
@ -121,7 +119,7 @@ func (r *Runner) builtin(pos syntax.Pos, name string, args []string) {
fallthrough
default:
r.errf("usage: break [n]\n")
exit = 2
return 2
}
case "continue":
if !r.inLoop {
@ -139,15 +137,14 @@ func (r *Runner) builtin(pos syntax.Pos, name string, args []string) {
fallthrough
default:
r.errf("usage: continue [n]\n")
exit = 2
return 2
}
case "pwd":
r.outf("%s\n", r.getVar("PWD"))
case "cd":
if len(args) > 1 {
r.errf("usage: cd [dir]\n")
exit = 2
break
return 2
}
var dir string
if len(args) == 0 {
@ -160,13 +157,12 @@ func (r *Runner) builtin(pos syntax.Pos, name string, args []string) {
}
_, err := os.Stat(dir)
if err != nil {
exit = 1
break
return 1
}
r.Dir = dir
case "wait":
if len(args) > 0 {
r.errf("wait with args not handled yet")
r.runErr(pos, "wait with args not handled yet")
break
}
r.bgShells.Wait()
@ -175,13 +171,16 @@ func (r *Runner) builtin(pos syntax.Pos, name string, args []string) {
break
}
if !isBuiltin(args[0]) {
exit = 1
break
return 1
}
// TODO: pos
r.builtin(0, args[0], args[1:])
return r.builtinCode(pos, args[0], args[1:])
case "type":
anyNotFound := false
for _, arg := range args {
if _, ok := r.funcs[arg]; ok {
r.outf("%s is a function\n", arg)
continue
}
if isBuiltin(arg) {
r.outf("%s is a shell builtin\n", arg)
continue
@ -190,12 +189,27 @@ func (r *Runner) builtin(pos syntax.Pos, name string, args []string) {
r.outf("%s is %s\n", arg, path)
continue
}
exit = 1
r.errf("type: %s: not found\n", arg)
anyNotFound = true
}
if anyNotFound {
return 1
}
case "eval":
src := strings.Join(args, " ")
p := syntax.NewParser()
file, err := p.Parse(strings.NewReader(src), "")
if err != nil {
r.errf("eval: %v\n", err)
return 1
}
r2 := *r
r2.File = file
r2.Run()
return r2.exit
case "trap", "source", "command", "pushd", "popd",
"umask", "alias", "unalias", "fg", "bg", "getopts":
r.errf("unhandled builtin: %s", name)
r.runErr(pos, "unhandled builtin: %s", name)
}
r.exit = exit
return 0
}

View File

@ -262,26 +262,25 @@ func (r *Runner) stmt(st *syntax.Stmt) {
}
}
func (r *Runner) assignValue(word *syntax.Word) varValue {
if word == nil {
return nil
func (r *Runner) assignValue(as *syntax.Assign) varValue {
if as.Value != nil {
return r.loneWord(as.Value)
}
ae, ok := word.Parts[0].(*syntax.ArrayExpr)
if !ok {
return r.loneWord(word)
if as.Array != nil {
strs := make([]string, len(as.Array.List))
for i, w := range as.Array.List {
strs[i] = r.loneWord(w)
}
return strs
}
strs := make([]string, len(ae.List))
for i, w := range ae.List {
strs[i] = r.loneWord(w)
}
return strs
return nil
}
func (r *Runner) stmtSync(st *syntax.Stmt) {
oldVars := r.cmdVars
for _, as := range st.Assigns {
name := as.Name.Value
val := r.assignValue(as.Value)
val := r.assignValue(as)
if st.Cmd == nil {
r.setVar(name, val)
continue
@ -383,22 +382,9 @@ func (r *Runner) cmd(cm syntax.Command) {
case *syntax.WhileClause:
for r.err == nil {
r.stmts(x.CondStmts)
if r.exit != 0 {
r.exit = 0
break
}
if r.loopStmtsBroken(x.DoStmts) {
break
}
}
case *syntax.UntilClause:
for r.err == nil {
r.stmts(x.CondStmts)
if r.exit == 0 {
break
}
stop := (r.exit == 0) == x.Until
r.exit = 0
if r.loopStmtsBroken(x.DoStmts) {
if stop || r.loopStmtsBroken(x.DoStmts) {
break
}
}
@ -440,9 +426,7 @@ func (r *Runner) cmd(cm syntax.Command) {
for _, pl := range x.List {
for _, word := range pl.Patterns {
pat := r.loneWord(word)
// TODO: error?
matched, _ := path.Match(pat, str)
if matched {
if match(pat, str) {
r.stmts(pl.Stmts)
return
}
@ -453,7 +437,7 @@ func (r *Runner) cmd(cm syntax.Command) {
r.exit = 1
}
default:
r.errf("unhandled command node: %T", x)
r.runErr(cm.Pos(), "unhandled command node: %T", x)
}
}
@ -463,6 +447,11 @@ func (r *Runner) stmts(stmts []*syntax.Stmt) {
}
}
func match(pattern, name string) bool {
matched, _ := path.Match(pattern, name)
return matched
}
func (r *Runner) redir(rd *syntax.Redirect) (io.Closer, error) {
if rd.Hdoc != nil {
hdoc := r.loneWord(rd.Hdoc)
@ -491,7 +480,7 @@ func (r *Runner) redir(rd *syntax.Redirect) (io.Closer, error) {
}
return nil, nil
case syntax.DplIn:
r.errf("unhandled redirect op: %v", rd.Op)
r.runErr(rd.Pos(), "unhandled redirect op: %v", rd.Op)
}
mode := os.O_RDONLY
switch rd.Op {
@ -514,7 +503,7 @@ func (r *Runner) redir(rd *syntax.Redirect) (io.Closer, error) {
r.Stdout = f
r.Stderr = f
default:
r.errf("unhandled redirect op: %v", rd.Op)
r.runErr(rd.Pos(), "unhandled redirect op: %v", rd.Op)
}
return f, nil
}
@ -605,7 +594,7 @@ func (r *Runner) wordParts(wps []syntax.WordPart, quoted bool) []string {
case *syntax.ArithmExp:
curBuf.WriteString(strconv.Itoa(r.arithm(x.X)))
default:
r.errf("unhandled word part: %T", x)
r.runErr(wp.Pos(), "unhandled word part: %T", x)
}
}
flush()
@ -622,7 +611,7 @@ func (r *Runner) call(pos syntax.Pos, name string, args []string) {
return
}
if isBuiltin(name) {
r.builtin(pos, name, args)
r.exit = r.builtinCode(pos, name, args)
return
}
cmd := exec.CommandContext(r.Context, name, args...)
@ -639,10 +628,9 @@ func (r *Runner) call(pos syntax.Pos, name string, args []string) {
case *exec.ExitError:
// started, but errored - default to 1 if OS
// doesn't have exit statuses
r.exit = 1
if status, ok := x.Sys().(syscall.WaitStatus); ok {
r.exit = status.ExitStatus()
} else {
r.exit = 1
}
case *exec.Error:
// did not start

View File

@ -4,7 +4,6 @@
package interp
import (
"path"
"strconv"
"strings"
"unicode"
@ -34,8 +33,8 @@ func (r *Runner) paramExp(pe *syntax.ParamExp) string {
}
}
str := varStr(val)
if pe.Ind != nil {
str = r.varInd(val, pe.Ind.Expr)
if pe.Index != nil {
str = r.varInd(val, pe.Index)
}
switch {
case pe.Length:
@ -155,9 +154,9 @@ func (r *Runner) paramExp(pe *syntax.ParamExp) string {
}
str = string(rns)
case "P", "A", "a":
r.errf("unhandled @%s param expansion", arg)
r.runErr(pe.Pos(), "unhandled @%s param expansion", arg)
default:
r.errf("unexpected @%s param expansion", arg)
r.runErr(pe.Pos(), "unexpected @%s param expansion", arg)
}
}
}
@ -173,7 +172,7 @@ func removePattern(str, pattern string, fromEnd, longest bool) string {
i = 0
}
for {
if m, _ := path.Match(pattern, s); m {
if match(pattern, s) {
last = str[i:]
if fromEnd {
last = str[:i]

View File

@ -6,7 +6,6 @@ package interp
import (
"os"
"os/exec"
"path"
"path/filepath"
"regexp"
@ -75,11 +74,9 @@ func (r *Runner) binTest(op syntax.BinTestOperator, x, y string) bool {
case syntax.OrTest:
return x != "" || y != ""
case syntax.TsMatch:
m, _ := path.Match(y, x)
return m
return match(y, x)
case syntax.TsNoMatch:
m, _ := path.Match(y, x)
return !m
return !match(y, x)
case syntax.TsBefore:
return x < y
default: // syntax.TsAfter
@ -154,7 +151,7 @@ func (r *Runner) unTest(op syntax.UnTestOperator, x string) bool {
case syntax.TsNot:
return x == ""
default:
r.errf("unhandled unary test op: %v", op)
r.runErr(0, "unhandled unary test op: %v", op)
return false
}
}

View File

@ -46,7 +46,7 @@ func wordBreak(r rune) bool {
return false
}
func (p *parser) rune() rune {
func (p *Parser) rune() rune {
retry:
if p.npos < len(p.bs) {
if b := p.bs[p.npos]; b < utf8.RuneSelf {
@ -86,7 +86,7 @@ retry:
return p.r
}
func (p *parser) unrune(r rune) {
func (p *Parser) unrune(r rune) {
if p.r != utf8.RuneSelf {
p.npos -= utf8.RuneLen(p.r)
p.r = r
@ -96,7 +96,7 @@ func (p *parser) unrune(r rune) {
// fill reads more bytes from the input src into readBuf. Any bytes that
// had not yet been used at the end of the buffer are slid into the
// beginning of the buffer.
func (p *parser) fill() {
func (p *Parser) fill() {
left := len(p.bs) - p.npos
p.offs += p.npos
copy(p.readBuf[:left], p.readBuf[p.npos:])
@ -124,7 +124,7 @@ func (p *parser) fill() {
p.npos = 0
}
func (p *parser) nextKeepSpaces() {
func (p *Parser) nextKeepSpaces() {
r := p.r
if p.pos = p.getPos(); r > utf8.RuneSelf {
p.pos -= Pos(utf8.RuneLen(r) - 1)
@ -174,7 +174,7 @@ func (p *parser) nextKeepSpaces() {
}
}
func (p *parser) next() {
func (p *Parser) next() {
if p.r == utf8.RuneSelf {
p.tok = _EOF
return
@ -231,7 +231,7 @@ skipSpace:
for r != utf8.RuneSelf && r != '\n' {
r = p.rune()
}
if p.mode&ParseComments > 0 {
if p.keepComments {
p.f.Comments = append(p.f.Comments, &Comment{
Hash: p.pos,
Text: p.endLit(),
@ -282,14 +282,14 @@ skipSpace:
}
}
func (p *parser) peekByte(b byte) bool {
func (p *Parser) peekByte(b byte) bool {
if p.npos == len(p.bs) && p.readErr == nil {
p.fill()
}
return p.npos < len(p.bs) && p.bs[p.npos] == b
}
func (p *parser) regToken(r rune) token {
func (p *Parser) regToken(r rune) token {
switch r {
case '\'':
p.rune()
@ -432,7 +432,7 @@ func (p *parser) regToken(r rune) token {
}
}
func (p *parser) dqToken(r rune) token {
func (p *Parser) dqToken(r rune) token {
switch r {
case '"':
p.rune()
@ -462,7 +462,7 @@ func (p *parser) dqToken(r rune) token {
}
}
func (p *parser) paramToken(r rune) token {
func (p *Parser) paramToken(r rune) token {
switch r {
case '}':
p.rune()
@ -537,7 +537,7 @@ func (p *parser) paramToken(r rune) token {
}
}
func (p *parser) arithmToken(r rune) token {
func (p *Parser) arithmToken(r rune) token {
switch r {
case '!':
if p.rune() == '=' {
@ -666,7 +666,7 @@ func (p *parser) arithmToken(r rune) token {
}
}
func (p *parser) newLit(r rune) {
func (p *Parser) newLit(r rune) {
// don't let r == utf8.RuneSelf go to the second case as RuneLen
// would return -1
if r <= utf8.RuneSelf {
@ -678,9 +678,9 @@ func (p *parser) newLit(r rune) {
}
}
func (p *parser) discardLit(n int) { p.litBs = p.litBs[:len(p.litBs)-n] }
func (p *Parser) discardLit(n int) { p.litBs = p.litBs[:len(p.litBs)-n] }
func (p *parser) endLit() (s string) {
func (p *Parser) endLit() (s string) {
if p.r == utf8.RuneSelf {
s = string(p.litBs)
} else if len(p.litBs) > 0 {
@ -690,7 +690,7 @@ func (p *parser) endLit() (s string) {
return
}
func (p *parser) advanceLitOther(r rune) {
func (p *Parser) advanceLitOther(r rune) {
tok := _LitWord
loop:
for p.newLit(r); r != utf8.RuneSelf; r = p.rune() {
@ -756,6 +756,9 @@ loop:
if p.quote&allParamReg != 0 {
break loop
}
if r == '[' && p.bash() && p.quote&allArithmExpr != 0 {
break loop
}
case '+':
if p.quote == paramName && p.peekByte('(') {
tok = _Lit
@ -764,8 +767,7 @@ loop:
fallthrough
case '-':
switch p.quote {
case paramExpInd, paramExpLen, paramExpOff,
paramExpExp, paramExpRepl, sglQuotes:
case paramExpExp, paramExpRepl, sglQuotes:
default:
break loop
}
@ -780,7 +782,7 @@ loop:
p.tok, p.val = tok, p.endLit()
}
func (p *parser) advanceLitNone(r rune) {
func (p *Parser) advanceLitNone(r rune) {
p.asPos = 0
tok := _LitWord
loop:
@ -826,12 +828,17 @@ loop:
}
case '=':
p.asPos = len(p.litBs) - 1
case '[':
if p.bash() && len(p.litBs) > 1 && p.litBs[0] != '[' {
tok = _Lit
break loop
}
}
}
p.tok, p.val = tok, p.endLit()
}
func (p *parser) advanceLitDquote(r rune) {
func (p *Parser) advanceLitDquote(r rune) {
tok := _LitWord
loop:
for p.newLit(r); r != utf8.RuneSelf; r = p.rune() {
@ -848,7 +855,7 @@ loop:
p.tok, p.val = tok, p.endLit()
}
func (p *parser) advanceLitHdoc(r rune) {
func (p *Parser) advanceLitHdoc(r rune) {
p.tok = _Lit
p.newLit(r)
if p.quote == hdocBodyTabs {
@ -886,7 +893,7 @@ loop:
}
}
func (p *parser) hdocLitWord() *Word {
func (p *Parser) hdocLitWord() *Word {
r := p.r
p.newLit(r)
pos, val := p.getPos(), ""
@ -918,7 +925,7 @@ func (p *parser) hdocLitWord() *Word {
return p.word(p.wps(l))
}
func (p *parser) advanceLitRe(r rune) {
func (p *Parser) advanceLitRe(r rune) {
lparens := 0
loop:
for p.newLit(r); r != utf8.RuneSelf; r = p.rune() {

View File

@ -145,9 +145,9 @@ func (s *Stmt) End() Pos {
// Command represents all nodes that are simple commands, which are
// directly placed in a Stmt.
//
// These are *CallExpr, *IfClause, *WhileClause, *UntilClause,
// *ForClause, *CaseClause, *Block, *Subshell, *BinaryCmd, *FuncDecl,
// *ArithmCmd, *TestClause, *DeclClause, *LetClause, and *CoprocClause.
// These are *CallExpr, *IfClause, *WhileClause, *ForClause,
// *CaseClause, *Block, *Subshell, *BinaryCmd, *FuncDecl, *ArithmCmd,
// *TestClause, *DeclClause, *LetClause, and *CoprocClause.
type Command interface {
Node
commandNode()
@ -156,7 +156,6 @@ type Command interface {
func (*CallExpr) commandNode() {}
func (*IfClause) commandNode() {}
func (*WhileClause) commandNode() {}
func (*UntilClause) commandNode() {}
func (*ForClause) commandNode() {}
func (*CaseClause) commandNode() {}
func (*Block) commandNode() {}
@ -173,7 +172,9 @@ func (*CoprocClause) commandNode() {}
type Assign struct {
Append bool
Name *Lit
Index ArithmExpr
Value *Word
Array *ArrayExpr
}
func (a *Assign) Pos() Pos {
@ -187,6 +188,12 @@ func (a *Assign) End() Pos {
if a.Value != nil {
return a.Value.End()
}
if a.Array != nil {
return a.Array.End()
}
if a.Index != nil {
return a.Index.End() + 2
}
return a.Name.End() + 1
}
@ -253,9 +260,10 @@ type Elif struct {
ThenStmts []*Stmt
}
// WhileClause represents a while clause.
// WhileClause represents a while or an until clause.
type WhileClause struct {
While, Do, Done Pos
Until bool
CondStmts []*Stmt
DoStmts []*Stmt
}
@ -263,16 +271,6 @@ type WhileClause struct {
func (w *WhileClause) Pos() Pos { return w.While }
func (w *WhileClause) End() Pos { return w.Done + 4 }
// UntilClause represents an until clause.
type UntilClause struct {
Until, Do, Done Pos
CondStmts []*Stmt
DoStmts []*Stmt
}
func (u *UntilClause) Pos() Pos { return u.Until }
func (u *UntilClause) End() Pos { return u.Done + 4 }
// ForClause represents a for clause.
type ForClause struct {
For, Do, Done Pos
@ -347,7 +345,7 @@ func (w *Word) End() Pos { return w.Parts[len(w.Parts)-1].End() }
// WordPart represents all nodes that can form a word.
//
// These are *Lit, *SglQuoted, *DblQuoted, *ParamExp, *CmdSubst,
// *ArithmExp, *ProcSubst, *ArrayExpr, and *ExtGlob.
// *ArithmExp, *ProcSubst, and *ExtGlob.
type WordPart interface {
Node
wordPartNode()
@ -360,7 +358,6 @@ func (*ParamExp) wordPartNode() {}
func (*CmdSubst) wordPartNode() {}
func (*ArithmExp) wordPartNode() {}
func (*ProcSubst) wordPartNode() {}
func (*ArrayExpr) wordPartNode() {}
func (*ExtGlob) wordPartNode() {}
// Lit represents an unquoted string consisting of characters that were
@ -423,7 +420,7 @@ type ParamExp struct {
Indirect bool
Length bool
Param *Lit
Ind *Index
Index ArithmExpr
Slice *Slice
Repl *Replace
Exp *Expansion
@ -437,12 +434,7 @@ func (p *ParamExp) End() Pos {
return p.Param.End()
}
// Index represents access to an array via an index inside a ParamExp.
//
// This node will never appear when in PosixConformant mode.
type Index struct {
Expr ArithmExpr
}
func (p *ParamExp) nakedIndex() bool { return p.Short && p.Index != nil }
// Slice represents character slicing inside a ParamExp.
//

View File

@ -8,29 +8,34 @@ import (
"fmt"
"io"
"strconv"
"sync"
"unicode/utf8"
)
// ParseMode controls the parser behaviour via a set of flags.
type ParseMode uint
func KeepComments(p *Parser) { p.keepComments = true }
type LangVariant int
const (
ParseComments ParseMode = 1 << iota // add comments to the AST
PosixConformant // match the POSIX standard where it differs from bash
LangBash LangVariant = iota
LangPOSIX
)
var parserFree = sync.Pool{
New: func() interface{} {
return &parser{helperBuf: new(bytes.Buffer)}
},
func Variant(l LangVariant) func(*Parser) {
return func(p *Parser) { p.lang = l }
}
func NewParser(options ...func(*Parser)) *Parser {
p := &Parser{helperBuf: new(bytes.Buffer)}
for _, opt := range options {
opt(p)
}
return p
}
// Parse reads and parses a shell program with an optional name. It
// returns the parsed program if no issues were encountered. Otherwise,
// an error is returned.
func Parse(src io.Reader, name string, mode ParseMode) (*File, error) {
p := parserFree.Get().(*parser)
func (p *Parser) Parse(src io.Reader, name string) (*File, error) {
p.reset()
alloc := &struct {
f File
@ -39,7 +44,7 @@ func Parse(src io.Reader, name string, mode ParseMode) (*File, error) {
p.f = &alloc.f
p.f.Name = name
p.f.lines = alloc.l[:1]
p.src, p.mode = src, mode
p.src = src
p.rune()
p.next()
p.f.Stmts = p.stmts()
@ -48,18 +53,15 @@ func Parse(src io.Reader, name string, mode ParseMode) (*File, error) {
// trigger it
p.doHeredocs()
}
f, err := p.f, p.err
parserFree.Put(p)
return f, err
return p.f, p.err
}
type parser struct {
type Parser struct {
src io.Reader
bs []byte // current chunk of read bytes
r rune
f *File
mode ParseMode
f *File
spaced bool // whether tok has whitespace on its left
newLine bool // whether tok is on a new line
@ -77,6 +79,9 @@ type parser struct {
quote quoteState // current lexer state
asPos int // position of '=' in a literal
keepComments bool
lang LangVariant
forbidNested bool
// list of pending heredoc bodies
@ -100,7 +105,7 @@ type parser struct {
const bufSize = 1 << 10
func (p *parser) reset() {
func (p *Parser) reset() {
p.bs = nil
p.offs, p.npos = 0, 0
p.r, p.err, p.readErr = 0, nil, nil
@ -108,9 +113,9 @@ func (p *parser) reset() {
p.heredocs, p.buriedHdocs = p.heredocs[:0], 0
}
func (p *parser) getPos() Pos { return Pos(p.offs + p.npos) }
func (p *Parser) getPos() Pos { return Pos(p.offs + p.npos) }
func (p *parser) lit(pos Pos, val string) *Lit {
func (p *Parser) lit(pos Pos, val string) *Lit {
if len(p.litBatch) == 0 {
p.litBatch = make([]Lit, 64)
}
@ -122,7 +127,7 @@ func (p *parser) lit(pos Pos, val string) *Lit {
return l
}
func (p *parser) word(parts []WordPart) *Word {
func (p *Parser) word(parts []WordPart) *Word {
if len(p.wordBatch) == 0 {
p.wordBatch = make([]Word, 32)
}
@ -132,7 +137,7 @@ func (p *parser) word(parts []WordPart) *Word {
return w
}
func (p *parser) wps(wp WordPart) []WordPart {
func (p *Parser) wps(wp WordPart) []WordPart {
if len(p.wpsBatch) == 0 {
p.wpsBatch = make([]WordPart, 64)
}
@ -142,7 +147,7 @@ func (p *parser) wps(wp WordPart) []WordPart {
return wps
}
func (p *parser) stmt(pos Pos) *Stmt {
func (p *Parser) stmt(pos Pos) *Stmt {
if len(p.stmtBatch) == 0 {
p.stmtBatch = make([]Stmt, 16)
}
@ -152,7 +157,7 @@ func (p *parser) stmt(pos Pos) *Stmt {
return s
}
func (p *parser) stList() []*Stmt {
func (p *Parser) stList() []*Stmt {
if len(p.stListBatch) == 0 {
p.stListBatch = make([]*Stmt, 128)
}
@ -166,7 +171,7 @@ type callAlloc struct {
ws [4]*Word
}
func (p *parser) call(w *Word) *CallExpr {
func (p *Parser) call(w *Word) *CallExpr {
if len(p.callBatch) == 0 {
p.callBatch = make([]callAlloc, 32)
}
@ -208,30 +213,30 @@ const (
allRegTokens = noState | subCmd | subCmdBckquo | hdocWord | switchCase
allArithmExpr = arithmExpr | arithmExprLet | arithmExprCmd |
arithmExprBrack | allParamArith
allRbrack = arithmExprBrack | paramExpInd
allRbrack = arithmExprBrack | paramExpInd | paramName
allParamArith = paramExpInd | paramExpOff | paramExpLen
allParamReg = paramName | paramExpName | allParamArith
allParamExp = allParamReg | paramExpRepl | paramExpExp
)
func (p *parser) bash() bool { return p.mode&PosixConformant == 0 }
func (p *Parser) bash() bool { return p.lang == LangBash }
type saveState struct {
quote quoteState
buriedHdocs int
}
func (p *parser) preNested(quote quoteState) (s saveState) {
func (p *Parser) preNested(quote quoteState) (s saveState) {
s.quote, s.buriedHdocs = p.quote, p.buriedHdocs
p.buriedHdocs, p.quote = len(p.heredocs), quote
return
}
func (p *parser) postNested(s saveState) {
func (p *Parser) postNested(s saveState) {
p.quote, p.buriedHdocs = s.quote, s.buriedHdocs
}
func (p *parser) unquotedWordBytes(w *Word) ([]byte, bool) {
func (p *Parser) unquotedWordBytes(w *Word) ([]byte, bool) {
p.helperBuf.Reset()
didUnquote := false
for _, wp := range w.Parts {
@ -242,7 +247,7 @@ func (p *parser) unquotedWordBytes(w *Word) ([]byte, bool) {
return p.helperBuf.Bytes(), didUnquote
}
func (p *parser) unquotedWordPart(buf *bytes.Buffer, wp WordPart, quotes bool) (quoted bool) {
func (p *Parser) unquotedWordPart(buf *bytes.Buffer, wp WordPart, quotes bool) (quoted bool) {
switch x := wp.(type) {
case *Lit:
for i := 0; i < len(x.Value); i++ {
@ -267,7 +272,7 @@ func (p *parser) unquotedWordPart(buf *bytes.Buffer, wp WordPart, quotes bool) (
return
}
func (p *parser) doHeredocs() {
func (p *Parser) doHeredocs() {
old := p.quote
hdocs := p.heredocs[p.buriedHdocs:]
p.heredocs = p.heredocs[:p.buriedHdocs]
@ -295,7 +300,7 @@ func (p *parser) doHeredocs() {
p.quote = old
}
func (p *parser) got(tok token) bool {
func (p *Parser) got(tok token) bool {
if p.tok == tok {
p.next()
return true
@ -303,7 +308,7 @@ func (p *parser) got(tok token) bool {
return false
}
func (p *parser) gotRsrv(val string) bool {
func (p *Parser) gotRsrv(val string) bool {
if p.tok == _LitWord && p.val == val {
p.next()
return true
@ -311,7 +316,7 @@ func (p *parser) gotRsrv(val string) bool {
return false
}
func (p *parser) gotSameLine(tok token) bool {
func (p *Parser) gotSameLine(tok token) bool {
if !p.newLine && p.tok == tok {
p.next()
return true
@ -327,16 +332,16 @@ func readableStr(s string) string {
return s
}
func (p *parser) followErr(pos Pos, left, right string) {
func (p *Parser) followErr(pos Pos, left, right string) {
leftStr := readableStr(left)
p.posErr(pos, "%s must be followed by %s", leftStr, right)
}
func (p *parser) followErrExp(pos Pos, left string) {
func (p *Parser) followErrExp(pos Pos, left string) {
p.followErr(pos, left, "an expression")
}
func (p *parser) follow(lpos Pos, left string, tok token) Pos {
func (p *Parser) follow(lpos Pos, left string, tok token) Pos {
pos := p.pos
if !p.got(tok) {
p.followErr(lpos, left, tok.String())
@ -344,7 +349,7 @@ func (p *parser) follow(lpos Pos, left string, tok token) Pos {
return pos
}
func (p *parser) followRsrv(lpos Pos, left, val string) Pos {
func (p *Parser) followRsrv(lpos Pos, left, val string) Pos {
pos := p.pos
if !p.gotRsrv(val) {
p.followErr(lpos, left, fmt.Sprintf("%q", val))
@ -352,7 +357,7 @@ func (p *parser) followRsrv(lpos Pos, left, val string) Pos {
return pos
}
func (p *parser) followStmts(left string, lpos Pos, stops ...string) []*Stmt {
func (p *Parser) followStmts(left string, lpos Pos, stops ...string) []*Stmt {
if p.gotSameLine(semicolon) {
return nil
}
@ -363,7 +368,7 @@ func (p *parser) followStmts(left string, lpos Pos, stops ...string) []*Stmt {
return sts
}
func (p *parser) followWordTok(tok token, pos Pos) *Word {
func (p *Parser) followWordTok(tok token, pos Pos) *Word {
w := p.getWord()
if w == nil {
p.followErr(pos, tok.String(), "a word")
@ -371,7 +376,7 @@ func (p *parser) followWordTok(tok token, pos Pos) *Word {
return w
}
func (p *parser) followWord(s string, pos Pos) *Word {
func (p *Parser) followWord(s string, pos Pos) *Word {
w := p.getWord()
if w == nil {
p.followErr(pos, s, "a word")
@ -379,7 +384,7 @@ func (p *parser) followWord(s string, pos Pos) *Word {
return w
}
func (p *parser) stmtEnd(n Node, start, end string) Pos {
func (p *Parser) stmtEnd(n Node, start, end string) Pos {
pos := p.pos
if !p.gotRsrv(end) {
p.posErr(n.Pos(), "%s statement must end with %q", start, end)
@ -387,17 +392,17 @@ func (p *parser) stmtEnd(n Node, start, end string) Pos {
return pos
}
func (p *parser) quoteErr(lpos Pos, quote token) {
func (p *Parser) quoteErr(lpos Pos, quote token) {
p.posErr(lpos, "reached %s without closing quote %s",
p.tok.String(), quote)
}
func (p *parser) matchingErr(lpos Pos, left, right interface{}) {
func (p *Parser) matchingErr(lpos Pos, left, right interface{}) {
p.posErr(lpos, "reached %s without matching %s with %s",
p.tok.String(), left, right)
}
func (p *parser) matched(lpos Pos, left, right token) Pos {
func (p *Parser) matched(lpos Pos, left, right token) Pos {
pos := p.pos
if !p.got(right) {
p.matchingErr(lpos, left, right)
@ -405,7 +410,7 @@ func (p *parser) matched(lpos Pos, left, right token) Pos {
return pos
}
func (p *parser) errPass(err error) {
func (p *Parser) errPass(err error) {
if p.err == nil {
p.err = err
p.npos = len(p.bs) + 1
@ -424,18 +429,18 @@ func (e *ParseError) Error() string {
return fmt.Sprintf("%s: %s", e.Position.String(), e.Text)
}
func (p *parser) posErr(pos Pos, format string, a ...interface{}) {
func (p *Parser) posErr(pos Pos, format string, a ...interface{}) {
p.errPass(&ParseError{
Position: p.f.Position(pos),
Text: fmt.Sprintf(format, a...),
})
}
func (p *parser) curErr(format string, a ...interface{}) {
func (p *Parser) curErr(format string, a ...interface{}) {
p.posErr(p.pos, format, a...)
}
func (p *parser) stmts(stops ...string) (sts []*Stmt) {
func (p *Parser) stmts(stops ...string) (sts []*Stmt) {
gotEnd := true
for p.tok != _EOF {
switch p.tok {
@ -478,7 +483,7 @@ func (p *parser) stmts(stops ...string) (sts []*Stmt) {
return
}
func (p *parser) invalidStmtStart() {
func (p *Parser) invalidStmtStart() {
switch p.tok {
case semicolon, and, or, andAnd, orOr:
p.curErr("%s can only immediately follow a statement", p.tok)
@ -489,14 +494,14 @@ func (p *parser) invalidStmtStart() {
}
}
func (p *parser) getWord() *Word {
func (p *Parser) getWord() *Word {
if parts := p.wordParts(); len(parts) > 0 {
return p.word(parts)
}
return nil
}
func (p *parser) getWordOrEmpty() *Word {
func (p *Parser) getWordOrEmpty() *Word {
parts := p.wordParts()
if len(parts) == 0 {
l := p.lit(p.pos, "")
@ -506,7 +511,7 @@ func (p *parser) getWordOrEmpty() *Word {
return p.word(parts)
}
func (p *parser) getLit() *Lit {
func (p *Parser) getLit() *Lit {
switch p.tok {
case _Lit, _LitWord, _LitRedir:
l := p.lit(p.pos, p.val)
@ -516,7 +521,7 @@ func (p *parser) getLit() *Lit {
return nil
}
func (p *parser) wordParts() (wps []WordPart) {
func (p *Parser) wordParts() (wps []WordPart) {
for {
n := p.wordPart()
if n == nil {
@ -533,13 +538,13 @@ func (p *parser) wordParts() (wps []WordPart) {
}
}
func (p *parser) ensureNoNested() {
func (p *Parser) ensureNoNested() {
if p.forbidNested {
p.curErr("expansions not allowed in heredoc words")
}
}
func (p *parser) wordPart() WordPart {
func (p *Parser) wordPart() WordPart {
switch p.tok {
case _Lit, _LitWord:
l := p.lit(p.pos, p.val)
@ -581,27 +586,8 @@ func (p *parser) wordPart() WordPart {
cs.Right = p.matched(cs.Left, leftParen, rightParen)
return cs
case dollar:
r := p.r
if r == utf8.RuneSelf || wordBreak(r) || r == '"' || r == '\'' || r == '`' || r == '[' {
l := p.lit(p.pos, "$")
p.next()
return l
}
p.ensureNoNested()
pe := &ParamExp{Dollar: p.pos, Short: true}
p.pos++
switch r {
case '@', '*', '#', '$', '?', '!', '0', '-':
p.rune()
p.tok, p.val = _LitWord, string(r)
default:
old := p.quote
p.quote = paramName
p.advanceLitOther(r)
p.quote = old
}
pe.Param = p.getLit()
return pe
return p.shortParamExp()
case cmdIn, cmdOut:
p.ensureNoNested()
ps := &ProcSubst{Op: ProcOperator(p.tok), OpPos: p.pos}
@ -736,7 +722,7 @@ func arithmOpLevel(op BinAritOperator) int {
return -1
}
func (p *parser) arithmExpr(ftok token, fpos Pos, level int, compact, tern bool) ArithmExpr {
func (p *Parser) arithmExpr(ftok token, fpos Pos, level int, compact, tern bool) ArithmExpr {
if p.tok == _EOF || p.peekArithmEnd() {
return nil
}
@ -804,7 +790,7 @@ func (p *parser) arithmExpr(ftok token, fpos Pos, level int, compact, tern bool)
return b
}
func (p *parser) arithmExprBase(compact bool) ArithmExpr {
func (p *Parser) arithmExprBase(compact bool) ArithmExpr {
var x ArithmExpr
switch p.tok {
case exclMark:
@ -837,16 +823,34 @@ func (p *parser) arithmExprBase(compact bool) ArithmExpr {
if p.next(); compact && p.spaced {
p.followErrExp(ue.OpPos, ue.Op.String())
}
ue.X = p.arithmExpr(token(ue.Op), ue.OpPos, 0, compact, false)
ue.X = p.arithmExprBase(compact)
if ue.X == nil {
p.followErrExp(ue.OpPos, ue.Op.String())
}
x = ue
case illegalTok, rightBrack, rightBrace, rightParen:
case _LitWord:
x = p.getLit()
case dollar, dollBrace:
x = p.wordPart().(*ParamExp)
l := p.getLit()
if p.r != '[' {
x = l
break
}
pe := &ParamExp{Dollar: l.ValuePos, Short: true, Param: l}
p.rune()
left := p.pos + 1
old := p.preNested(arithmExprBrack)
p.next()
pe.Index = p.arithmExpr(leftBrack, left, 0, false, false)
if pe.Index == nil {
p.followErrExp(left, "[")
}
p.postNested(old)
p.matched(left, leftBrack, rightBrack)
x = pe
case dollar:
x = p.shortParamExp()
case dollBrace:
x = p.paramExp()
case bckQuote:
if p.quote == arithmExprLet {
return nil
@ -862,7 +866,16 @@ func (p *parser) arithmExprBase(compact bool) ArithmExpr {
return x
}
if p.tok == addAdd || p.tok == subSub {
if l, ok := x.(*Lit); !ok || !validIdent(l.Value, p.bash()) {
switch y := x.(type) {
case *Lit:
if !validIdent(y.Value, p.bash()) {
p.curErr("%s must follow a name", p.tok.String())
}
case *ParamExp:
if !y.nakedIndex() {
p.curErr("%s must follow a name", p.tok.String())
}
default:
p.curErr("%s must follow a name", p.tok.String())
}
u := &UnaryArithm{
@ -877,7 +890,27 @@ func (p *parser) arithmExprBase(compact bool) ArithmExpr {
return x
}
func (p *parser) paramExp() *ParamExp {
func (p *Parser) shortParamExp() *ParamExp {
pe := &ParamExp{Dollar: p.pos, Short: true}
p.pos++
switch p.r {
case '@', '*', '#', '$', '?', '!', '0', '-':
p.tok, p.val = _LitWord, string(p.r)
p.rune()
default:
old := p.quote
p.quote = paramName
p.advanceLitOther(p.r)
p.quote = old
if p.val == "" || p.val == "\x80" {
p.posErr(pe.Dollar, "$ must be escaped or followed by a literal")
}
}
pe.Param = p.getLit()
return pe
}
func (p *Parser) paramExp() *ParamExp {
pe := &ParamExp{Dollar: p.pos}
old := p.quote
p.quote = paramExpName
@ -934,15 +967,11 @@ func (p *parser) paramExp() *ParamExp {
p.quote = paramExpInd
p.next()
switch p.tok {
case star:
p.tok, p.val = _LitWord, "*"
case at:
p.tok, p.val = _LitWord, "@"
case star, at:
p.tok, p.val = _LitWord, p.tok.String()
}
pe.Ind = &Index{
Expr: p.arithmExpr(leftBrack, lpos, 0, false, false),
}
if pe.Ind.Expr == nil {
pe.Index = p.arithmExpr(leftBrack, lpos, 0, false, false)
if pe.Index == nil {
p.followErrExp(lpos, "[")
}
p.quote = paramExpName
@ -1009,11 +1038,11 @@ func (p *parser) paramExp() *ParamExp {
return pe
}
func (p *parser) peekArithmEnd() bool {
func (p *Parser) peekArithmEnd() bool {
return p.tok == rightParen && p.r == ')'
}
func (p *parser) arithmEnd(ltok token, lpos Pos, old saveState) Pos {
func (p *Parser) arithmEnd(ltok token, lpos Pos, old saveState) Pos {
if p.peekArithmEnd() {
p.rune()
} else {
@ -1050,47 +1079,70 @@ func validIdent(val string, bash bool) bool {
return true
}
func (p *parser) hasValidIdent() bool {
if p.asPos < 1 {
return false
func (p *Parser) hasValidIdent() bool {
if p.asPos > 0 && validIdent(p.val[:p.asPos], p.bash()) {
return true
}
return validIdent(p.val[:p.asPos], p.bash())
return p.tok == _Lit && p.r == '['
}
func (p *parser) getAssign() *Assign {
func (p *Parser) getAssign() *Assign {
as := &Assign{}
nameEnd := p.asPos
if p.bash() && p.val[p.asPos-1] == '+' {
// a+=b
as.Append = true
nameEnd--
if p.asPos > 0 { // foo=bar
nameEnd := p.asPos
if p.bash() && p.val[p.asPos-1] == '+' {
// a+=b
as.Append = true
nameEnd--
}
as.Name = p.lit(p.pos, p.val[:nameEnd])
// since we're not using the entire p.val
as.Name.ValueEnd = as.Name.ValuePos + Pos(nameEnd)
left := p.lit(p.pos+1, p.val[p.asPos+1:])
if left.Value != "" {
left.ValuePos += Pos(p.asPos)
as.Value = p.word(p.wps(left))
}
if p.next(); p.spaced {
return as
}
} else { // foo[i]=bar
as.Name = p.lit(p.pos, p.val)
// hasValidIdent already checks p.r is '['
p.rune()
left := p.pos + 1
old := p.preNested(arithmExprBrack)
p.next()
as.Index = p.arithmExpr(leftBrack, left, 0, false, false)
if as.Index == nil {
p.followErrExp(left, "[")
}
p.postNested(old)
p.matched(left, leftBrack, rightBrack)
if p.tok == _EOF || p.val[0] != '=' {
p.followErr(as.Pos(), "a[b]", "=")
return nil
}
p.pos++
p.val = p.val[1:]
if p.val == "" {
p.next()
}
}
as.Name = p.lit(p.pos, p.val[:nameEnd])
// since we're not using the entire p.val
as.Name.ValueEnd = as.Name.ValuePos + Pos(nameEnd)
start := p.lit(p.pos+1, p.val[p.asPos+1:])
if start.Value != "" {
start.ValuePos += Pos(p.asPos)
as.Value = p.word(p.wps(start))
}
if p.next(); p.spaced {
return as
}
if start.Value == "" && p.tok == leftParen {
if as.Value == nil && p.tok == leftParen {
if !p.bash() {
p.curErr("arrays are a bash feature")
}
ae := &ArrayExpr{Lparen: p.pos}
as.Array = &ArrayExpr{Lparen: p.pos}
p.next()
for p.tok != _EOF && p.tok != rightParen {
if w := p.getWord(); w == nil {
p.curErr("array elements must be words")
} else {
ae.List = append(ae.List, w)
as.Array.List = append(as.Array.List, w)
}
}
ae.Rparen = p.matched(ae.Lparen, leftParen, rightParen)
as.Value = p.word(p.wps(ae))
as.Array.Rparen = p.matched(as.Array.Lparen, leftParen, rightParen)
} else if !p.newLine && !stopToken(p.tok) {
if w := p.getWord(); w != nil {
if as.Value == nil {
@ -1103,7 +1155,7 @@ func (p *parser) getAssign() *Assign {
return as
}
func (p *parser) peekRedir() bool {
func (p *Parser) peekRedir() bool {
switch p.tok {
case rdrOut, appOut, rdrIn, dplIn, dplOut, clbOut, rdrInOut,
hdoc, dashHdoc, wordHdoc, rdrAll, appAll, _LitRedir:
@ -1112,7 +1164,7 @@ func (p *parser) peekRedir() bool {
return false
}
func (p *parser) doRedirect(s *Stmt) {
func (p *Parser) doRedirect(s *Stmt) {
r := &Redirect{}
r.N = p.getLit()
r.Op, r.OpPos = RedirOperator(p.tok), p.pos
@ -1136,7 +1188,7 @@ func (p *parser) doRedirect(s *Stmt) {
s.Redirs = append(s.Redirs, r)
}
func (p *parser) getStmt(readEnd, binCmd bool) (s *Stmt, gotEnd bool) {
func (p *Parser) getStmt(readEnd, binCmd bool) (s *Stmt, gotEnd bool) {
s = p.stmt(p.pos)
if p.gotRsrv("!") {
s.Negated = true
@ -1209,7 +1261,7 @@ preLoop:
return
}
func (p *parser) gotStmtPipe(s *Stmt) *Stmt {
func (p *Parser) gotStmtPipe(s *Stmt) *Stmt {
switch p.tok {
case _LitWord:
switch p.val {
@ -1217,10 +1269,8 @@ func (p *parser) gotStmtPipe(s *Stmt) *Stmt {
s.Cmd = p.block()
case "if":
s.Cmd = p.ifClause()
case "while":
s.Cmd = p.whileClause()
case "until":
s.Cmd = p.untilClause()
case "while", "until":
s.Cmd = p.whileClause(p.val == "until")
case "for":
s.Cmd = p.forClause()
case "case":
@ -1307,7 +1357,7 @@ func (p *parser) gotStmtPipe(s *Stmt) *Stmt {
return s
}
func (p *parser) subshell() *Subshell {
func (p *Parser) subshell() *Subshell {
s := &Subshell{Lparen: p.pos}
old := p.preNested(subCmd)
p.next()
@ -1317,7 +1367,7 @@ func (p *parser) subshell() *Subshell {
return s
}
func (p *parser) arithmExpCmd() Command {
func (p *Parser) arithmExpCmd() Command {
ar := &ArithmCmd{Left: p.pos}
old := p.preNested(arithmExprCmd)
p.next()
@ -1326,7 +1376,7 @@ func (p *parser) arithmExpCmd() Command {
return ar
}
func (p *parser) block() *Block {
func (p *Parser) block() *Block {
b := &Block{Lbrace: p.pos}
p.next()
b.Stmts = p.stmts("}")
@ -1337,7 +1387,7 @@ func (p *parser) block() *Block {
return b
}
func (p *parser) ifClause() *IfClause {
func (p *Parser) ifClause() *IfClause {
ic := &IfClause{If: p.pos}
p.next()
ic.CondStmts = p.followStmts("if", ic.If, "then")
@ -1359,27 +1409,23 @@ func (p *parser) ifClause() *IfClause {
return ic
}
func (p *parser) whileClause() *WhileClause {
wc := &WhileClause{While: p.pos}
func (p *Parser) whileClause(until bool) *WhileClause {
wc := &WhileClause{While: p.pos, Until: until}
rsrv := "while"
rsrvCond := "while <cond>"
if wc.Until {
rsrv = "until"
rsrvCond = "until <cond>"
}
p.next()
wc.CondStmts = p.followStmts("while", wc.While, "do")
wc.Do = p.followRsrv(wc.While, "while <cond>", "do")
wc.CondStmts = p.followStmts(rsrv, wc.While, "do")
wc.Do = p.followRsrv(wc.While, rsrvCond, "do")
wc.DoStmts = p.followStmts("do", wc.Do, "done")
wc.Done = p.stmtEnd(wc, "while", "done")
wc.Done = p.stmtEnd(wc, rsrv, "done")
return wc
}
func (p *parser) untilClause() *UntilClause {
uc := &UntilClause{Until: p.pos}
p.next()
uc.CondStmts = p.followStmts("until", uc.Until, "do")
uc.Do = p.followRsrv(uc.Until, "until <cond>", "do")
uc.DoStmts = p.followStmts("do", uc.Do, "done")
uc.Done = p.stmtEnd(uc, "until", "done")
return uc
}
func (p *parser) forClause() *ForClause {
func (p *Parser) forClause() *ForClause {
fc := &ForClause{For: p.pos}
p.next()
fc.Loop = p.loop(fc.For)
@ -1389,7 +1435,7 @@ func (p *parser) forClause() *ForClause {
return fc
}
func (p *parser) loop(forPos Pos) Loop {
func (p *Parser) loop(forPos Pos) Loop {
if p.tok == dblLeftParen {
cl := &CStyleLoop{Lparen: p.pos}
old := p.preNested(arithmExprCmd)
@ -1437,7 +1483,7 @@ func (p *parser) loop(forPos Pos) Loop {
return wi
}
func (p *parser) caseClause() *CaseClause {
func (p *Parser) caseClause() *CaseClause {
cc := &CaseClause{Case: p.pos}
p.next()
cc.Word = p.followWord("case", cc.Case)
@ -1447,7 +1493,7 @@ func (p *parser) caseClause() *CaseClause {
return cc
}
func (p *parser) patLists() (pls []*PatternList) {
func (p *Parser) patLists() (pls []*PatternList) {
for p.tok != _EOF && !(p.tok == _LitWord && p.val == "esac") {
pl := &PatternList{}
p.got(leftParen)
@ -1481,7 +1527,7 @@ func (p *parser) patLists() (pls []*PatternList) {
return
}
func (p *parser) testClause() *TestClause {
func (p *Parser) testClause() *TestClause {
tc := &TestClause{Left: p.pos}
if p.next(); p.tok == _EOF || p.gotRsrv("]]") {
p.posErr(tc.Left, "test clause requires at least one expression")
@ -1494,7 +1540,7 @@ func (p *parser) testClause() *TestClause {
return tc
}
func (p *parser) testExpr(ftok token, fpos Pos, level int) TestExpr {
func (p *Parser) testExpr(ftok token, fpos Pos, level int) TestExpr {
var left TestExpr
if level > 1 {
left = p.testExprBase(ftok, fpos)
@ -1517,7 +1563,7 @@ func (p *parser) testExpr(ftok token, fpos Pos, level int) TestExpr {
case _EOF, rightParen:
return left
case _Lit:
p.curErr("not a valid test operator: %s", p.val)
p.curErr("test operator words must consist of a single literal")
default:
p.curErr("not a valid test operator: %v", p.tok)
}
@ -1555,7 +1601,7 @@ func (p *parser) testExpr(ftok token, fpos Pos, level int) TestExpr {
return b
}
func (p *parser) testExprBase(ftok token, fpos Pos) TestExpr {
func (p *Parser) testExprBase(ftok token, fpos Pos) TestExpr {
switch p.tok {
case _EOF:
return nil
@ -1598,7 +1644,7 @@ func (p *parser) testExprBase(ftok token, fpos Pos) TestExpr {
}
}
func (p *parser) declClause() *DeclClause {
func (p *Parser) declClause() *DeclClause {
name := p.val
ds := &DeclClause{Position: p.pos}
switch name {
@ -1637,7 +1683,7 @@ func isBashCompoundCommand(tok token, val string) bool {
return false
}
func (p *parser) coprocClause() *CoprocClause {
func (p *Parser) coprocClause() *CoprocClause {
cc := &CoprocClause{Coproc: p.pos}
if p.next(); isBashCompoundCommand(p.tok, p.val) {
// has no name
@ -1669,7 +1715,7 @@ func (p *parser) coprocClause() *CoprocClause {
return cc
}
func (p *parser) letClause() *LetClause {
func (p *Parser) letClause() *LetClause {
lc := &LetClause{Let: p.pos}
old := p.preNested(arithmExprLet)
p.next()
@ -1690,7 +1736,7 @@ func (p *parser) letClause() *LetClause {
return lc
}
func (p *parser) bashFuncDecl() *FuncDecl {
func (p *Parser) bashFuncDecl() *FuncDecl {
fpos := p.pos
p.next()
if p.tok != _LitWord {
@ -1705,7 +1751,7 @@ func (p *parser) bashFuncDecl() *FuncDecl {
return p.funcDecl(name, fpos)
}
func (p *parser) callExpr(s *Stmt, w *Word) *CallExpr {
func (p *Parser) callExpr(s *Stmt, w *Word) *CallExpr {
ce := p.call(w)
for !p.newLine {
switch p.tok {
@ -1743,7 +1789,7 @@ func (p *parser) callExpr(s *Stmt, w *Word) *CallExpr {
return ce
}
func (p *parser) funcDecl(name *Lit, pos Pos) *FuncDecl {
func (p *Parser) funcDecl(name *Lit, pos Pos) *FuncDecl {
fd := &FuncDecl{
Position: pos,
BashStyle: pos != name.ValuePos,

View File

@ -6,54 +6,40 @@ package syntax
import (
"bufio"
"io"
"sync"
"strings"
)
// PrintConfig controls how the printing of an AST node will behave.
type PrintConfig struct {
// Spaces dictates the indentation style. The default value of 0
// uses tabs, and any positive value uses that number of spaces.
Spaces int
// BinaryNextLine makes binary operators (such as &&, || and |)
// be at the start of a line if the statement that follows them
// is on a separate line. This means that the operator will come
// after an escaped newline.
BinaryNextLine bool
func Indent(spaces int) func(*Printer) {
return func(p *Printer) { p.indentSpaces = spaces }
}
var printerFree = sync.Pool{
New: func() interface{} {
return &printer{
bufWriter: bufio.NewWriter(nil),
lenPrinter: new(printer),
}
},
func BinaryNextLine(p *Printer) { p.binNextLine = true }
func NewPrinter(options ...func(*Printer)) *Printer {
p := &Printer{
bufWriter: bufio.NewWriter(nil),
lenPrinter: new(Printer),
}
for _, opt := range options {
opt(p)
}
return p
}
// Fprint "pretty-prints" the given AST file to the given writer.
func (c PrintConfig) Fprint(w io.Writer, f *File) error {
p := printerFree.Get().(*printer)
// Print "pretty-prints" the given AST file to the given writer.
func (p *Printer) Print(w io.Writer, f *File) error {
p.reset()
p.PrintConfig = c
p.lines, p.comments = f.lines, f.Comments
p.bufWriter.Reset(w)
p.stmts(f.Stmts)
p.commentsUpTo(0)
p.newline(0)
var err error
if flusher, ok := p.bufWriter.(interface {
Flush() error
}); ok {
err = flusher.Flush()
return flusher.Flush()
}
printerFree.Put(p)
return err
}
// Fprint "pretty-prints" the given AST file to the given writer. It
// calls PrintConfig.Fprint with its default settings.
func Fprint(w io.Writer, f *File) error {
return PrintConfig{}.Fprint(w, f)
return nil
}
type bufWriter interface {
@ -62,10 +48,12 @@ type bufWriter interface {
Reset(io.Writer)
}
type printer struct {
type Printer struct {
bufWriter
PrintConfig
indentSpaces int
binNextLine bool
lines []Pos
wantSpace bool
@ -94,12 +82,12 @@ type printer struct {
// pendingHdocs is the list of pending heredocs to write.
pendingHdocs []*Redirect
// used in stmtLen to align comments
lenPrinter *printer
// used in stmtCols to align comments
lenPrinter *Printer
lenCounter byteCounter
}
func (p *printer) reset() {
func (p *Printer) reset() {
p.wantSpace, p.wantNewline = false, false
p.commentPadding = 0
p.nline, p.nlineIndex = 0, 0
@ -109,7 +97,7 @@ func (p *printer) reset() {
p.pendingHdocs = p.pendingHdocs[:0]
}
func (p *printer) incLine() {
func (p *Printer) incLine() {
if p.nlineIndex++; p.nlineIndex >= len(p.lines) {
p.nline = maxPos
} else {
@ -117,19 +105,19 @@ func (p *printer) incLine() {
}
}
func (p *printer) incLines(pos Pos) {
func (p *Printer) incLines(pos Pos) {
for p.nline < pos {
p.incLine()
}
}
func (p *printer) spaces(n int) {
func (p *Printer) spaces(n int) {
for i := 0; i < n; i++ {
p.WriteByte(' ')
}
}
func (p *printer) bslashNewl() {
func (p *Printer) bslashNewl() {
if p.wantSpace {
p.WriteByte(' ')
}
@ -138,7 +126,7 @@ func (p *printer) bslashNewl() {
p.incLine()
}
func (p *printer) spacedString(s string) {
func (p *Printer) spacedString(s string) {
if p.wantSpace {
p.WriteByte(' ')
}
@ -146,7 +134,7 @@ func (p *printer) spacedString(s string) {
p.wantSpace = true
}
func (p *printer) semiOrNewl(s string, pos Pos) {
func (p *Printer) semiOrNewl(s string, pos Pos) {
if p.wantNewline {
p.newline(pos)
p.indent()
@ -161,7 +149,7 @@ func (p *printer) semiOrNewl(s string, pos Pos) {
p.wantSpace = true
}
func (p *printer) incLevel() {
func (p *Printer) incLevel() {
inc := false
if p.level <= p.lastLevel || len(p.levelIncs) == 0 {
p.level++
@ -173,27 +161,27 @@ func (p *printer) incLevel() {
p.levelIncs = append(p.levelIncs, inc)
}
func (p *printer) decLevel() {
func (p *Printer) decLevel() {
if p.levelIncs[len(p.levelIncs)-1] {
p.level--
}
p.levelIncs = p.levelIncs[:len(p.levelIncs)-1]
}
func (p *printer) indent() {
func (p *Printer) indent() {
p.lastLevel = p.level
switch {
case p.level == 0:
case p.Spaces == 0:
case p.indentSpaces == 0:
for i := 0; i < p.level; i++ {
p.WriteByte('\t')
}
case p.Spaces > 0:
p.spaces(p.Spaces * p.level)
case p.indentSpaces > 0:
p.spaces(p.indentSpaces * p.level)
}
}
func (p *printer) newline(pos Pos) {
func (p *Printer) newline(pos Pos) {
p.wantNewline, p.wantSpace = false, false
p.WriteByte('\n')
if pos > p.nline {
@ -211,7 +199,7 @@ func (p *printer) newline(pos Pos) {
}
}
func (p *printer) newlines(pos Pos) {
func (p *Printer) newlines(pos Pos) {
p.newline(pos)
if pos > p.nline {
// preserve single empty lines
@ -221,14 +209,14 @@ func (p *printer) newlines(pos Pos) {
p.indent()
}
func (p *printer) commentsAndSeparate(pos Pos) {
func (p *Printer) commentsAndSeparate(pos Pos) {
p.commentsUpTo(pos)
if p.wantNewline || pos > p.nline {
p.newlines(pos)
}
}
func (p *printer) sepTok(s string, pos Pos) {
func (p *Printer) sepTok(s string, pos Pos) {
p.level++
p.commentsUpTo(pos)
p.level--
@ -239,7 +227,7 @@ func (p *printer) sepTok(s string, pos Pos) {
p.wantSpace = true
}
func (p *printer) semiRsrv(s string, pos Pos, fallback bool) {
func (p *Printer) semiRsrv(s string, pos Pos, fallback bool) {
p.level++
p.commentsUpTo(pos)
p.level--
@ -257,14 +245,14 @@ func (p *printer) semiRsrv(s string, pos Pos, fallback bool) {
p.wantSpace = true
}
func (p *printer) anyCommentsBefore(pos Pos) bool {
func (p *Printer) anyCommentsBefore(pos Pos) bool {
if !pos.IsValid() || len(p.comments) < 1 {
return false
}
return p.comments[0].Hash < pos
}
func (p *printer) commentsUpTo(pos Pos) {
func (p *Printer) commentsUpTo(pos Pos) {
if len(p.comments) < 1 {
return
}
@ -286,7 +274,7 @@ func (p *printer) commentsUpTo(pos Pos) {
p.commentsUpTo(pos)
}
func (p *printer) wordPart(wp WordPart) {
func (p *Printer) wordPart(wp WordPart) {
switch x := wp.(type) {
case *Lit:
p.WriteString(x.Value)
@ -320,13 +308,8 @@ func (p *printer) wordPart(wp WordPart) {
p.paramExp(x)
case *ArithmExp:
p.WriteString("$((")
p.arithmExpr(x.X, false)
p.arithmExpr(x.X, false, false)
p.WriteString("))")
case *ArrayExpr:
p.wantSpace = false
p.WriteByte('(')
p.wordJoin(x.List, false)
p.sepTok(")", x.Rparen)
case *ExtGlob:
p.WriteString(x.Op.String())
p.WriteString(x.Pattern.Value)
@ -343,12 +326,20 @@ func (p *printer) wordPart(wp WordPart) {
}
}
func (p *printer) paramExp(pe *ParamExp) {
if pe.Short {
func (p *Printer) paramExp(pe *ParamExp) {
if pe.nakedIndex() { // arr[i]
p.WriteString(pe.Param.Value)
p.WriteByte('[')
p.arithmExpr(pe.Index, false, false)
p.WriteByte(']')
return
}
if pe.Short { // $var
p.WriteByte('$')
p.WriteString(pe.Param.Value)
return
}
// ${var...}
p.WriteString("${")
switch {
case pe.Length:
@ -359,23 +350,17 @@ func (p *printer) paramExp(pe *ParamExp) {
if pe.Param != nil {
p.WriteString(pe.Param.Value)
}
if pe.Ind != nil {
if pe.Index != nil {
p.WriteByte('[')
p.arithmExpr(pe.Ind.Expr, false)
p.arithmExpr(pe.Index, false, false)
p.WriteByte(']')
}
if pe.Slice != nil {
p.WriteByte(':')
if un, ok := pe.Slice.Offset.(*UnaryArithm); ok {
if un.Op == Plus || un.Op == Minus {
// to avoid :+ and :-
p.WriteByte(' ')
}
}
p.arithmExpr(pe.Slice.Offset, true)
p.arithmExpr(pe.Slice.Offset, true, true)
if pe.Slice.Length != nil {
p.WriteByte(':')
p.arithmExpr(pe.Slice.Length, true)
p.arithmExpr(pe.Slice.Length, true, false)
}
} else if pe.Repl != nil {
if pe.Repl.All {
@ -392,7 +377,7 @@ func (p *printer) paramExp(pe *ParamExp) {
p.WriteByte('}')
}
func (p *printer) loop(loop Loop) {
func (p *Printer) loop(loop Loop) {
switch x := loop.(type) {
case *WordIter:
p.WriteString(x.Name.Value)
@ -405,16 +390,16 @@ func (p *printer) loop(loop Loop) {
if x.Init == nil {
p.WriteByte(' ')
}
p.arithmExpr(x.Init, false)
p.arithmExpr(x.Init, false, false)
p.WriteString("; ")
p.arithmExpr(x.Cond, false)
p.arithmExpr(x.Cond, false, false)
p.WriteString("; ")
p.arithmExpr(x.Post, false)
p.arithmExpr(x.Post, false, false)
p.WriteString("))")
}
}
func (p *printer) arithmExpr(expr ArithmExpr, compact bool) {
func (p *Printer) arithmExpr(expr ArithmExpr, compact, spacePlusMinus bool) {
switch x := expr.(type) {
case *Lit:
p.WriteString(x.Value)
@ -422,34 +407,40 @@ func (p *printer) arithmExpr(expr ArithmExpr, compact bool) {
p.paramExp(x)
case *BinaryArithm:
if compact {
p.arithmExpr(x.X, compact)
p.arithmExpr(x.X, compact, spacePlusMinus)
p.WriteString(x.Op.String())
p.arithmExpr(x.Y, compact)
p.arithmExpr(x.Y, compact, false)
} else {
p.arithmExpr(x.X, compact)
p.arithmExpr(x.X, compact, spacePlusMinus)
if x.Op != Comma {
p.WriteByte(' ')
}
p.WriteString(x.Op.String())
p.WriteByte(' ')
p.arithmExpr(x.Y, compact)
p.arithmExpr(x.Y, compact, false)
}
case *UnaryArithm:
if x.Post {
p.arithmExpr(x.X, compact)
p.arithmExpr(x.X, compact, spacePlusMinus)
p.WriteString(x.Op.String())
} else {
if spacePlusMinus {
switch x.Op {
case Plus, Minus:
p.WriteByte(' ')
}
}
p.WriteString(x.Op.String())
p.arithmExpr(x.X, compact)
p.arithmExpr(x.X, compact, false)
}
case *ParenArithm:
p.WriteByte('(')
p.arithmExpr(x.X, false)
p.arithmExpr(x.X, false, false)
p.WriteByte(')')
}
}
func (p *printer) testExpr(expr TestExpr) {
func (p *Printer) testExpr(expr TestExpr) {
switch x := expr.(type) {
case *Word:
p.word(x)
@ -470,14 +461,14 @@ func (p *printer) testExpr(expr TestExpr) {
}
}
func (p *printer) word(w *Word) {
func (p *Printer) word(w *Word) {
for _, n := range w.Parts {
p.wordPart(n)
}
p.wantSpace = true
}
func (p *printer) unquotedWord(w *Word) {
func (p *Printer) unquotedWord(w *Word) {
for _, wp := range w.Parts {
switch x := wp.(type) {
case *SglQuoted:
@ -500,7 +491,7 @@ func (p *printer) unquotedWord(w *Word) {
}
}
func (p *printer) wordJoin(ws []*Word, backslash bool) {
func (p *Printer) wordJoin(ws []*Word, backslash bool) {
anyNewline := false
for _, w := range ws {
if pos := w.Pos(); pos > p.nline {
@ -527,7 +518,7 @@ func (p *printer) wordJoin(ws []*Word, backslash bool) {
}
}
func (p *printer) stmt(s *Stmt) {
func (p *Printer) stmt(s *Stmt) {
if s.Negated {
p.spacedString("!")
}
@ -575,7 +566,7 @@ func (p *printer) stmt(s *Stmt) {
}
}
func (p *printer) command(cmd Command, redirs []*Redirect) (startRedirs int) {
func (p *Printer) command(cmd Command, redirs []*Redirect) (startRedirs int) {
if p.wantSpace {
p.WriteByte(' ')
p.wantSpace = false
@ -631,7 +622,11 @@ func (p *printer) command(cmd Command, redirs []*Redirect) (startRedirs int) {
p.nestedStmts(x.Stmts, x.Rparen)
p.sepTok(")", x.Rparen)
case *WhileClause:
p.spacedString("while")
if x.Until {
p.spacedString("until")
} else {
p.spacedString("while")
}
p.nestedStmts(x.CondStmts, 0)
p.semiOrNewl("do", x.Do)
p.nestedStmts(x.DoStmts, 0)
@ -649,7 +644,7 @@ func (p *printer) command(cmd Command, redirs []*Redirect) (startRedirs int) {
p.incLevel()
}
_, p.nestedBinary = x.Y.Cmd.(*BinaryCmd)
if p.BinaryNextLine {
if p.binNextLine {
if len(p.pendingHdocs) == 0 && x.Y.Pos() > p.nline {
p.bslashNewl()
p.indent()
@ -665,8 +660,12 @@ func (p *printer) command(cmd Command, redirs []*Redirect) (startRedirs int) {
p.indent()
}
} else {
p.wantSpace = true
p.spacedString(x.Op.String())
if x.Y.Pos() > p.nline {
if x.OpPos > p.nline {
p.incLines(x.OpPos)
}
p.commentsUpTo(x.Y.Pos())
p.newline(0)
p.indent()
@ -720,15 +719,9 @@ func (p *printer) command(cmd Command, redirs []*Redirect) (startRedirs int) {
}
p.decLevel()
p.semiRsrv("esac", x.Esac, len(x.List) == 0)
case *UntilClause:
p.spacedString("until")
p.nestedStmts(x.CondStmts, 0)
p.semiOrNewl("do", x.Do)
p.nestedStmts(x.DoStmts, 0)
p.semiRsrv("done", x.Done, true)
case *ArithmCmd:
p.WriteString("((")
p.arithmExpr(x.X, false)
p.arithmExpr(x.X, false, false)
p.WriteString("))")
case *TestClause:
p.WriteString("[[ ")
@ -756,7 +749,7 @@ func (p *printer) command(cmd Command, redirs []*Redirect) (startRedirs int) {
p.spacedString("let")
for _, n := range x.Exprs {
p.WriteByte(' ')
p.arithmExpr(n, true)
p.arithmExpr(n, true, false)
}
}
return startRedirs
@ -772,7 +765,7 @@ func startsWithLparen(s *Stmt) bool {
return false
}
func (p *printer) hasInline(pos, npos, nline Pos) bool {
func (p *Printer) hasInline(pos, npos, nline Pos) bool {
for _, c := range p.comments {
if c.Hash > nline {
return false
@ -784,7 +777,7 @@ func (p *printer) hasInline(pos, npos, nline Pos) bool {
return false
}
func (p *printer) stmts(stmts []*Stmt) {
func (p *Printer) stmts(stmts []*Stmt) {
switch len(stmts) {
case 0:
return
@ -836,10 +829,10 @@ func (p *printer) stmts(stmts []*Stmt) {
if j+1 < len(follow) {
npos2 = follow[j+1].Pos()
}
if pos2 > nline2 || !p.hasInline(pos2, npos2, nline2) {
if !p.hasInline(pos2, npos2, nline2) {
break
}
if l := p.stmtLen(s2); l > inlineIndent {
if l := p.stmtCols(s2); l > inlineIndent {
inlineIndent = l
}
if ind2++; ind2 >= len(p.lines) {
@ -854,7 +847,9 @@ func (p *printer) stmts(stmts []*Stmt) {
}
}
if inlineIndent > 0 {
p.commentPadding = inlineIndent - p.stmtLen(s)
if l := p.stmtCols(s); l > 0 {
p.commentPadding = inlineIndent - l
}
}
}
p.wantNewline = true
@ -863,24 +858,41 @@ func (p *printer) stmts(stmts []*Stmt) {
type byteCounter int
func (c *byteCounter) WriteByte(b byte) error {
*c++
switch {
case *c < 0:
case b == '\n':
*c = -1
default:
*c++
}
return nil
}
func (c *byteCounter) WriteString(s string) (int, error) {
*c += byteCounter(len(s))
switch {
case *c < 0:
case strings.Contains(s, "\n"):
*c = -1
default:
*c += byteCounter(len(s))
}
return 0, nil
}
func (c *byteCounter) Reset(io.Writer) { *c = 0 }
func (p *printer) stmtLen(s *Stmt) int {
*p.lenPrinter = printer{bufWriter: &p.lenCounter}
// stmtCols reports the length that s will take when formatted in a
// single line. If it will span multiple lines, stmtCols will return -1.
func (p *Printer) stmtCols(s *Stmt) int {
*p.lenPrinter = Printer{
bufWriter: &p.lenCounter,
lines: p.lines,
}
p.lenPrinter.bufWriter.Reset(nil)
p.lenPrinter.incLines(s.Pos())
p.lenPrinter.stmt(s)
return int(p.lenCounter)
}
func (p *printer) nestedStmts(stmts []*Stmt, closing Pos) {
func (p *Printer) nestedStmts(stmts []*Stmt, closing Pos) {
p.incLevel()
if len(stmts) == 1 && closing > p.nline && stmts[0].End() <= p.nline {
p.newline(0)
@ -890,7 +902,7 @@ func (p *printer) nestedStmts(stmts []*Stmt, closing Pos) {
p.decLevel()
}
func (p *printer) assigns(assigns []*Assign) {
func (p *Printer) assigns(assigns []*Assign) {
anyNewline := false
for _, a := range assigns {
if a.Pos() > p.nline {
@ -905,6 +917,11 @@ func (p *printer) assigns(assigns []*Assign) {
}
if a.Name != nil {
p.WriteString(a.Name.Value)
if a.Index != nil {
p.WriteByte('[')
p.arithmExpr(a.Index, false, false)
p.WriteByte(']')
}
if a.Append {
p.WriteByte('+')
}
@ -912,6 +929,11 @@ func (p *printer) assigns(assigns []*Assign) {
}
if a.Value != nil {
p.word(a.Value)
} else if a.Array != nil {
p.wantSpace = false
p.WriteByte('(')
p.wordJoin(a.Array.List, false)
p.sepTok(")", a.Array.Rparen)
}
p.wantSpace = true
}

View File

@ -46,6 +46,9 @@ func Walk(node Node, f func(Node) bool) {
if x.Value != nil {
Walk(x.Value, f)
}
if x.Array != nil {
Walk(x.Array, f)
}
case *Redirect:
if x.N != nil {
Walk(x.N, f)
@ -71,9 +74,6 @@ func Walk(node Node, f func(Node) bool) {
case *WhileClause:
walkStmts(x.CondStmts, f)
walkStmts(x.DoStmts, f)
case *UntilClause:
walkStmts(x.CondStmts, f)
walkStmts(x.DoStmts, f)
case *ForClause:
Walk(x.Loop, f)
walkStmts(x.DoStmts, f)
@ -112,8 +112,8 @@ func Walk(node Node, f func(Node) bool) {
if x.Param != nil {
Walk(x.Param, f)
}
if x.Ind != nil {
Walk(x.Ind.Expr, f)
if x.Index != nil {
Walk(x.Index, f)
}
if x.Repl != nil {
Walk(x.Repl.Orig, f)

12
vendor/vendor.json vendored
View File

@ -51,16 +51,16 @@
"revisionTime": "2017-01-24T11:57:57Z"
},
{
"checksumSHA1": "ZjfvXVu+OyeRBysQ8uowAkPD/6o=",
"checksumSHA1": "ohm6oyTSFu/xZk5HtTUG7RIONO4=",
"path": "github.com/mvdan/sh/interp",
"revision": "faf782d3a498f50cfc6aa9071d04e6f1e82e8035",
"revisionTime": "2017-04-30T14:10:52Z"
"revision": "380eaf2df0412887a2240f5b15e76ac810ca2e71",
"revisionTime": "2017-05-17T16:44:15Z"
},
{
"checksumSHA1": "RIR7FOsCR78SmOJOUsclJe9lvxo=",
"checksumSHA1": "2OcfNJuStj/eAcEPW5yRdU02DCc=",
"path": "github.com/mvdan/sh/syntax",
"revision": "faf782d3a498f50cfc6aa9071d04e6f1e82e8035",
"revisionTime": "2017-04-30T14:10:52Z"
"revision": "380eaf2df0412887a2240f5b15e76ac810ca2e71",
"revisionTime": "2017-05-17T16:44:15Z"
},
{
"checksumSHA1": "HUXE+Nrcau8FSaVEvPYHMvDjxOE=",