package ctx import ( "path" "strings" ice "shylinux.com/x/icebergs" "shylinux.com/x/icebergs/base/aaa" "shylinux.com/x/icebergs/base/mdb" "shylinux.com/x/icebergs/base/nfs" kit "shylinux.com/x/toolkits" "shylinux.com/x/toolkits/logs" ) func _command_list(m *ice.Message, name string) { if nfs.ExistsFile(m, path.Join(ice.SRC, name)) { switch kit.Ext(name) { case nfs.JS: m.Push(DISPLAY, FileURI(name)) name = kit.Select(CAN_PLUGIN, GetFileCmd(name)) case nfs.GO: name = GetFileCmd(name) default: if msg := m.Cmd(mdb.RENDER, kit.Ext(name)); msg.Length() > 0 { m.Push(ARGS, kit.Format(kit.List(name))) name = kit.Keys(msg.Append(mdb.TEXT), msg.Append(mdb.NAME)) } } } if strings.HasPrefix(name, "can.") { m.Push(mdb.INDEX, name) return } if name == "" { for k, v := range m.Source().Commands { if k[0] == '/' || k[0] == '_' { continue } m.Push(mdb.KEY, k) m.Push(mdb.NAME, v.Name) m.Push(mdb.HELP, v.Help) } m.Sort(mdb.KEY) return } m.Spawn(m.Source()).Search(name, func(p *ice.Context, s *ice.Context, key string, cmd *ice.Command) { m.Push(mdb.INDEX, kit.Keys(s.Cap(ice.CTX_FOLLOW), key)) m.Push(mdb.NAME, kit.Format(cmd.Name)) m.Push(mdb.HELP, kit.Format(cmd.Help)) m.Push(mdb.META, kit.Format(cmd.Meta)) m.Push(mdb.LIST, kit.Format(cmd.List)) }) } func _command_search(m *ice.Message, kind, name, text string) { m.Travel(func(p *ice.Context, s *ice.Context, key string, cmd *ice.Command) { if key[0] == '/' || key[0] == '_' { return } if name != "" && !strings.HasPrefix(key, name) && !strings.Contains(s.Name, name) { return } m.PushSearch(ice.CTX, kit.PathName(1), ice.CMD, kit.FileName(1), kit.SimpleKV("", s.Cap(ice.CTX_FOLLOW), cmd.Name, cmd.Help), CONTEXT, s.Cap(ice.CTX_FOLLOW), COMMAND, key, mdb.HELP, cmd.Help, INDEX, kit.Keys(s.Cap(ice.CTX_FOLLOW), key), ) }) } const ( INDEX = "index" ARGS = "args" STYLE = "style" DISPLAY = "display" ACTION = "action" TRANS = "trans" CAN_PLUGIN = "can.plugin" ) const COMMAND = "command" func init() { Index.MergeCommands(ice.Commands{ COMMAND: {Name: "command key auto", Help: "命令", Actions: ice.MergeActions(ice.Actions{ ice.CTX_INIT: {Hand: func(m *ice.Message, arg ...string) { TravelCmd(m, func(key, file, line string) { if strings.Contains(file, ice.ICEBERGS) { AddFileCmd(file, key) } }) }}, mdb.SEARCH: {Hand: func(m *ice.Message, arg ...string) { if arg[0] == m.CommandKey() || len(arg) > 1 && arg[1] != "" { _command_search(m, arg[0], kit.Select("", arg, 1), kit.Select("", arg, 2)) } }}, "tags": {Hand: func(m *ice.Message, arg ...string) { TravelCmd(m, func(key, file, line string) { m.Push("name", key) m.Push("file", file) m.Push("line", line) }) m.Sort("name") m.Tables(func(value ice.Maps) { m.Echo("%s\t%s\t%s;\" f\n", value["name"], value["file"], value["line"]) }) m.Cmd("nfs.save", "tags", m.Result()) }}, }, aaa.RoleAction()), Hand: func(m *ice.Message, arg ...string) { if len(arg) == 0 { arg = append(arg, "") } for _, key := range arg { _command_list(m, key) } }}, }) } var runChecker = []func(*ice.Message, string, string, ...string) bool{} func AddRunChecker(cb func(*ice.Message, string, string, ...string) bool) { runChecker = append(runChecker, cb) } func init() { AddRunChecker(func(m *ice.Message, cmd, check string, arg ...string) bool { switch check { case mdb.REMOVE: m.Cmd(CONFIG, mdb.REMOVE, cmd) return true case mdb.SELECT: m.Cmdy(CONFIG, cmd) return true default: return false } }) } func Run(m *ice.Message, arg ...string) { if len(arg) > 3 && arg[1] == ACTION && arg[2] == CONFIG { for _, check := range runChecker { if check(m, arg[0], arg[3], arg...) { return } } } if !PodCmd(m, arg) && aaa.Right(m, arg) { m.Cmdy(arg) } } func PodCmd(m *ice.Message, arg ...ice.Any) bool { if pod := m.Option(ice.POD); pod != "" { if m.Option(ice.POD, ""); len(kit.Simple(m.Optionv(ice.MSG_UPLOAD))) == 1 { m.Cmdy("cache", "upload").Option(ice.MSG_UPLOAD, m.Append(mdb.HASH), m.Append(mdb.NAME), m.Append(nfs.SIZE)) } m.Cmdy(append(kit.List(ice.SPACE, pod), arg...)) return true } return false } func CmdHandler(args ...ice.Any) ice.Handler { return func(m *ice.Message, arg ...string) { m.Cmdy(args...) } } func CmdAction(args ...ice.Any) ice.Actions { return ice.Actions{ice.CTX_INIT: mdb.AutoConfig(args...), COMMAND: {Hand: func(m *ice.Message, arg ...string) { if !PodCmd(m, COMMAND, arg) { m.Cmdy(COMMAND, arg) } }}, ice.RUN: {Hand: Run}, } } func FileURI(dir string) string { if strings.Contains(dir, "go/pkg/mod") { return path.Join("/require", strings.Split(dir, "go/pkg/mod")[1]) } if ice.Info.Make.Path != "" && strings.HasPrefix(dir, ice.Info.Make.Path+ice.PS) { dir = strings.TrimPrefix(dir, ice.Info.Make.Path+ice.PS) } if strings.HasPrefix(dir, kit.Path("")+ice.PS) { dir = strings.TrimPrefix(dir, kit.Path("")+ice.PS) } if strings.HasPrefix(dir, ice.USR) { return path.Join("/require", dir) } if strings.HasPrefix(dir, ice.SRC) { return path.Join("/require", dir) } if nfs.ExistsFile(ice.Pulse, path.Join(ice.SRC, dir)) { return path.Join("/require/src/", dir) } return dir } func FileCmd(dir string) string { dir = strings.Split(dir, ice.DF)[0] dir = strings.ReplaceAll(dir, ".js", ".go") dir = strings.ReplaceAll(dir, ".sh", ".go") return FileURI(dir) } func AddFileCmd(dir, key string) { ice.Info.File[FileCmd(dir)] = key } func GetFileCmd(dir string) string { if strings.HasPrefix(dir, "usr/") { // p := ice.Pulse.Cmdx("cli.system", "git", "config", "remote.origin.url", kit.Dict("cmd_dir", path.Dir(dir))) // p = strings.Replace(strings.TrimSpace(p), "https://", "/require/", 1) // dir = path.Join(p, strings.Join(strings.Split(dir, "/")[2:], "/")) } if strings.HasPrefix(dir, ".ish/pluged/") { dir = strings.Replace(dir, ".ish/pluged/", "/require/", 1) } if strings.HasPrefix(dir, "require/") { dir = "/" + dir } for _, dir := range []string{dir, path.Join("/require/", ice.Info.Make.Module, dir), path.Join("/require/", ice.Info.Make.Module, ice.SRC, dir)} { if cmd, ok := ice.Info.File[FileCmd(dir)]; ok { return cmd } p := path.Dir(dir) if cmd, ok := ice.Info.File[FileCmd(path.Join(p, path.Base(p)+".go"))]; ok { return cmd } } return "" } func GetCmdFile(m *ice.Message, cmds string) (file string) { m.Search(cmds, func(key string, cmd *ice.Command) { if cmd.RawHand == nil { file = kit.Split(logs.FileLines(cmd.Hand), ":")[0] } else { for k, v := range ice.Info.File { if v == cmds { file = strings.Replace(k, "/require/", ".ish/pluged/", 1) } } } }) return } func TravelCmd(m *ice.Message, cb func(key, file, line string)) { m.Travel(func(p *ice.Context, s *ice.Context, key string, cmd *ice.Command) { if key[0] == '/' || key[0] == '_' { return // 内部命令 } ls := kit.Split(cmd.GetFileLine(), ":") if len(ls) > 1 { cb(kit.Keys(s.Cap(ice.CTX_FOLLOW), key), strings.TrimPrefix(ls[0], kit.Path("")+ice.PS), ls[1]) } else { m.Warn(true, "not founc", cmd.Name) } }) }