arghing #1

Merged
meatballhat merged 23 commits from arghing into main 2022-06-06 22:44:25 +00:00
5 changed files with 84 additions and 26 deletions
Showing only changes of commit 56c6a8cf09 - Show all commits

View File

@ -26,7 +26,7 @@ func tracef(format string, v ...any) {
return return
} }
if _, file, line, ok := runtime.Caller(2); ok { if _, file, line, ok := runtime.Caller(1); ok {
format = fmt.Sprintf("%v:%v ", filepath.Base(file), line) + format format = fmt.Sprintf("%v:%v ", filepath.Base(file), line) + format
} }

View File

@ -42,3 +42,5 @@ type StdinFlag struct{}
type StopFlag struct{} type StopFlag struct{}
type ArgDelimiter struct{} type ArgDelimiter struct{}
type Assign struct{}

View File

@ -65,7 +65,7 @@ func (p *parser2) parseArgs() (*ParseTree, error) {
tracef("parseArgs() returning ParseTree") tracef("parseArgs() returning ParseTree")
return &ParseTree{Nodes: nodes}, nil return &ParseTree{Nodes: nodes}, p.errors.Err()
} }
func (p *parser2) next() { func (p *parser2) next() {
@ -149,6 +149,12 @@ func (p *parser2) parseCommand(cCfg *CommandConfig) Node {
tracef("parseCommand(...) appending %s node=%+#v", tok, flagNode) tracef("parseCommand(...) appending %s node=%+#v", tok, flagNode)
nodes = append(nodes, flagNode) nodes = append(nodes, flagNode)
case ASSIGN:
tracef("parseCommand(...) error on bare %s", p.tok)
p.errors.Add(Position{Column: int(p.pos)}, "invalid bare assignment")
break
default: default:
tracef("parseCommand(...) breaking on %s", p.tok) tracef("parseCommand(...) breaking on %s", p.tok)
break break
@ -175,10 +181,13 @@ func (p *parser2) parseIdent() Node {
func (p *parser2) parseFlag(flCfgMap map[string]FlagConfig) Node { func (p *parser2) parseFlag(flCfgMap map[string]FlagConfig) Node {
switch p.tok { switch p.tok {
case SHORT_FLAG: case SHORT_FLAG:
tracef("parseFlag(...) parsing short flag with config=%+#v", flCfgMap)
return p.parseShortFlag(flCfgMap) return p.parseShortFlag(flCfgMap)
case LONG_FLAG: case LONG_FLAG:
tracef("parseFlag(...) parsing long flag with config=%+#v", flCfgMap)
return p.parseLongFlag(flCfgMap) return p.parseLongFlag(flCfgMap)
case COMPOUND_SHORT_FLAG: case COMPOUND_SHORT_FLAG:
tracef("parseFlag(...) parsing compound short flag with config=%+#v", flCfgMap)
return p.parseCompoundShortFlag(flCfgMap) return p.parseCompoundShortFlag(flCfgMap)
} }
@ -238,7 +247,7 @@ func (p *parser2) parseConfiguredFlag(node *Flag, flCfg FlagConfig) Node {
for i := 0; p.tok != EOL; i++ { for i := 0; p.tok != EOL; i++ {
if !flCfg.NValue.Contains(identIndex) { if !flCfg.NValue.Contains(identIndex) {
tracef("parseLongFlag(...) identIndex=%d exceeds expected=%s; breaking") tracef("parseConfiguredFlag(...) identIndex=%d exceeds expected=%v; breaking", identIndex, flCfg.NValue)
break break
} }
@ -248,18 +257,24 @@ func (p *parser2) parseConfiguredFlag(node *Flag, flCfg FlagConfig) Node {
case ARG_DELIMITER: case ARG_DELIMITER:
nodes = append(nodes, &ArgDelimiter{}) nodes = append(nodes, &ArgDelimiter{})
continue
case ASSIGN:
nodes = append(nodes, &Assign{})
continue continue
case IDENT, STDIN_FLAG: case IDENT, STDIN_FLAG:
name := fmt.Sprintf("%d", identIndex) name := fmt.Sprintf("%d", identIndex)
tracef("parseLongFlag(...) checking for name of identIndex=%d", identIndex) tracef("parseConfiguredFlag(...) checking for name of identIndex=%d", identIndex)
if len(flCfg.ValueNames) > identIndex { if len(flCfg.ValueNames) > identIndex {
name = flCfg.ValueNames[identIndex] name = flCfg.ValueNames[identIndex]
tracef("parseLongFlag(...) setting name=%s from config value names", name) tracef("parseConfiguredFlag(...) setting name=%s from config value names", name)
} else if len(flCfg.ValueNames) == 1 && (flCfg.NValue == OneOrMoreValue || flCfg.NValue == ZeroOrMoreValue) { } else if len(flCfg.ValueNames) == 1 && (flCfg.NValue == OneOrMoreValue || flCfg.NValue == ZeroOrMoreValue) {
name = fmt.Sprintf("%s.%d", flCfg.ValueNames[0], identIndex) name = fmt.Sprintf("%s.%d", flCfg.ValueNames[0], identIndex)
tracef("parseLongFlag(...) setting name=%s from repeating value name", name) tracef("parseConfiguredFlag(...) setting name=%s from repeating value name", name)
} else {
tracef("parseConfiguredFlag(...) setting name=%s", name)
} }
values[name] = p.lit values[name] = p.lit
@ -272,7 +287,7 @@ func (p *parser2) parseConfiguredFlag(node *Flag, flCfg FlagConfig) Node {
identIndex++ identIndex++
default: default:
tracef("parseLongFlag(...) breaking on %s %q %v; setting buffered=true", p.tok, p.lit, p.pos) tracef("parseConfiguredFlag(...) breaking on %s %q %v; setting buffered=true", p.tok, p.lit, p.pos)
p.buffered = true p.buffered = true
if len(nodes) > 0 { if len(nodes) > 0 {

View File

@ -673,21 +673,23 @@ func TestParser2(t *testing.T) {
}, },
}, },
{ {
skip: true,
name: "total weirdo", name: "total weirdo",
args: []string{"PIZZAs", "^wAT@golf", "^^hecKing", "goose", "bonk", "^^FIERCENESS@-2"}, args: []string{"PIZZAs", "^wAT@golf", "^^hecKing", "goose", "bonk", "^^FIERCENESS@-2"},
cfg: &argh.ParserConfig{ cfg: &argh.ParserConfig{
Prog: argh.CommandConfig{ Prog: argh.CommandConfig{
Commands: map[string]argh.CommandConfig{ Commands: map[string]argh.CommandConfig{
"goose": argh.CommandConfig{NValue: 1}, "goose": argh.CommandConfig{
NValue: 1,
Flags: map[string]argh.FlagConfig{
"FIERCENESS": argh.FlagConfig{NValue: 1},
},
},
}, },
Flags: map[string]argh.FlagConfig{ Flags: map[string]argh.FlagConfig{
"w": argh.FlagConfig{}, "w": argh.FlagConfig{},
"A": argh.FlagConfig{}, "A": argh.FlagConfig{},
"T": argh.FlagConfig{NValue: 1}, "T": argh.FlagConfig{NValue: 1},
"hecking": argh.FlagConfig{}, "hecking": argh.FlagConfig{},
"FIERCENESS": argh.FlagConfig{NValue: 1},
}, },
}, },
ScannerConfig: &argh.ScannerConfig{ ScannerConfig: &argh.ScannerConfig{
@ -705,25 +707,46 @@ func TestParser2(t *testing.T) {
Nodes: []argh.Node{ Nodes: []argh.Node{
&argh.Flag{Name: "w"}, &argh.Flag{Name: "w"},
&argh.Flag{Name: "A"}, &argh.Flag{Name: "A"},
&argh.Flag{Name: "T", Values: map[string]string{"0": "golf"}}, &argh.Flag{
Name: "T",
Values: map[string]string{"0": "golf"},
Nodes: []argh.Node{
&argh.Assign{},
&argh.Ident{Literal: "golf"},
},
},
}, },
}, },
&argh.ArgDelimiter{}, &argh.ArgDelimiter{},
&argh.Flag{Name: "hecKing"}, &argh.Flag{Name: "hecKing"},
&argh.ArgDelimiter{}, &argh.ArgDelimiter{},
&argh.Command{Name: "goose", Values: map[string]string{"0": "bonk"}}, &argh.Command{
Name: "goose",
Values: map[string]string{"0": "bonk"},
Nodes: []argh.Node{
&argh.ArgDelimiter{}, &argh.ArgDelimiter{},
&argh.Flag{Name: "FIERCENESS", Values: map[string]string{"0": "-2"}}, &argh.Ident{Literal: "bonk"},
&argh.ArgDelimiter{},
&argh.Flag{
Name: "FIERCENESS",
Values: map[string]string{"0": "-2"},
Nodes: []argh.Node{
&argh.Assign{},
&argh.Ident{Literal: "-2"},
},
},
},
},
}, },
}, },
}, },
}, },
{ {
skip: true,
name: "invalid bare assignment", name: "invalid bare assignment",
args: []string{"pizzas", "=", "--wat"}, args: []string{"pizzas", "=", "--wat"},
expErr: argh.ErrSyntax, expErr: argh.ScannerErrorList{
&argh.ScannerError{Pos: argh.Position{Column: 8}, Msg: "invalid bare assignment"},
},
expPT: []argh.Node{ expPT: []argh.Node{
&argh.Command{ &argh.Command{
Name: "pizzas", Name: "pizzas",
@ -745,7 +768,10 @@ func TestParser2(t *testing.T) {
pt, err := argh.ParseArgs2(tc.args, tc.cfg) pt, err := argh.ParseArgs2(tc.args, tc.cfg)
if err != nil || tc.expErr != nil { if err != nil || tc.expErr != nil {
assert.ErrorIs(ct, err, tc.expErr) if !assert.ErrorIs(ct, err, tc.expErr) {
spew.Dump(err, tc.expErr)
spew.Dump(pt)
}
return return
} }
@ -764,7 +790,9 @@ func TestParser2(t *testing.T) {
pt, err := argh.ParseArgs2(tc.args, tc.cfg) pt, err := argh.ParseArgs2(tc.args, tc.cfg)
if err != nil || tc.expErr != nil { if err != nil || tc.expErr != nil {
assert.ErrorIs(ct, err, tc.expErr) if !assert.ErrorIs(ct, err, tc.expErr) {
spew.Dump(pt)
}
return return
} }

View File

@ -16,6 +16,7 @@ func (e ScannerError) Error() string {
if e.Pos.IsValid() { if e.Pos.IsValid() {
return e.Pos.String() + ":" + e.Msg return e.Pos.String() + ":" + e.Msg
} }
return e.Msg return e.Msg
} }
@ -64,6 +65,18 @@ func (el ScannerErrorList) Err() error {
return el return el
} }
func (el ScannerErrorList) Is(other error) bool {
if _, ok := other.(ScannerErrorList); ok {
return el.Error() == other.Error()
}
if v, ok := other.(*ScannerErrorList); ok {
return el.Error() == (*v).Error()
}
return false
}
func PrintScannerError(w io.Writer, err error) { func PrintScannerError(w io.Writer, err error) {
if list, ok := err.(ScannerErrorList); ok { if list, ok := err.(ScannerErrorList); ok {
for _, e := range list { for _, e := range list {