argh/parser.go

347 lines
7.4 KiB
Go
Raw Normal View History

package argh
import (
2022-05-19 00:15:31 +00:00
"fmt"
"io"
"strings"
)
2022-05-29 23:10:16 +00:00
type parser struct {
2022-05-15 18:22:56 +00:00
s *Scanner
2022-05-14 00:58:55 +00:00
cfg *ParserConfig
2022-05-14 00:58:55 +00:00
errors ParserErrorList
2022-05-29 23:10:16 +00:00
tok Token
lit string
pos Pos
buffered bool
}
2022-05-19 00:15:31 +00:00
type ParseTree struct {
Nodes []Node `json:"nodes"`
}
2022-05-29 23:16:13 +00:00
func ParseArgs(args []string, pCfg *ParserConfig) (*ParseTree, error) {
2022-05-29 23:10:16 +00:00
p := &parser{}
p.init(
strings.NewReader(strings.Join(args, string(nul))),
pCfg,
)
2022-05-29 23:16:13 +00:00
tracef("ParseArgs(...) parser=%+#v", p)
2022-05-29 23:10:16 +00:00
return p.parseArgs()
2022-05-19 00:15:31 +00:00
}
func (p *parser) addError(msg string) {
p.errors.Add(Position{Column: int(p.pos)}, msg)
}
2022-05-29 23:10:16 +00:00
func (p *parser) init(r io.Reader, pCfg *ParserConfig) {
p.errors = ParserErrorList{}
2022-05-29 23:10:16 +00:00
2022-05-14 00:58:55 +00:00
if pCfg == nil {
pCfg = POSIXyParserConfig
}
2022-05-29 23:10:16 +00:00
p.cfg = pCfg
2022-05-14 00:58:55 +00:00
2022-05-29 23:10:16 +00:00
p.s = NewScanner(r, pCfg.ScannerConfig)
2022-05-14 00:58:55 +00:00
2022-05-29 23:10:16 +00:00
p.next()
2022-05-14 00:58:55 +00:00
}
2022-05-29 23:10:16 +00:00
func (p *parser) parseArgs() (*ParseTree, error) {
if p.errors.Len() != 0 {
tracef("parseArgs() bailing due to initial error")
return nil, p.errors.Err()
2022-05-14 00:58:55 +00:00
}
2022-05-29 23:10:16 +00:00
tracef("parseArgs() parsing %q as program command; cfg=%+#v", p.lit, p.cfg.Prog)
prog := p.parseCommand(&p.cfg.Prog)
2022-05-14 00:58:55 +00:00
2022-05-29 23:10:16 +00:00
tracef("parseArgs() top level node is %T", prog)
2022-05-14 00:58:55 +00:00
2022-05-29 23:10:16 +00:00
nodes := []Node{prog}
if v := p.parsePassthrough(); v != nil {
tracef("parseArgs() appending passthrough argument %v", v)
nodes = append(nodes, v)
2022-05-14 00:58:55 +00:00
}
2022-05-29 23:10:16 +00:00
tracef("parseArgs() returning ParseTree")
2022-05-14 00:58:55 +00:00
2022-05-29 23:10:16 +00:00
return &ParseTree{Nodes: nodes}, p.errors.Err()
}
2022-05-14 00:58:55 +00:00
2022-05-29 23:10:16 +00:00
func (p *parser) next() {
tracef("next() before scan: %v %q %v", p.tok, p.lit, p.pos)
2022-05-14 00:58:55 +00:00
2022-05-29 23:10:16 +00:00
p.tok, p.lit, p.pos = p.s.Scan()
2022-05-14 00:58:55 +00:00
2022-05-29 23:10:16 +00:00
tracef("next() after scan: %v %q %v", p.tok, p.lit, p.pos)
2022-05-14 00:58:55 +00:00
}
2022-05-29 23:10:16 +00:00
func (p *parser) parseCommand(cCfg *CommandConfig) Node {
tracef("parseCommand(%+#v)", cCfg)
2022-05-14 00:58:55 +00:00
2022-05-29 23:10:16 +00:00
node := &Command{
Name: p.lit,
2022-05-19 00:15:31 +00:00
}
2022-05-29 23:10:16 +00:00
values := map[string]string{}
nodes := []Node{}
2022-05-29 23:10:16 +00:00
identIndex := 0
2022-05-29 23:10:16 +00:00
for i := 0; p.tok != EOL; i++ {
if !p.buffered {
tracef("parseCommand(...) buffered=false; scanning next")
p.next()
}
2022-05-29 23:10:16 +00:00
p.buffered = false
2022-05-14 00:58:55 +00:00
2022-05-29 23:10:16 +00:00
tracef("parseCommand(...) for=%d values=%+#v", i, values)
tracef("parseCommand(...) for=%d nodes=%+#v", i, nodes)
tracef("parseCommand(...) for=%d tok=%s lit=%q pos=%v", i, p.tok, p.lit, p.pos)
2022-05-31 12:24:58 +00:00
tracef("parseCommand(...) cCfg=%+#v", cCfg)
if subCfg, ok := cCfg.GetCommandConfig(p.lit); ok {
2022-05-29 23:10:16 +00:00
subCommand := p.lit
2022-05-29 23:10:16 +00:00
nodes = append(nodes, p.parseCommand(&subCfg))
2022-05-29 23:10:16 +00:00
tracef("parseCommand(...) breaking after sub-command=%v", subCommand)
break
}
2022-05-29 23:10:16 +00:00
switch p.tok {
case ARG_DELIMITER:
tracef("parseCommand(...) handling %s", p.tok)
2022-05-29 23:10:16 +00:00
nodes = append(nodes, &ArgDelimiter{})
2022-05-19 00:15:31 +00:00
2022-05-29 23:10:16 +00:00
continue
case IDENT, STDIN_FLAG:
tracef("parseCommand(...) handling %s", p.tok)
2022-05-14 00:58:55 +00:00
2022-05-29 23:10:16 +00:00
if cCfg.NValue.Contains(identIndex) {
name := fmt.Sprintf("%d", identIndex)
2022-05-29 23:10:16 +00:00
tracef("parseCommand(...) checking for name of identIndex=%d", identIndex)
2022-05-14 00:58:55 +00:00
2022-05-29 23:10:16 +00:00
if len(cCfg.ValueNames) > identIndex {
name = cCfg.ValueNames[identIndex]
tracef("parseCommand(...) setting name=%s from config value names", name)
} else if len(cCfg.ValueNames) == 1 && (cCfg.NValue == OneOrMoreValue || cCfg.NValue == ZeroOrMoreValue) {
name = fmt.Sprintf("%s.%d", cCfg.ValueNames[0], identIndex)
tracef("parseCommand(...) setting name=%s from repeating value name", name)
}
2022-05-19 00:15:31 +00:00
2022-05-29 23:10:16 +00:00
values[name] = p.lit
}
2022-05-19 00:15:31 +00:00
2022-05-29 23:10:16 +00:00
if p.tok == STDIN_FLAG {
nodes = append(nodes, &StdinFlag{})
} else {
nodes = append(nodes, &Ident{Literal: p.lit})
}
2022-05-29 23:10:16 +00:00
identIndex++
case LONG_FLAG, SHORT_FLAG, COMPOUND_SHORT_FLAG:
tok := p.tok
2022-05-19 00:15:31 +00:00
2022-05-29 23:10:16 +00:00
flagNode := p.parseFlag(cCfg.Flags)
2022-05-19 00:15:31 +00:00
2022-05-29 23:10:16 +00:00
tracef("parseCommand(...) appending %s node=%+#v", tok, flagNode)
2022-05-19 00:15:31 +00:00
2022-05-29 23:10:16 +00:00
nodes = append(nodes, flagNode)
case ASSIGN:
tracef("parseCommand(...) error on bare %s", p.tok)
2022-05-19 00:15:31 +00:00
p.addError("invalid bare assignment")
2022-05-29 23:10:16 +00:00
break
default:
tracef("parseCommand(...) breaking on %s", p.tok)
break
2022-05-14 00:58:55 +00:00
}
2022-05-29 23:10:16 +00:00
}
2022-05-29 23:10:16 +00:00
if len(nodes) > 0 {
node.Nodes = nodes
}
2022-05-29 23:10:16 +00:00
if len(values) > 0 {
node.Values = values
}
tracef("parseCommand(...) returning node=%+#v", node)
return node
2022-05-19 00:15:31 +00:00
}
2022-05-29 23:10:16 +00:00
func (p *parser) parseIdent() Node {
node := &Ident{Literal: p.lit}
return node
}
2022-05-31 12:24:58 +00:00
func (p *parser) parseFlag(flags *Flags) Node {
2022-05-29 23:10:16 +00:00
switch p.tok {
case SHORT_FLAG:
2022-05-31 12:24:58 +00:00
tracef("parseFlag(...) parsing short flag with config=%+#v", flags)
return p.parseShortFlag(flags)
2022-05-29 23:10:16 +00:00
case LONG_FLAG:
2022-05-31 12:24:58 +00:00
tracef("parseFlag(...) parsing long flag with config=%+#v", flags)
return p.parseLongFlag(flags)
2022-05-29 23:10:16 +00:00
case COMPOUND_SHORT_FLAG:
2022-05-31 12:24:58 +00:00
tracef("parseFlag(...) parsing compound short flag with config=%+#v", flags)
return p.parseCompoundShortFlag(flags)
2022-05-29 23:10:16 +00:00
}
2022-05-19 00:15:31 +00:00
2022-05-29 23:10:16 +00:00
panic(fmt.Sprintf("token %v cannot be parsed as flag", p.tok))
}
2022-05-15 18:22:56 +00:00
2022-05-31 12:24:58 +00:00
func (p *parser) parseShortFlag(flags *Flags) Node {
2022-05-29 23:10:16 +00:00
node := &Flag{Name: string(p.lit[1])}
2022-05-15 18:22:56 +00:00
2022-05-31 12:24:58 +00:00
flCfg, ok := flags.Get(node.Name)
2022-05-29 23:10:16 +00:00
if !ok {
p.addError(fmt.Sprintf("unknown flag %q", node.Name))
2022-05-29 23:10:16 +00:00
return node
}
2022-05-15 18:22:56 +00:00
2022-05-29 23:10:16 +00:00
return p.parseConfiguredFlag(node, flCfg)
}
2022-05-15 18:22:56 +00:00
2022-05-31 12:24:58 +00:00
func (p *parser) parseLongFlag(flags *Flags) Node {
2022-05-29 23:10:16 +00:00
node := &Flag{Name: string(p.lit[2:])}
2022-05-15 18:22:56 +00:00
2022-05-31 12:24:58 +00:00
flCfg, ok := flags.Get(node.Name)
2022-05-29 23:10:16 +00:00
if !ok {
p.addError(fmt.Sprintf("unknown flag %q", node.Name))
2022-05-29 23:10:16 +00:00
return node
}
2022-05-19 00:15:31 +00:00
2022-05-29 23:10:16 +00:00
return p.parseConfiguredFlag(node, flCfg)
}
2022-05-15 18:22:56 +00:00
2022-05-31 12:24:58 +00:00
func (p *parser) parseCompoundShortFlag(flags *Flags) Node {
2022-05-29 23:10:16 +00:00
flagNodes := []Node{}
2022-05-19 00:15:31 +00:00
2022-05-29 23:10:16 +00:00
withoutFlagPrefix := p.lit[1:]
2022-05-15 18:22:56 +00:00
2022-05-29 23:10:16 +00:00
for i, r := range withoutFlagPrefix {
node := &Flag{Name: string(r)}
2022-05-14 00:58:55 +00:00
2022-05-29 23:10:16 +00:00
if i == len(withoutFlagPrefix)-1 {
2022-05-31 12:24:58 +00:00
flCfg, ok := flags.Get(node.Name)
if !ok {
p.addError(fmt.Sprintf("unknown flag %q", node.Name))
2022-05-14 00:58:55 +00:00
2022-05-29 23:10:16 +00:00
continue
}
flagNodes = append(flagNodes, p.parseConfiguredFlag(node, flCfg))
continue
2022-05-29 23:10:16 +00:00
}
flagNodes = append(flagNodes, node)
}
2022-05-15 18:22:56 +00:00
2022-05-29 23:10:16 +00:00
return &CompoundShortFlag{Nodes: flagNodes}
2022-05-14 00:58:55 +00:00
}
2022-05-29 23:10:16 +00:00
func (p *parser) parseConfiguredFlag(node *Flag, flCfg FlagConfig) Node {
values := map[string]string{}
nodes := []Node{}
2022-05-14 00:58:55 +00:00
2022-05-29 23:10:16 +00:00
identIndex := 0
2022-05-29 23:10:16 +00:00
for i := 0; p.tok != EOL; i++ {
if !flCfg.NValue.Contains(identIndex) {
tracef("parseConfiguredFlag(...) identIndex=%d exceeds expected=%v; breaking", identIndex, flCfg.NValue)
break
}
2022-05-14 00:58:55 +00:00
2022-05-29 23:10:16 +00:00
p.next()
2022-05-29 23:10:16 +00:00
switch p.tok {
case ARG_DELIMITER:
nodes = append(nodes, &ArgDelimiter{})
2022-05-29 23:10:16 +00:00
continue
case ASSIGN:
nodes = append(nodes, &Assign{})
2022-05-15 18:22:56 +00:00
2022-05-29 23:10:16 +00:00
continue
case IDENT, STDIN_FLAG:
name := fmt.Sprintf("%d", identIndex)
2022-05-14 00:58:55 +00:00
2022-05-29 23:10:16 +00:00
tracef("parseConfiguredFlag(...) checking for name of identIndex=%d", identIndex)
2022-05-14 00:58:55 +00:00
2022-05-29 23:10:16 +00:00
if len(flCfg.ValueNames) > identIndex {
name = flCfg.ValueNames[identIndex]
tracef("parseConfiguredFlag(...) setting name=%s from config value names", name)
} else if len(flCfg.ValueNames) == 1 && (flCfg.NValue == OneOrMoreValue || flCfg.NValue == ZeroOrMoreValue) {
name = fmt.Sprintf("%s.%d", flCfg.ValueNames[0], identIndex)
tracef("parseConfiguredFlag(...) setting name=%s from repeating value name", name)
} else {
tracef("parseConfiguredFlag(...) setting name=%s", name)
}
2022-05-29 23:10:16 +00:00
values[name] = p.lit
2022-05-29 23:10:16 +00:00
if p.tok == STDIN_FLAG {
nodes = append(nodes, &StdinFlag{})
} else {
nodes = append(nodes, &Ident{Literal: p.lit})
}
2022-05-15 18:22:56 +00:00
2022-05-29 23:10:16 +00:00
identIndex++
default:
tracef("parseConfiguredFlag(...) breaking on %s %q %v; setting buffered=true", p.tok, p.lit, p.pos)
p.buffered = true
2022-05-14 00:58:55 +00:00
2022-05-29 23:10:16 +00:00
if len(nodes) > 0 {
node.Nodes = nodes
}
2022-05-14 00:58:55 +00:00
2022-05-29 23:10:16 +00:00
if len(values) > 0 {
node.Values = values
}
2022-05-15 18:22:56 +00:00
2022-05-29 23:10:16 +00:00
return node
}
}
2022-05-29 23:10:16 +00:00
if len(nodes) > 0 {
node.Nodes = nodes
}
2022-05-29 23:10:16 +00:00
if len(values) > 0 {
node.Values = values
}
2022-05-29 23:10:16 +00:00
return node
}
2022-05-29 23:10:16 +00:00
func (p *parser) parsePassthrough() Node {
nodes := []Node{}
2022-05-15 18:22:56 +00:00
2022-05-29 23:10:16 +00:00
for ; p.tok != EOL; p.next() {
nodes = append(nodes, &Ident{Literal: p.lit})
}
if len(nodes) == 0 {
return nil
}
2022-05-14 00:58:55 +00:00
2022-05-29 23:10:16 +00:00
return &PassthroughArgs{Nodes: nodes}
2022-05-14 00:58:55 +00:00
}