arghing #1
@ -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
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -42,3 +42,5 @@ type StdinFlag struct{}
|
|||||||
type StopFlag struct{}
|
type StopFlag struct{}
|
||||||
|
|
||||||
type ArgDelimiter struct{}
|
type ArgDelimiter struct{}
|
||||||
|
|
||||||
|
type Assign struct{}
|
||||||
|
@ -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 {
|
||||||
|
@ -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
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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 {
|
||||||
|
Loading…
Reference in New Issue
Block a user