1
0
mirror of https://github.com/go-task/task.git synced 2024-12-16 10:59:23 +02:00
task/vendor/mvdan.cc/sh/syntax/printer.go

986 lines
19 KiB
Go
Raw Normal View History

2017-04-24 14:47:10 +02:00
// Copyright (c) 2016, Daniel Martí <mvdan@mvdan.cc>
// See LICENSE for licensing information
package syntax
import (
"bufio"
"io"
2017-05-17 19:49:27 +02:00
"strings"
2017-07-31 00:11:34 +02:00
"unicode"
2017-04-24 14:47:10 +02:00
)
2017-07-31 00:11:34 +02:00
// Indent sets the number of spaces used for indentation. If set to 0,
// tabs will be used instead.
func Indent(spaces uint) func(*Printer) {
2017-05-17 19:49:27 +02:00
return func(p *Printer) { p.indentSpaces = spaces }
2017-04-24 14:47:10 +02:00
}
2017-07-31 00:11:34 +02:00
// BinaryNextLine will make binary operators appear on the next line
// when a binary command, such as a pipe, spans multiple lines. A
// backslash will be used.
2017-05-17 19:49:27 +02:00
func BinaryNextLine(p *Printer) { p.binNextLine = true }
// SwitchCaseIndent will make switch cases be indented. As such, switch
// case bodies will be two levels deeper than the switch itself.
func SwitchCaseIndent(p *Printer) { p.swtCaseIndent = true }
2017-07-31 00:11:34 +02:00
// NewPrinter allocates a new Printer and applies any number of options.
2017-05-17 19:49:27 +02:00
func NewPrinter(options ...func(*Printer)) *Printer {
p := &Printer{
bufWriter: bufio.NewWriter(nil),
lenPrinter: new(Printer),
}
for _, opt := range options {
opt(p)
}
return p
2017-04-24 14:47:10 +02:00
}
2017-07-31 00:11:34 +02:00
// Print "pretty-prints" the given AST file to the given writer. Writes
// to w are buffered.
2017-05-17 19:49:27 +02:00
func (p *Printer) Print(w io.Writer, f *File) error {
2017-04-24 14:47:10 +02:00
p.reset()
p.bufWriter.Reset(w)
2017-07-06 01:46:05 +02:00
p.stmts(f.StmtList)
p.newline(Pos{})
2017-05-27 16:17:49 +02:00
return p.bufWriter.Flush()
2017-04-24 14:47:10 +02:00
}
type bufWriter interface {
WriteByte(byte) error
WriteString(string) (int, error)
Reset(io.Writer)
2017-05-27 16:17:49 +02:00
Flush() error
2017-04-24 14:47:10 +02:00
}
2017-07-31 00:11:34 +02:00
// Printer holds the internal state of the printing mechanism of a
// program.
2017-05-17 19:49:27 +02:00
type Printer struct {
2017-04-24 14:47:10 +02:00
bufWriter
indentSpaces uint
binNextLine bool
swtCaseIndent bool
2017-05-17 19:49:27 +02:00
2017-04-24 14:47:10 +02:00
wantSpace bool
wantNewline bool
wroteSemi bool
2017-07-31 00:11:34 +02:00
commentPadding uint
2017-04-24 14:47:10 +02:00
2017-07-06 01:46:05 +02:00
// line is the current line number
line uint
2017-04-24 14:47:10 +02:00
// lastLevel is the last level of indentation that was used.
2017-07-31 00:11:34 +02:00
lastLevel uint
2017-04-24 14:47:10 +02:00
// level is the current level of indentation.
2017-07-31 00:11:34 +02:00
level uint
2017-04-24 14:47:10 +02:00
// levelIncs records which indentation level increments actually
// took place, to revert them once their section ends.
levelIncs []bool
nestedBinary bool
// pendingHdocs is the list of pending heredocs to write.
pendingHdocs []*Redirect
2017-05-17 19:49:27 +02:00
// used in stmtCols to align comments
lenPrinter *Printer
2017-04-24 14:47:10 +02:00
lenCounter byteCounter
}
2017-05-17 19:49:27 +02:00
func (p *Printer) reset() {
2017-04-24 14:47:10 +02:00
p.wantSpace, p.wantNewline = false, false
p.commentPadding = 0
2017-07-06 01:46:05 +02:00
p.line = 0
2017-04-24 14:47:10 +02:00
p.lastLevel, p.level = 0, 0
p.levelIncs = p.levelIncs[:0]
p.nestedBinary = false
p.pendingHdocs = p.pendingHdocs[:0]
}
2017-07-31 00:11:34 +02:00
func (p *Printer) spaces(n uint) {
for i := uint(0); i < n; i++ {
2017-04-24 14:47:10 +02:00
p.WriteByte(' ')
}
}
2017-05-17 19:49:27 +02:00
func (p *Printer) bslashNewl() {
2017-04-24 14:47:10 +02:00
if p.wantSpace {
p.WriteByte(' ')
}
p.WriteString("\\\n")
p.wantSpace = false
2017-07-06 01:46:05 +02:00
p.line++
p.indent()
2017-04-24 14:47:10 +02:00
}
2017-05-17 19:49:27 +02:00
func (p *Printer) spacedString(s string) {
2017-04-24 14:47:10 +02:00
if p.wantSpace {
p.WriteByte(' ')
}
p.WriteString(s)
p.wantSpace = true
}
2017-05-17 19:49:27 +02:00
func (p *Printer) semiOrNewl(s string, pos Pos) {
2017-04-24 14:47:10 +02:00
if p.wantNewline {
p.newline(pos)
p.indent()
} else {
if !p.wroteSemi {
p.WriteByte(';')
}
p.WriteByte(' ')
2017-07-06 01:46:05 +02:00
p.line = pos.Line()
2017-04-24 14:47:10 +02:00
}
p.WriteString(s)
p.wantSpace = true
}
2017-05-17 19:49:27 +02:00
func (p *Printer) incLevel() {
2017-04-24 14:47:10 +02:00
inc := false
if p.level <= p.lastLevel || len(p.levelIncs) == 0 {
p.level++
inc = true
} else if last := &p.levelIncs[len(p.levelIncs)-1]; *last {
*last = false
inc = true
}
p.levelIncs = append(p.levelIncs, inc)
}
2017-05-17 19:49:27 +02:00
func (p *Printer) decLevel() {
2017-04-24 14:47:10 +02:00
if p.levelIncs[len(p.levelIncs)-1] {
p.level--
}
p.levelIncs = p.levelIncs[:len(p.levelIncs)-1]
}
2017-05-17 19:49:27 +02:00
func (p *Printer) indent() {
2017-04-24 14:47:10 +02:00
p.lastLevel = p.level
switch {
case p.level == 0:
2017-05-17 19:49:27 +02:00
case p.indentSpaces == 0:
2017-07-31 00:11:34 +02:00
for i := uint(0); i < p.level; i++ {
2017-04-24 14:47:10 +02:00
p.WriteByte('\t')
}
2017-07-31 00:11:34 +02:00
default:
2017-05-17 19:49:27 +02:00
p.spaces(p.indentSpaces * p.level)
2017-04-24 14:47:10 +02:00
}
}
2017-05-17 19:49:27 +02:00
func (p *Printer) newline(pos Pos) {
2017-04-24 14:47:10 +02:00
p.wantNewline, p.wantSpace = false, false
p.WriteByte('\n')
2017-07-06 01:46:05 +02:00
if p.line < pos.Line() {
p.line++
2017-04-24 14:47:10 +02:00
}
hdocs := p.pendingHdocs
p.pendingHdocs = p.pendingHdocs[:0]
for _, r := range hdocs {
2017-05-27 16:17:49 +02:00
if r.Hdoc != nil {
p.word(r.Hdoc)
2017-07-06 01:46:05 +02:00
p.line = r.Hdoc.End().Line()
2017-05-27 16:17:49 +02:00
}
2017-04-24 14:47:10 +02:00
p.unquotedWord(r.Word)
2017-07-06 01:46:05 +02:00
p.line++
2017-04-24 14:47:10 +02:00
p.WriteByte('\n')
p.wantSpace = false
}
}
2017-05-17 19:49:27 +02:00
func (p *Printer) newlines(pos Pos) {
2017-04-24 14:47:10 +02:00
p.newline(pos)
2017-07-06 01:46:05 +02:00
if pos.Line() > p.line {
2017-04-24 14:47:10 +02:00
// preserve single empty lines
p.WriteByte('\n')
2017-07-06 01:46:05 +02:00
p.line++
2017-04-24 14:47:10 +02:00
}
p.indent()
}
func (p *Printer) rightParen(pos Pos) {
2017-07-06 01:46:05 +02:00
if p.wantNewline || pos.Line() > p.line {
2017-04-24 14:47:10 +02:00
p.newlines(pos)
}
p.WriteByte(')')
2017-04-24 14:47:10 +02:00
p.wantSpace = true
}
2017-05-17 19:49:27 +02:00
func (p *Printer) semiRsrv(s string, pos Pos, fallback bool) {
2017-07-06 01:46:05 +02:00
if p.wantNewline || pos.Line() > p.line {
2017-04-24 14:47:10 +02:00
p.newlines(pos)
2017-05-27 16:17:49 +02:00
} else {
if fallback && !p.wroteSemi {
2017-04-24 14:47:10 +02:00
p.WriteByte(';')
}
2017-05-27 16:17:49 +02:00
if p.wantSpace {
p.WriteByte(' ')
}
2017-04-24 14:47:10 +02:00
}
p.WriteString(s)
p.wantSpace = true
}
2017-07-06 01:46:05 +02:00
func (p *Printer) comment(c Comment) {
2017-04-24 14:47:10 +02:00
switch {
2017-07-06 01:46:05 +02:00
case p.line == 0:
case c.Hash.Line() > p.line:
2017-04-24 14:47:10 +02:00
p.newlines(c.Hash)
case p.wantSpace:
p.spaces(p.commentPadding + 1)
}
2017-07-06 01:46:05 +02:00
p.line = c.Hash.Line()
2017-04-24 14:47:10 +02:00
p.WriteByte('#')
2017-07-31 00:11:34 +02:00
p.WriteString(strings.TrimRightFunc(c.Text, unicode.IsSpace))
2017-07-06 01:46:05 +02:00
}
func (p *Printer) comments(cs []Comment) {
for _, c := range cs {
p.comment(c)
}
2017-04-24 14:47:10 +02:00
}
2017-05-17 19:49:27 +02:00
func (p *Printer) wordPart(wp WordPart) {
2017-04-24 14:47:10 +02:00
switch x := wp.(type) {
case *Lit:
p.WriteString(x.Value)
case *SglQuoted:
if x.Dollar {
p.WriteByte('$')
}
p.WriteByte('\'')
p.WriteString(x.Value)
p.WriteByte('\'')
2017-07-06 01:46:05 +02:00
p.line = x.End().Line()
2017-04-24 14:47:10 +02:00
case *DblQuoted:
2017-05-27 16:17:49 +02:00
p.dblQuoted(x)
2017-04-24 14:47:10 +02:00
case *CmdSubst:
2017-07-06 01:46:05 +02:00
p.line = x.Pos().Line()
2017-05-27 16:17:49 +02:00
switch {
2017-06-04 21:06:04 +02:00
case x.TempFile:
2017-05-27 16:17:49 +02:00
p.WriteString("${")
p.wantSpace = true
2017-07-06 01:46:05 +02:00
p.nestedStmts(x.StmtList, x.Right)
2017-05-27 16:17:49 +02:00
p.wantSpace = false
p.semiRsrv("}", x.Right, true)
2017-06-04 21:06:04 +02:00
case x.ReplyVar:
2017-05-27 16:17:49 +02:00
p.WriteString("${|")
2017-07-06 01:46:05 +02:00
p.nestedStmts(x.StmtList, x.Right)
2017-05-27 16:17:49 +02:00
p.wantSpace = false
p.semiRsrv("}", x.Right, true)
default:
p.WriteString("$(")
p.wantSpace = len(x.Stmts) > 0 && startsWithLparen(x.Stmts[0])
2017-07-06 01:46:05 +02:00
p.nestedStmts(x.StmtList, x.Right)
p.rightParen(x.Right)
2017-05-27 16:17:49 +02:00
}
2017-04-24 14:47:10 +02:00
case *ParamExp:
2017-05-01 00:50:22 +02:00
p.paramExp(x)
2017-04-24 14:47:10 +02:00
case *ArithmExp:
p.WriteString("$((")
2017-05-27 16:17:49 +02:00
if x.Unsigned {
p.WriteString("# ")
}
2017-05-17 19:49:27 +02:00
p.arithmExpr(x.X, false, false)
2017-04-24 14:47:10 +02:00
p.WriteString("))")
case *ExtGlob:
p.WriteString(x.Op.String())
p.WriteString(x.Pattern.Value)
p.WriteByte(')')
case *ProcSubst:
// avoid conflict with << and others
if p.wantSpace {
p.WriteByte(' ')
p.wantSpace = false
}
p.WriteString(x.Op.String())
2017-07-06 01:46:05 +02:00
p.nestedStmts(x.StmtList, Pos{})
2017-04-24 14:47:10 +02:00
p.WriteByte(')')
}
}
2017-05-27 16:17:49 +02:00
func (p *Printer) dblQuoted(dq *DblQuoted) {
if dq.Dollar {
p.WriteByte('$')
}
p.WriteByte('"')
for i, n := range dq.Parts {
p.wordPart(n)
if i == len(dq.Parts)-1 {
2017-07-06 01:46:05 +02:00
p.line = n.End().Line()
2017-05-27 16:17:49 +02:00
}
}
p.WriteByte('"')
}
func (p *Printer) wroteIndex(index ArithmExpr) bool {
if index == nil {
2017-05-27 16:17:49 +02:00
return false
}
p.WriteByte('[')
p.arithmExpr(index, false, false)
2017-05-27 16:17:49 +02:00
p.WriteByte(']')
return true
}
2017-05-17 19:49:27 +02:00
func (p *Printer) paramExp(pe *ParamExp) {
2017-05-27 16:17:49 +02:00
if pe.nakedIndex() { // arr[x]
2017-05-17 19:49:27 +02:00
p.WriteString(pe.Param.Value)
p.wroteIndex(pe.Index)
2017-05-17 19:49:27 +02:00
return
}
if pe.Short { // $var
2017-05-01 00:50:22 +02:00
p.WriteByte('$')
p.WriteString(pe.Param.Value)
return
}
2017-05-17 19:49:27 +02:00
// ${var...}
2017-05-01 00:50:22 +02:00
p.WriteString("${")
switch {
case pe.Length:
p.WriteByte('#')
2017-05-27 16:17:49 +02:00
case pe.Width:
p.WriteByte('%')
case pe.Excl:
2017-05-01 00:50:22 +02:00
p.WriteByte('!')
}
2017-06-04 21:06:04 +02:00
p.WriteString(pe.Param.Value)
p.wroteIndex(pe.Index)
2017-05-01 00:50:22 +02:00
if pe.Slice != nil {
p.WriteByte(':')
2017-05-17 19:49:27 +02:00
p.arithmExpr(pe.Slice.Offset, true, true)
2017-05-01 00:50:22 +02:00
if pe.Slice.Length != nil {
p.WriteByte(':')
2017-05-17 19:49:27 +02:00
p.arithmExpr(pe.Slice.Length, true, false)
2017-05-01 00:50:22 +02:00
}
} else if pe.Repl != nil {
if pe.Repl.All {
p.WriteByte('/')
}
p.WriteByte('/')
2017-05-27 16:17:49 +02:00
if pe.Repl.Orig != nil {
p.word(pe.Repl.Orig)
}
2017-05-01 00:50:22 +02:00
p.WriteByte('/')
2017-05-27 16:17:49 +02:00
if pe.Repl.With != nil {
p.word(pe.Repl.With)
}
2017-05-01 00:50:22 +02:00
} else if pe.Exp != nil {
p.WriteString(pe.Exp.Op.String())
2017-05-27 16:17:49 +02:00
if pe.Exp.Word != nil {
p.word(pe.Exp.Word)
}
2017-05-01 00:50:22 +02:00
}
p.WriteByte('}')
}
2017-05-17 19:49:27 +02:00
func (p *Printer) loop(loop Loop) {
2017-04-24 14:47:10 +02:00
switch x := loop.(type) {
case *WordIter:
p.WriteString(x.Name.Value)
2017-05-27 16:17:49 +02:00
if len(x.Items) > 0 {
2017-04-24 14:47:10 +02:00
p.spacedString(" in")
2017-05-27 16:17:49 +02:00
p.wordJoin(x.Items)
2017-04-24 14:47:10 +02:00
}
case *CStyleLoop:
p.WriteString("((")
if x.Init == nil {
p.WriteByte(' ')
}
2017-05-17 19:49:27 +02:00
p.arithmExpr(x.Init, false, false)
2017-04-24 14:47:10 +02:00
p.WriteString("; ")
2017-05-17 19:49:27 +02:00
p.arithmExpr(x.Cond, false, false)
2017-04-24 14:47:10 +02:00
p.WriteString("; ")
2017-05-17 19:49:27 +02:00
p.arithmExpr(x.Post, false, false)
2017-04-24 14:47:10 +02:00
p.WriteString("))")
}
}
2017-05-17 19:49:27 +02:00
func (p *Printer) arithmExpr(expr ArithmExpr, compact, spacePlusMinus bool) {
2017-04-24 14:47:10 +02:00
switch x := expr.(type) {
2017-05-27 16:17:49 +02:00
case *Word:
p.word(x)
2017-04-24 14:47:10 +02:00
case *BinaryArithm:
if compact {
2017-05-17 19:49:27 +02:00
p.arithmExpr(x.X, compact, spacePlusMinus)
2017-04-24 14:47:10 +02:00
p.WriteString(x.Op.String())
2017-05-17 19:49:27 +02:00
p.arithmExpr(x.Y, compact, false)
2017-04-24 14:47:10 +02:00
} else {
2017-05-17 19:49:27 +02:00
p.arithmExpr(x.X, compact, spacePlusMinus)
2017-04-24 14:47:10 +02:00
if x.Op != Comma {
p.WriteByte(' ')
}
p.WriteString(x.Op.String())
p.WriteByte(' ')
2017-05-17 19:49:27 +02:00
p.arithmExpr(x.Y, compact, false)
2017-04-24 14:47:10 +02:00
}
case *UnaryArithm:
if x.Post {
2017-05-17 19:49:27 +02:00
p.arithmExpr(x.X, compact, spacePlusMinus)
2017-04-24 14:47:10 +02:00
p.WriteString(x.Op.String())
} else {
2017-05-17 19:49:27 +02:00
if spacePlusMinus {
switch x.Op {
case Plus, Minus:
p.WriteByte(' ')
}
}
2017-04-24 14:47:10 +02:00
p.WriteString(x.Op.String())
2017-05-17 19:49:27 +02:00
p.arithmExpr(x.X, compact, false)
2017-04-24 14:47:10 +02:00
}
case *ParenArithm:
p.WriteByte('(')
2017-05-17 19:49:27 +02:00
p.arithmExpr(x.X, false, false)
2017-04-24 14:47:10 +02:00
p.WriteByte(')')
}
}
2017-05-17 19:49:27 +02:00
func (p *Printer) testExpr(expr TestExpr) {
2017-04-24 14:47:10 +02:00
switch x := expr.(type) {
case *Word:
p.word(x)
case *BinaryTest:
p.testExpr(x.X)
p.WriteByte(' ')
p.WriteString(x.Op.String())
p.WriteByte(' ')
p.testExpr(x.Y)
case *UnaryTest:
p.WriteString(x.Op.String())
p.WriteByte(' ')
p.testExpr(x.X)
case *ParenTest:
p.WriteByte('(')
p.testExpr(x.X)
p.WriteByte(')')
}
}
2017-05-17 19:49:27 +02:00
func (p *Printer) word(w *Word) {
2017-04-24 14:47:10 +02:00
for _, n := range w.Parts {
p.wordPart(n)
}
p.wantSpace = true
}
2017-05-17 19:49:27 +02:00
func (p *Printer) unquotedWord(w *Word) {
2017-04-24 14:47:10 +02:00
for _, wp := range w.Parts {
switch x := wp.(type) {
case *SglQuoted:
p.WriteString(x.Value)
case *DblQuoted:
for _, qp := range x.Parts {
p.wordPart(qp)
}
case *Lit:
for i := 0; i < len(x.Value); i++ {
if b := x.Value[i]; b == '\\' {
if i++; i < len(x.Value) {
p.WriteByte(x.Value[i])
}
} else {
p.WriteByte(b)
}
}
}
}
}
2017-05-27 16:17:49 +02:00
func (p *Printer) wordJoin(ws []*Word) {
2017-04-24 14:47:10 +02:00
anyNewline := false
for _, w := range ws {
2017-07-06 01:46:05 +02:00
if pos := w.Pos(); pos.Line() > p.line {
2017-04-24 14:47:10 +02:00
if !anyNewline {
p.incLevel()
anyNewline = true
}
2017-07-06 01:46:05 +02:00
p.bslashNewl()
2017-04-24 14:47:10 +02:00
} else if p.wantSpace {
p.WriteByte(' ')
p.wantSpace = false
}
p.word(w)
}
if anyNewline {
p.decLevel()
}
}
2017-07-06 01:46:05 +02:00
func (p *Printer) elemJoin(elems []*ArrayElem, last []Comment) {
p.incLevel()
2017-05-27 16:17:49 +02:00
for _, el := range elems {
2017-07-06 01:46:05 +02:00
var left *Comment
for _, c := range el.Comments {
if c.Pos().After(el.Pos()) {
left = &c
break
2017-05-27 16:17:49 +02:00
}
2017-07-06 01:46:05 +02:00
p.comment(c)
}
if el.Pos().Line() > p.line {
p.newline(el.Pos())
2017-05-27 16:17:49 +02:00
p.indent()
} else if p.wantSpace {
p.WriteByte(' ')
}
if p.wroteIndex(el.Index) {
2017-05-27 16:17:49 +02:00
p.WriteByte('=')
}
p.word(el.Value)
2017-07-06 01:46:05 +02:00
if left != nil {
p.comment(*left)
}
2017-05-27 16:17:49 +02:00
}
2017-07-06 01:46:05 +02:00
if len(last) > 0 {
p.comments(last)
2017-05-27 16:17:49 +02:00
}
2017-07-06 01:46:05 +02:00
p.decLevel()
2017-05-27 16:17:49 +02:00
}
2017-05-17 19:49:27 +02:00
func (p *Printer) stmt(s *Stmt) {
2017-04-24 14:47:10 +02:00
if s.Negated {
p.spacedString("!")
}
var startRedirs int
if s.Cmd != nil {
startRedirs = p.command(s.Cmd, s.Redirs)
}
2017-07-06 01:46:05 +02:00
p.incLevel()
2017-04-24 14:47:10 +02:00
for _, r := range s.Redirs[startRedirs:] {
2017-07-06 01:46:05 +02:00
if r.OpPos.Line() > p.line {
2017-04-24 14:47:10 +02:00
p.bslashNewl()
}
if p.wantSpace {
p.WriteByte(' ')
}
if r.N != nil {
p.WriteString(r.N.Value)
}
p.WriteString(r.Op.String())
2017-05-27 16:17:49 +02:00
p.wantSpace = true
2017-04-24 14:47:10 +02:00
p.word(r.Word)
if r.Op == Hdoc || r.Op == DashHdoc {
p.pendingHdocs = append(p.pendingHdocs, r)
}
}
p.wroteSemi = false
2017-05-27 16:17:49 +02:00
switch {
2017-07-06 01:46:05 +02:00
case s.Semicolon.IsValid() && s.Semicolon.Line() > p.line:
2017-04-24 14:47:10 +02:00
p.bslashNewl()
p.WriteByte(';')
p.wroteSemi = true
2017-05-27 16:17:49 +02:00
case s.Background:
2017-04-24 14:47:10 +02:00
p.WriteString(" &")
2017-05-27 16:17:49 +02:00
case s.Coprocess:
p.WriteString(" |&")
2017-04-24 14:47:10 +02:00
}
2017-07-06 01:46:05 +02:00
p.decLevel()
2017-04-24 14:47:10 +02:00
}
2017-05-17 19:49:27 +02:00
func (p *Printer) command(cmd Command, redirs []*Redirect) (startRedirs int) {
2017-04-24 14:47:10 +02:00
if p.wantSpace {
p.WriteByte(' ')
p.wantSpace = false
}
switch x := cmd.(type) {
case *CallExpr:
2017-07-31 00:11:34 +02:00
p.assigns(x.Assigns, true)
2017-04-24 14:47:10 +02:00
if len(x.Args) <= 1 {
2017-05-27 16:17:49 +02:00
p.wordJoin(x.Args)
2017-04-24 14:47:10 +02:00
return 0
}
2017-05-27 16:17:49 +02:00
p.wordJoin(x.Args[:1])
2017-04-24 14:47:10 +02:00
for _, r := range redirs {
2017-07-06 01:46:05 +02:00
if r.Pos().After(x.Args[1].Pos()) || r.Op == Hdoc || r.Op == DashHdoc {
2017-04-24 14:47:10 +02:00
break
}
if p.wantSpace {
p.WriteByte(' ')
}
if r.N != nil {
p.WriteString(r.N.Value)
}
p.WriteString(r.Op.String())
2017-05-27 16:17:49 +02:00
p.wantSpace = true
2017-04-24 14:47:10 +02:00
p.word(r.Word)
startRedirs++
}
2017-05-27 16:17:49 +02:00
p.wordJoin(x.Args[1:])
2017-04-24 14:47:10 +02:00
case *Block:
p.WriteByte('{')
p.wantSpace = true
2017-07-06 01:46:05 +02:00
p.nestedStmts(x.StmtList, x.Rbrace)
2017-04-24 14:47:10 +02:00
p.semiRsrv("}", x.Rbrace, true)
case *IfClause:
2017-07-06 01:46:05 +02:00
p.ifClause(x, false)
2017-04-24 14:47:10 +02:00
case *Subshell:
p.WriteByte('(')
p.wantSpace = len(x.Stmts) > 0 && startsWithLparen(x.Stmts[0])
2017-07-06 01:46:05 +02:00
p.nestedStmts(x.StmtList, x.Rparen)
p.rightParen(x.Rparen)
2017-04-24 14:47:10 +02:00
case *WhileClause:
2017-05-17 19:49:27 +02:00
if x.Until {
p.spacedString("until")
} else {
p.spacedString("while")
}
2017-07-06 01:46:05 +02:00
p.nestedStmts(x.Cond, Pos{})
p.semiOrNewl("do", x.DoPos)
p.nestedStmts(x.Do, Pos{})
p.semiRsrv("done", x.DonePos, true)
2017-04-24 14:47:10 +02:00
case *ForClause:
if x.Select {
p.WriteString("select ")
} else {
p.WriteString("for ")
}
2017-04-24 14:47:10 +02:00
p.loop(x.Loop)
2017-07-06 01:46:05 +02:00
p.semiOrNewl("do", x.DoPos)
p.nestedStmts(x.Do, Pos{})
p.semiRsrv("done", x.DonePos, true)
2017-04-24 14:47:10 +02:00
case *BinaryCmd:
p.stmt(x.X)
2017-07-06 01:46:05 +02:00
if x.Y.Pos().Line() <= p.line {
2017-05-27 16:17:49 +02:00
// leave p.nestedBinary untouched
p.spacedString(x.Op.String())
p.stmt(x.Y)
break
}
2017-04-24 14:47:10 +02:00
indent := !p.nestedBinary
if indent {
p.incLevel()
}
2017-05-17 19:49:27 +02:00
if p.binNextLine {
2017-05-27 16:17:49 +02:00
if len(p.pendingHdocs) == 0 {
2017-05-01 00:50:22 +02:00
p.bslashNewl()
}
p.spacedString(x.Op.String())
2017-07-06 01:46:05 +02:00
if len(x.Y.Comments) > 0 {
2017-05-01 00:50:22 +02:00
p.wantSpace = false
p.WriteByte('\n')
p.indent()
2017-07-06 01:46:05 +02:00
p.comments(x.Y.Comments)
2017-05-01 00:50:22 +02:00
p.WriteByte('\n')
p.indent()
}
} else {
2017-05-17 19:49:27 +02:00
p.wantSpace = true
2017-05-01 00:50:22 +02:00
p.spacedString(x.Op.String())
2017-07-06 01:46:05 +02:00
p.line = x.OpPos.Line()
p.comments(x.Y.Comments)
p.newline(Pos{})
2017-05-27 16:17:49 +02:00
p.indent()
2017-04-24 14:47:10 +02:00
}
2017-07-06 01:46:05 +02:00
p.line = x.Y.Pos().Line()
2017-05-27 16:17:49 +02:00
_, p.nestedBinary = x.Y.Cmd.(*BinaryCmd)
2017-04-24 14:47:10 +02:00
p.stmt(x.Y)
if indent {
p.decLevel()
}
p.nestedBinary = false
case *FuncDecl:
2017-06-04 21:06:04 +02:00
if x.RsrvWord {
2017-04-24 14:47:10 +02:00
p.WriteString("function ")
}
p.WriteString(x.Name.Value)
p.WriteString("() ")
2017-07-06 01:46:05 +02:00
p.line = x.Body.Pos().Line()
2017-04-24 14:47:10 +02:00
p.stmt(x.Body)
case *CaseClause:
p.WriteString("case ")
p.word(x.Word)
p.WriteString(" in")
if p.swtCaseIndent {
p.incLevel()
}
2017-05-27 16:17:49 +02:00
for _, ci := range x.Items {
2017-07-06 01:46:05 +02:00
var inlineCom *Comment
for _, c := range ci.Comments {
if c.Pos().After(ci.Patterns[0].Pos()) {
inlineCom = &c
break
}
p.comment(c)
}
if pos := ci.Patterns[0].Pos(); pos.Line() > p.line {
p.newlines(pos)
}
2017-05-27 16:17:49 +02:00
for i, w := range ci.Patterns {
2017-04-24 14:47:10 +02:00
if i > 0 {
p.spacedString("|")
}
if p.wantSpace {
p.WriteByte(' ')
}
p.word(w)
}
p.WriteByte(')')
p.wantSpace = true
2017-07-06 01:46:05 +02:00
sep := len(ci.Stmts) > 1 || ci.StmtList.pos().Line() > p.line
2017-07-31 00:11:34 +02:00
if ci.OpPos != x.Esac && !ci.StmtList.empty() &&
ci.OpPos.Line() > ci.StmtList.end().Line() {
sep = true
}
2017-07-06 01:46:05 +02:00
sl := ci.StmtList
p.nestedStmts(sl, Pos{})
2017-04-24 14:47:10 +02:00
p.level++
if sep {
2017-05-27 16:17:49 +02:00
p.newlines(ci.OpPos)
2017-07-06 01:46:05 +02:00
p.wantNewline = true
2017-04-24 14:47:10 +02:00
}
2017-05-27 16:17:49 +02:00
p.spacedString(ci.Op.String())
2017-07-06 01:46:05 +02:00
if inlineCom != nil {
p.comment(*inlineCom)
2017-04-24 14:47:10 +02:00
}
2017-07-06 01:46:05 +02:00
p.level--
2017-04-24 14:47:10 +02:00
}
2017-07-06 01:46:05 +02:00
p.comments(x.Last)
if p.swtCaseIndent {
p.decLevel()
}
2017-05-27 16:17:49 +02:00
p.semiRsrv("esac", x.Esac, len(x.Items) == 0)
2017-04-24 14:47:10 +02:00
case *ArithmCmd:
p.WriteString("((")
2017-05-27 16:17:49 +02:00
if x.Unsigned {
p.WriteString("# ")
}
2017-05-17 19:49:27 +02:00
p.arithmExpr(x.X, false, false)
2017-04-24 14:47:10 +02:00
p.WriteString("))")
case *TestClause:
p.WriteString("[[ ")
p.testExpr(x.X)
p.spacedString("]]")
case *DeclClause:
2017-07-06 01:46:05 +02:00
p.spacedString(x.Variant.Value)
2017-04-24 14:47:10 +02:00
for _, w := range x.Opts {
p.WriteByte(' ')
p.word(w)
}
2017-05-27 16:17:49 +02:00
p.assigns(x.Assigns, false)
case *TimeClause:
p.spacedString("time")
if x.Stmt != nil {
p.stmt(x.Stmt)
}
2017-04-24 14:47:10 +02:00
case *CoprocClause:
p.spacedString("coproc")
if x.Name != nil {
p.WriteByte(' ')
p.WriteString(x.Name.Value)
}
p.stmt(x.Stmt)
case *LetClause:
p.spacedString("let")
for _, n := range x.Exprs {
p.WriteByte(' ')
2017-05-17 19:49:27 +02:00
p.arithmExpr(n, true, false)
2017-04-24 14:47:10 +02:00
}
}
return startRedirs
}
2017-07-06 01:46:05 +02:00
func (p *Printer) ifClause(ic *IfClause, elif bool) {
if !elif {
p.spacedString("if")
}
p.nestedStmts(ic.Cond, Pos{})
p.semiOrNewl("then", ic.ThenPos)
p.nestedStmts(ic.Then, Pos{})
if ic.FollowedByElif() {
p.semiRsrv("elif", ic.ElsePos, true)
p.ifClause(ic.Else.Stmts[0].Cmd.(*IfClause), true)
return
}
if !ic.Else.empty() {
p.semiRsrv("else", ic.ElsePos, true)
p.nestedStmts(ic.Else, Pos{})
} else if ic.ElsePos.IsValid() {
p.line = ic.ElsePos.Line()
}
p.semiRsrv("fi", ic.FiPos, true)
}
2017-04-24 14:47:10 +02:00
func startsWithLparen(s *Stmt) bool {
switch x := s.Cmd.(type) {
case *Subshell:
return true
case *BinaryCmd:
return startsWithLparen(x.X)
}
return false
}
2017-07-06 01:46:05 +02:00
func (p *Printer) hasInline(s *Stmt) bool {
for _, c := range s.Comments {
if c.Pos().Line() == s.End().Line() {
2017-04-24 14:47:10 +02:00
return true
}
}
return false
}
2017-07-06 01:46:05 +02:00
func (p *Printer) stmts(sl StmtList) {
switch len(sl.Stmts) {
2017-04-24 14:47:10 +02:00
case 0:
2017-07-06 01:46:05 +02:00
p.comments(sl.Last)
2017-04-24 14:47:10 +02:00
return
case 1:
2017-07-06 01:46:05 +02:00
s := sl.Stmts[0]
2017-04-24 14:47:10 +02:00
pos := s.Pos()
2017-07-06 01:46:05 +02:00
var inlineCom *Comment
for _, c := range s.Comments {
if c.Pos().After(s.Pos()) {
inlineCom = &c
break
}
p.comment(c)
}
if pos.Line() <= p.line {
2017-04-24 14:47:10 +02:00
p.stmt(s)
} else {
2017-07-06 01:46:05 +02:00
if p.line > 0 {
2017-04-24 14:47:10 +02:00
p.newlines(pos)
}
2017-07-06 01:46:05 +02:00
p.line = pos.Line()
2017-04-24 14:47:10 +02:00
p.stmt(s)
p.wantNewline = true
}
2017-07-06 01:46:05 +02:00
if inlineCom != nil {
p.comment(*inlineCom)
}
p.comments(sl.Last)
2017-04-24 14:47:10 +02:00
return
}
inlineIndent := 0
2017-07-06 01:46:05 +02:00
lastIndentedLine := uint(0)
for i, s := range sl.Stmts {
2017-04-24 14:47:10 +02:00
pos := s.Pos()
2017-07-06 01:46:05 +02:00
var inlineCom *Comment
for _, c := range s.Comments {
if c.Pos().After(s.Pos()) {
inlineCom = &c
break
}
p.comment(c)
}
if p.line > 0 {
2017-04-24 14:47:10 +02:00
p.newlines(pos)
}
2017-07-06 01:46:05 +02:00
p.line = pos.Line()
2017-04-24 14:47:10 +02:00
p.stmt(s)
2017-07-06 01:46:05 +02:00
if !p.hasInline(s) {
2017-04-24 14:47:10 +02:00
inlineIndent = 0
p.commentPadding = 0
continue
}
2017-07-06 01:46:05 +02:00
if s.Pos().Line() > lastIndentedLine+1 {
2017-04-24 14:47:10 +02:00
inlineIndent = 0
}
if inlineIndent == 0 {
2017-07-06 01:46:05 +02:00
for _, s2 := range sl.Stmts[i:] {
if !p.hasInline(s2) {
2017-04-24 14:47:10 +02:00
break
}
2017-05-17 19:49:27 +02:00
if l := p.stmtCols(s2); l > inlineIndent {
2017-04-24 14:47:10 +02:00
inlineIndent = l
}
}
}
if inlineIndent > 0 {
2017-05-17 19:49:27 +02:00
if l := p.stmtCols(s); l > 0 {
2017-07-31 00:11:34 +02:00
p.commentPadding = uint(inlineIndent - l)
2017-05-17 19:49:27 +02:00
}
2017-07-06 01:46:05 +02:00
lastIndentedLine = p.line
}
if inlineCom != nil {
p.comment(*inlineCom)
2017-04-24 14:47:10 +02:00
}
}
p.wantNewline = true
2017-07-06 01:46:05 +02:00
p.comments(sl.Last)
2017-04-24 14:47:10 +02:00
}
type byteCounter int
func (c *byteCounter) WriteByte(b byte) error {
2017-05-17 19:49:27 +02:00
switch {
case *c < 0:
case b == '\n':
*c = -1
default:
*c++
}
2017-04-24 14:47:10 +02:00
return nil
}
func (c *byteCounter) WriteString(s string) (int, error) {
2017-05-17 19:49:27 +02:00
switch {
case *c < 0:
case strings.Contains(s, "\n"):
*c = -1
default:
*c += byteCounter(len(s))
}
2017-04-24 14:47:10 +02:00
return 0, nil
}
func (c *byteCounter) Reset(io.Writer) { *c = 0 }
2017-05-27 16:17:49 +02:00
func (c *byteCounter) Flush() error { return nil }
2017-04-24 14:47:10 +02:00
2017-05-17 19:49:27 +02:00
// 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 {
2017-07-06 01:46:05 +02:00
if p.lenPrinter == nil {
return -1 // stmtCols call within stmtCols, bail
}
2017-05-17 19:49:27 +02:00
*p.lenPrinter = Printer{
bufWriter: &p.lenCounter,
}
2017-04-24 14:47:10 +02:00
p.lenPrinter.bufWriter.Reset(nil)
2017-07-06 01:46:05 +02:00
p.lenPrinter.line = s.Pos().Line()
2017-04-24 14:47:10 +02:00
p.lenPrinter.stmt(s)
return int(p.lenCounter)
}
2017-07-06 01:46:05 +02:00
func (p *Printer) nestedStmts(sl StmtList, closing Pos) {
2017-04-24 14:47:10 +02:00
p.incLevel()
2017-07-06 01:46:05 +02:00
if len(sl.Stmts) == 1 && closing.Line() > p.line && sl.Stmts[0].End().Line() <= p.line {
p.newline(Pos{})
2017-04-24 14:47:10 +02:00
p.indent()
}
2017-07-06 01:46:05 +02:00
p.stmts(sl)
2017-04-24 14:47:10 +02:00
p.decLevel()
}
2017-05-27 16:17:49 +02:00
func (p *Printer) assigns(assigns []*Assign, alwaysEqual bool) {
2017-07-06 01:46:05 +02:00
p.incLevel()
2017-04-24 14:47:10 +02:00
for _, a := range assigns {
2017-07-06 01:46:05 +02:00
if a.Pos().Line() > p.line {
2017-04-24 14:47:10 +02:00
p.bslashNewl()
} else if p.wantSpace {
p.WriteByte(' ')
}
if a.Name != nil {
p.WriteString(a.Name.Value)
p.wroteIndex(a.Index)
2017-04-24 14:47:10 +02:00
if a.Append {
p.WriteByte('+')
}
2017-05-27 16:17:49 +02:00
if alwaysEqual || a.Value != nil || a.Array != nil {
p.WriteByte('=')
}
2017-04-24 14:47:10 +02:00
}
if a.Value != nil {
p.word(a.Value)
2017-05-17 19:49:27 +02:00
} else if a.Array != nil {
p.wantSpace = false
p.WriteByte('(')
2017-07-06 01:46:05 +02:00
p.elemJoin(a.Array.Elems, a.Array.Last)
p.rightParen(a.Array.Rparen)
2017-04-24 14:47:10 +02:00
}
p.wantSpace = true
}
2017-07-06 01:46:05 +02:00
p.decLevel()
2017-04-24 14:47:10 +02:00
}