mirror of
https://github.com/foomo/posh.git
synced 2025-10-16 12:45:38 +00:00
147 lines
3.7 KiB
Go
147 lines
3.7 KiB
Go
package tree
|
|
|
|
import (
|
|
"context"
|
|
"sort"
|
|
|
|
"github.com/c-bata/go-prompt"
|
|
"github.com/foomo/posh/pkg/readline"
|
|
)
|
|
|
|
type Root struct {
|
|
Name string
|
|
Description string
|
|
Node *Node
|
|
Nodes Nodes
|
|
}
|
|
|
|
// ------------------------------------------------------------------------------------------------
|
|
// ~ Public methods
|
|
// ------------------------------------------------------------------------------------------------
|
|
|
|
func (t *Root) RunExecution(ctx context.Context, r *readline.Readline) error {
|
|
var (
|
|
cmd *Node
|
|
index int
|
|
)
|
|
|
|
switch {
|
|
case t.Node == nil && len(t.Nodes) == 0:
|
|
return ErrNoop
|
|
case r.Args().LenIs(0) && t.Node == nil:
|
|
return ErrMissingCommand
|
|
}
|
|
|
|
if r.Args().LenIs(0) {
|
|
cmd = t.Node
|
|
} else if found, i := t.find(ctx, t.Nodes, r, 0); found != nil {
|
|
cmd = found
|
|
index = i
|
|
} else if t.Node == nil {
|
|
return ErrInvalidCommand
|
|
} else {
|
|
cmd = t.Node
|
|
}
|
|
|
|
if err := cmd.setFlags(r, true); err != nil {
|
|
return err
|
|
} else if err := cmd.execute(ctx, r, index); err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (t *Root) RunCompletion(ctx context.Context, r *readline.Readline) []prompt.Suggest {
|
|
var suggests []prompt.Suggest
|
|
switch r.Mode() {
|
|
case readline.ModeArgs:
|
|
if r.Args().LenLte(1) && len(t.Nodes) > 0 {
|
|
for _, command := range t.Nodes {
|
|
if command.Values != nil {
|
|
for _, name := range command.Values(ctx, r) {
|
|
suggests = append(suggests, prompt.Suggest{Text: name, Description: command.Description})
|
|
}
|
|
} else {
|
|
suggests = append(suggests, prompt.Suggest{Text: command.Name, Description: command.Description})
|
|
}
|
|
}
|
|
} else if cmd, i := t.find(ctx, t.Nodes, r, 0); cmd == nil && t.Node != nil {
|
|
if err := t.Node.setFlags(r, false); err != nil {
|
|
return nil
|
|
} else {
|
|
suggests = t.Node.completeArguments(ctx, t, r, 0)
|
|
}
|
|
} else if cmd == nil {
|
|
return nil
|
|
} else if err := cmd.setFlags(r, false); err != nil {
|
|
return nil
|
|
} else {
|
|
suggests = cmd.completeArguments(ctx, t, r, i+1)
|
|
}
|
|
case readline.ModeFlags:
|
|
if cmd, _ := t.find(ctx, t.Nodes, r, 0); cmd == nil && t.Node != nil {
|
|
if err := t.Node.setFlags(r, false); err != nil {
|
|
return nil
|
|
} else {
|
|
suggests = t.Node.completeFlags(r)
|
|
}
|
|
} else if cmd == nil {
|
|
return nil
|
|
} else if err := cmd.setFlags(r, false); err != nil {
|
|
return nil
|
|
} else {
|
|
suggests = cmd.completeFlags(r)
|
|
}
|
|
case readline.ModePassThroughFlags:
|
|
if cmd, _ := t.find(ctx, t.Nodes, r, 0); cmd == nil && t.Node != nil {
|
|
if err := t.Node.setFlags(r, false); err != nil {
|
|
return nil
|
|
} else {
|
|
suggests = t.Node.completePassThroughFlags(r)
|
|
}
|
|
} else if cmd == nil {
|
|
return nil
|
|
} else if err := cmd.setFlags(r, false); err != nil {
|
|
return nil
|
|
} else {
|
|
suggests = cmd.completePassThroughFlags(r)
|
|
}
|
|
case readline.ModeAdditionalArgs:
|
|
// do nothing
|
|
}
|
|
sort.Slice(suggests, func(i, j int) bool {
|
|
return suggests[i].Text < suggests[j].Text
|
|
})
|
|
return suggests
|
|
}
|
|
|
|
// ------------------------------------------------------------------------------------------------
|
|
// ~ Private methods
|
|
// ------------------------------------------------------------------------------------------------
|
|
|
|
func (t *Root) find(ctx context.Context, cmds []*Node, r *readline.Readline, i int) (*Node, int) {
|
|
if r.Args().LenLt(i + 1) {
|
|
return nil, i
|
|
}
|
|
arg := r.Args().At(i)
|
|
for _, cmd := range cmds {
|
|
if cmd.Name == arg {
|
|
if subCmd, j := t.find(ctx, cmd.Nodes, r, i+1); subCmd != nil {
|
|
return subCmd, j
|
|
}
|
|
return cmd, i
|
|
}
|
|
if cmd.Values != nil {
|
|
for _, name := range cmd.Values(ctx, r) {
|
|
if name == arg {
|
|
if subCmd, j := t.find(ctx, cmd.Nodes, r, i+1); subCmd != nil {
|
|
return subCmd, j
|
|
}
|
|
return cmd, i
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return nil, i
|
|
}
|