package code import ( "encoding/base64" "os" "os/exec" "path" "strings" "sync" "time" pty "shylinux.com/x/creackpty" ice "shylinux.com/x/icebergs" "shylinux.com/x/icebergs/base/cli" "shylinux.com/x/icebergs/base/ctx" "shylinux.com/x/icebergs/base/mdb" kit "shylinux.com/x/toolkits" ) const XTERM = "xterm" func init() { cache := sync.Map{} add := func(m *ice.Message, key string) string { cmd := exec.Command(cli.SystemFind(m, kit.Select("sh", m.Option(mdb.TYPE)))) cmd.Env = append(os.Environ(), "TERM=xterm") tty, err := pty.Start(cmd) m.Assert(err) cache.Store(key, tty) m.Go(func() { defer m.Cmd(m.PrefixKey(), mdb.PRUNES) defer cache.Delete(key) buf := make([]byte, ice.MOD_BUFS) for { if n, e := tty.Read(buf); !m.Warn(e) { m.Option(mdb.HASH, key) m.Option(mdb.TEXT, base64.StdEncoding.EncodeToString(buf[:n])) m.Option(ice.MSG_DAEMON, m.Conf(m.PrefixKey(), kit.Keys(mdb.HASH, key, mdb.META, mdb.TEXT))) m.PushNoticeGrow("data") } else { break } } m.PushNoticeGrow("exit") }) return key } get := func(m *ice.Message, key string) *os.File { if w, ok := cache.Load(key); ok { if w, ok := w.(*os.File); ok { return w } } add(m, key) if w, ok := cache.Load(key); ok { if w, ok := w.(*os.File); ok { return w } } return nil } Index.MergeCommands(ice.Commands{ XTERM: {Name: "xterm hash refresh", Help: "终端", Actions: ice.MergeAction(ice.Actions{ mdb.INPUTS: {Name: "inputs", Help: "补全", Hand: func(m *ice.Message, arg ...string) { switch mdb.HashInputs(m, arg); arg[0] { case mdb.TYPE: m.Push(arg[0], "python") m.Push(arg[0], "node") m.Push(arg[0], "bash") m.Push(arg[0], "sh") case mdb.NAME: m.Push(arg[0], path.Base(m.Option(mdb.TYPE))) } }}, mdb.CREATE: {Name: "create type name", Help: "创建", Hand: func(m *ice.Message, arg ...string) { if m.Option(mdb.TEXT, m.Option(ice.MSG_DAEMON)) != "" { m.Echo(add(m, mdb.HashCreate(m, arg, m.OptionSimple(mdb.TEXT)).Result())) } }}, mdb.REMOVE: {Name: "remove", Help: "删除", Hand: func(m *ice.Message, arg ...string) { if w, ok := cache.Load(m.Option(mdb.HASH)); ok { if w, ok := w.(*os.File); ok { w.Close() } cache.Delete(m.Option(mdb.HASH)) } mdb.HashRemove(m, m.OptionSimple(mdb.HASH)) }}, mdb.MODIFY: {Name: "modify", Help: "编辑", Hand: func(m *ice.Message, arg ...string) { mdb.HashModify(m, m.OptionSimple(mdb.HASH), arg) }}, mdb.PRUNES: {Name: "prunes", Help: "清理", Hand: func(m *ice.Message, arg ...string) { mdb.HashSelect(m).Tables(func(value ice.Maps) { if _, ok := cache.Load(value[mdb.HASH]); !ok || kit.Time(m.Time())-kit.Time(value[mdb.TIME]) > int64(time.Hour) { m.Cmd(m.PrefixKey(), mdb.REMOVE, kit.Dict(value)) } }) }}, "resize": {Name: "resize", Help: "大小", Hand: func(m *ice.Message, arg ...string) { pty.Setsize(get(m, m.Option(mdb.HASH)), &pty.Winsize{Rows: uint16(kit.Int(m.Option("rows"))), Cols: uint16(kit.Int(m.Option("cols")))}) }}, "rename": {Name: "rename", Help: "重命名", Hand: func(m *ice.Message, arg ...string) { mdb.HashModify(m, m.OptionSimple(mdb.HASH), arg) }}, "select": {Name: "select", Help: "连接", Hand: func(m *ice.Message, arg ...string) { mdb.HashModify(m, m.OptionSimple(mdb.HASH), mdb.TEXT, m.Option(ice.MSG_DAEMON)) m.Cmd(m.PrefixKey(), "input", arg) }}, "input": {Name: "input", Help: "输入", Hand: func(m *ice.Message, arg ...string) { mdb.HashModify(m, m.OptionSimple(mdb.HASH), mdb.TIME, m.Time()) if b, e := base64.StdEncoding.DecodeString(strings.Join(arg, "")); m.Assert(e) { get(m, m.Option(mdb.HASH)).Write(b) } }}, }, mdb.HashAction(mdb.FIELD, "time,hash,type,name,text,extra"), ctx.CmdAction()), Hand: func(m *ice.Message, arg ...string) { mdb.HashSelect(m, kit.Slice(arg, 0, 1)...) m.DisplayLocal("") }}, }) }