123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244 |
- package cli
- import (
- "errors"
- "fmt"
- "git.nspix.com/golang/micro/helper/console"
- "git.nspix.com/golang/micro/helper/unsafestr"
- "sort"
- "strings"
- "sync"
- "sync/atomic"
- "time"
- )
- var (
- _seq int64
- )
- type Executor struct {
- name string
- parent *Executor
- usage string
- description string
- handleFunc HandleFunc
- middleware []Middleware
- children map[string]*Executor
- createdTime time.Time
- seq int64
- locker sync.RWMutex
- params []string //params (eg: /show/user/:uid => :uid)
- NotFoundHandle func(ctx *Context, cmd string) (*Response, error)
- }
- type (
- executors []*Executor
- )
- func (p executors) Swap(i, j int) { p[i], p[j] = p[j], p[i] }
- func (p executors) Len() int { return len(p) }
- func (p executors) Less(i, j int) bool { return p[i].seq < p[j].seq }
- func (exec *Executor) hasChildren(name string) bool {
- exec.locker.RLock()
- defer exec.locker.RUnlock()
- if name == "" {
- return len(exec.children) > 0
- }
- if _, ok := exec.children[name]; ok {
- return true
- }
- return false
- }
- func (exec *Executor) getDescription() string {
- if exec.description != "" {
- return exec.description
- }
- var p *Executor
- var text string
- p = exec
- for {
- if p.parent == nil {
- break
- }
- text = p.name + " " + text
- p = p.parent
- }
- return text
- }
- func (exec *Executor) Children(name string) (*Executor, error) {
- exec.locker.RLock()
- defer exec.locker.RUnlock()
- if cmd, ok := exec.children[name]; ok {
- return cmd, nil
- }
- return nil, errors.New("children not found")
- }
- func (exec *Executor) Completer(tokens ...string) []string {
- exec.locker.RLock()
- defer exec.locker.RUnlock()
- p := exec.children
- length := len(tokens)
- var token, prefix string
- result := make([]string, 0)
- if length > 0 {
- for i := 0; i < length; i++ {
- token = tokens[i]
- if i == length-1 {
- for k, e := range p {
- if token == k {
- if e.children != nil {
- for kk, _ := range e.children {
- result = append(result, prefix+token+" "+kk)
- }
- }
- } else if strings.HasPrefix(k, token) {
- result = append(result, prefix+k)
- }
- }
- } else {
- if next, ok := p[token]; !ok || next.children == nil {
- return []string{}
- } else {
- p = next.children
- }
- }
- prefix += token + " "
- }
- } else {
- for k, _ := range p {
- result = append(result, k)
- }
- }
- return result
- }
- func (exec *Executor) String() string {
- table := console.NewTable()
- var loop func(string, *Executor)
- loop = func(prefix string, e *Executor) {
- if e.children == nil {
- return
- }
- vs := make(executors, len(e.children))
- var i int
- for _, v := range e.children {
- vs[i] = v
- i++
- }
- sort.Sort(vs)
- for _, v := range vs {
- if prefix == "" {
- if v.handleFunc != nil {
- table.AddRow(v.name, v.getDescription())
- }
- } else {
- if v.handleFunc != nil {
- table.AddRow(prefix+" "+v.name, v.getDescription())
- }
- }
- if prefix == "" {
- loop(v.name, v)
- } else {
- loop(prefix+" "+v.name, v)
- }
- }
- }
- loop("", exec)
- buf, _ := table.Marshal()
- return unsafestr.BytesToString(buf)
- }
- func (exec *Executor) Append(child ...*Executor) *Executor {
- exec.locker.Lock()
- defer exec.locker.Unlock()
- for _, v := range child {
- v.parent = exec
- exec.children[v.name] = v
- }
- return exec
- }
- func (exec *Executor) WithHandle(cb HandleFunc) *Executor {
- exec.handleFunc = cb
- return exec
- }
- func (exec *Executor) Do(ctx *Context, args ...string) (res *Response, err error) {
- var (
- root interface{}
- )
- if root = ctx.Get("ROOT"); root == nil {
- ctx.Set("ROOT", exec)
- }
- ctx.Set("arguments", args)
- if len(args) > 0 && exec.hasChildren(args[0]) {
- if len(args) > 1 {
- return exec.children[args[0]].Do(ctx, args[1:]...)
- } else {
- return exec.children[args[0]].Do(ctx)
- }
- }
- if exec.handleFunc == nil {
- if exec.NotFoundHandle != nil {
- res, err = exec.NotFoundHandle(ctx, ctx.CmdStr)
- } else {
- err = fmt.Errorf("%s not found", ctx.CmdStr)
- }
- return
- }
- if len(exec.params) > 0 && len(args) < len(exec.params) {
- s := ""
- for _, v := range exec.params {
- s += "{" + v + "} "
- }
- res = &Response{Code: 1000, Error: fmt.Sprintf("Usage: %s %s", ctx.CmdStr, s)}
- err = nil
- } else {
- ctx.Args = args
- ctx.params = exec.params
- h := exec.handleFunc
- if len(exec.middleware) > 0 {
- for i := len(exec.middleware) - 1; i >= 0; i-- {
- h = exec.middleware[i](h)
- }
- }
- if err = h(ctx); err == nil {
- res = ctx.response
- } else {
- if err == ErrInvalidArgument {
- res = &Response{Code: 1000, Error: fmt.Sprint(ctx.Get("usage"))}
- err = nil
- }
- }
- }
- return
- }
- func NewExecutor(args ...string) *Executor {
- var (
- name string
- usage string
- description string
- )
- if len(args) > 0 {
- name = args[0]
- }
- if len(args) > 1 {
- usage = args[1]
- }
- if len(args) > 2 {
- description = args[2]
- }
- return &Executor{
- name: name,
- usage: usage,
- createdTime: time.Now(),
- description: description,
- children: make(map[string]*Executor),
- seq: atomic.AddInt64(&_seq, 1),
- }
- }
|