main.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504
  1. package main
  2. import (
  3. "bufio"
  4. "encoding/base64"
  5. "encoding/json"
  6. "fmt"
  7. "io"
  8. "io/fs"
  9. "os"
  10. "os/exec"
  11. "os/signal"
  12. "os/user"
  13. "reflect"
  14. "regexp"
  15. "strings"
  16. "syscall"
  17. "time"
  18. "git.wecise.com/wecise/common/matrix/logger"
  19. "git.wecise.com/wecise/common/matrix/util"
  20. "github.com/creack/pty"
  21. "golang.org/x/term"
  22. "gopkg.in/yaml.v2"
  23. )
  24. func init() {
  25. logger.SetConsole(true)
  26. logger.SetLevel(logger.TRACE)
  27. logger.SetFormat("yyyy-MM-dd HH:mm:ss.SSS [level] msg", "\n")
  28. }
  29. type KV struct{ Key, Val string }
  30. func parseArgs(args []string) (kvs []*KV) {
  31. argk := ""
  32. argv := ""
  33. for _, arg := range args {
  34. if argk != "" {
  35. argv = arg
  36. } else if regexp.MustCompile(`^\-\w+$`).MatchString(arg) {
  37. argk = arg[1:]
  38. continue
  39. } else {
  40. kv := strings.SplitN(arg, "=", 2)
  41. if len(kv) == 2 {
  42. argk = kv[0]
  43. argv = kv[1]
  44. } else {
  45. argk = ""
  46. argv = arg
  47. }
  48. }
  49. kvs = append(kvs, &KV{argk, argv})
  50. argk, argv = "", ""
  51. }
  52. if argk != "" {
  53. kvs = append(kvs, &KV{argk, argv})
  54. }
  55. return
  56. }
  57. func usage() {
  58. fmt.Println("usage:")
  59. fmt.Println(" msh p=password|a=passcode [[c=]command [p=password|a=passcode] [x=cmd-end-regexp] [[r=regexp] [o=output|n=outputline]]...]...")
  60. fmt.Println(" a=passcode should be base64 encoded, or use p=password")
  61. fmt.Println(" debug info include: p(progress) a(argments) m(match) 1(all)")
  62. }
  63. func main() {
  64. // get user
  65. u := func() string {
  66. user, err := user.Current()
  67. if err != nil {
  68. logger.Error(util.ErrorWithSourceLine(err))
  69. return ""
  70. }
  71. return user.Username
  72. }()
  73. // get passcode,取默认密码设置
  74. c := ""
  75. kvs := parseArgs(os.Args)
  76. for _, kv := range kvs {
  77. if kv.Key == "a" {
  78. c = kv.Val
  79. if c == "" {
  80. c = "="
  81. }
  82. break
  83. }
  84. if kv.Key == "p" {
  85. c = "=" + kv.Val
  86. break
  87. }
  88. }
  89. // get password
  90. p := c
  91. if p == "" {
  92. usage()
  93. return
  94. } else if p[0:1] == "=" {
  95. p = p[1:]
  96. } else {
  97. x, e := base64.RawStdEncoding.DecodeString(p)
  98. if e == nil {
  99. p = string(x)
  100. }
  101. // else 不是Base64编码,保持原值
  102. }
  103. // explainArgs key, val
  104. cmds := []*Command{{Cmd: "sh", Password: c, Regexps: []*Matcher{{Regexp: nil, Output: ""}}, Endregx: regxprompt}}
  105. i := 0
  106. // 掠过第一个参数,当前执行程序
  107. kvs = kvs[1:]
  108. for _, kv := range kvs {
  109. key, val := kv.Key, kv.Val
  110. switch key {
  111. case "", "cmd", "command", "c":
  112. cmds = append(cmds, &Command{Cmd: val, Password: c, Regexps: []*Matcher{{Regexp: nil, Output: ""}}, Endregx: regxprompt})
  113. case "ry", "regex_yes_no":
  114. re, err := regexp.Compile(val)
  115. if err != nil {
  116. logger.Error("arg", i, util.ErrorWithSourceLine(err))
  117. return
  118. } else {
  119. regxyesno = &Regexp{re}
  120. }
  121. case "rc", "regex_passcode", "regex_password":
  122. re, err := regexp.Compile(val)
  123. if err != nil {
  124. logger.Error("arg", i, util.ErrorWithSourceLine(err))
  125. return
  126. } else {
  127. regxpassword = &Regexp{re}
  128. }
  129. case "rp", "regex_prompt":
  130. re, err := regexp.Compile(val)
  131. if err != nil {
  132. logger.Error("arg", i, util.ErrorWithSourceLine(err))
  133. return
  134. } else {
  135. regxprompt = &Regexp{re}
  136. }
  137. case "password", "code", "pass", "p":
  138. cmds[len(cmds)-1].Password = "=" + val
  139. case "passcode", "b64code", "a":
  140. cmds[len(cmds)-1].Password = val
  141. case "re", "r", "regex":
  142. if val == "" {
  143. cmds[len(cmds)-1].Regexps = append(cmds[len(cmds)-1].Regexps, &Matcher{Regexp: nil})
  144. } else {
  145. re, err := regexp.Compile(val)
  146. if err != nil {
  147. logger.Error("arg", i, util.ErrorWithSourceLine(err))
  148. return
  149. } else {
  150. cmds[len(cmds)-1].Regexps = append(cmds[len(cmds)-1].Regexps, &Matcher{Regexp: &Regexp{re}})
  151. }
  152. }
  153. case "x", "end":
  154. if val == "" {
  155. cmds[len(cmds)-1].Endregx = regxprompt
  156. } else {
  157. re, err := regexp.Compile(val)
  158. if err != nil {
  159. logger.Error("arg", i, util.ErrorWithSourceLine(err))
  160. return
  161. } else {
  162. cmds[len(cmds)-1].Endregx = &Regexp{re}
  163. }
  164. }
  165. case "out", "o", "output":
  166. cmds[len(cmds)-1].Regexps[len(cmds[len(cmds)-1].Regexps)-1].Output += val
  167. case "outln", "n", "outputline":
  168. cmds[len(cmds)-1].Regexps[len(cmds[len(cmds)-1].Regexps)-1].Output += val + "\n"
  169. case "debug", "d":
  170. cmds[len(cmds)-1].Regexps[len(cmds[len(cmds)-1].Regexps)-1].Debug += val
  171. }
  172. }
  173. if strings.Index(cmds[0].Regexps[0].Debug, "a") >= 0 || strings.Index(cmds[0].Regexps[0].Debug, "1") >= 0 {
  174. //bs, _ := json.MarshalIndent(cmds, "", " ")
  175. bs, _ := yaml.Marshal(cmds)
  176. logger.Debug("arguments:\n" + string(bs))
  177. }
  178. client := &Client{
  179. User: u,
  180. Password: p,
  181. Commands: cmds,
  182. }
  183. client.Run()
  184. }
  185. type Regexp struct {
  186. *regexp.Regexp
  187. }
  188. func (r *Regexp) MarshalJSON() ([]byte, error) {
  189. if r == nil || r.Regexp == nil {
  190. return json.Marshal(nil)
  191. }
  192. return json.Marshal(r.String())
  193. }
  194. func (r *Regexp) MarshalYAML() (interface{}, error) {
  195. if r == nil || r.Regexp == nil {
  196. return nil, nil
  197. }
  198. return r.String(), nil
  199. }
  200. type Matcher struct {
  201. Regexp *Regexp
  202. Output string `yaml:"output"`
  203. Debug string `yaml:"debug"`
  204. }
  205. type Command struct {
  206. Cmd string `yaml:"cmd"`
  207. Password string `yaml:"password"`
  208. Regexps []*Matcher
  209. Endregx *Regexp
  210. }
  211. var regxyesno = &Regexp{regexp.MustCompile(`.*(\(yes\/no\)\?)\s*$`)}
  212. var regxpassword = &Regexp{regexp.MustCompile(`.*([Pp]assword:)\s*$`)}
  213. var regxprompt = &Regexp{regexp.MustCompile(`.*([%#\$\>])\s*$`)}
  214. type Client struct {
  215. User string
  216. Password string
  217. Commands []*Command
  218. }
  219. func Pipe(reader io.Reader, writer io.Writer, pfs ...func(lastbs []byte, sin string) (sout string)) {
  220. btr := NewBTReader("", reader, 30*time.Millisecond, 1024)
  221. lastbs := []byte{'\n'}
  222. for {
  223. bs, err := btr.Read()
  224. if err != nil {
  225. if err == io.EOF {
  226. return
  227. }
  228. logger.Error(util.ErrorWithSourceLine(err))
  229. return
  230. }
  231. if len(bs) > 0 {
  232. xbs := bs
  233. for _, pf := range pfs {
  234. if pf != nil {
  235. s := pf(lastbs, string(xbs))
  236. xbs = []byte(s)
  237. }
  238. }
  239. if writer != nil {
  240. writer.Write(xbs)
  241. }
  242. lastbs = bs
  243. }
  244. }
  245. }
  246. func (c *Client) Run() {
  247. logger.Info("msh ready")
  248. // Create arbitrary command.
  249. cmd := exec.Command("sh", "-c", "export PS1='\\h:> ';sh -i")
  250. // Start the command with a pty.
  251. ptmx, err := pty.Start(cmd)
  252. if err != nil {
  253. logger.Error(err)
  254. return
  255. }
  256. // Make sure to close the pty at the end.
  257. defer func() { _ = ptmx.Close() }() // Best effort.
  258. // Handle pty size.
  259. ch := make(chan os.Signal, 1)
  260. signal.Notify(ch, syscall.SIGWINCH)
  261. go func() {
  262. for range ch {
  263. if err := pty.InheritSize(os.Stdin, ptmx); err != nil {
  264. logger.Error("error resizing pty:", err)
  265. }
  266. }
  267. }()
  268. ch <- syscall.SIGWINCH // Initial resize.
  269. defer func() { signal.Stop(ch); close(ch) }() // Cleanup signals when done.
  270. // Set stdin in raw mode.
  271. oldState, err := term.MakeRaw(int(os.Stdin.Fd()))
  272. if err != nil {
  273. logger.Error(err)
  274. }
  275. go Pipe(os.Stdin, ptmx)
  276. e := c.IOProc(ptmx, os.Stdout, ptmx)
  277. if oldState != nil {
  278. term.Restore(int(os.Stdin.Fd()), oldState)
  279. fmt.Println()
  280. }
  281. if cmd.ProcessState.ExitCode() == -1 && e != nil {
  282. logger.Error(e)
  283. return
  284. }
  285. logger.Info("msh exit")
  286. return
  287. }
  288. func (c *Client) IOProc(cmdout io.Reader, stdout io.Writer, cmdin io.Writer) error {
  289. cmdidx := 0
  290. as := ""
  291. change := false
  292. stdoutbtr := NewBTReader("", cmdout, 50*time.Millisecond, 1024)
  293. for {
  294. bs, err := stdoutbtr.Read()
  295. if err != nil {
  296. if err == io.EOF {
  297. return nil
  298. }
  299. if _, y := err.(*fs.PathError); y {
  300. return nil
  301. }
  302. return util.ErrorWithSourceLine(err, reflect.TypeOf(err))
  303. }
  304. if len(bs) > 0 {
  305. change = true
  306. s := string(bs)
  307. stdout.Write([]byte(s))
  308. as += s
  309. } else if change {
  310. change = false
  311. if cmdidx < len(c.Commands) {
  312. if msi := c.Commands[cmdidx].Endregx.FindStringSubmatchIndex(as); msi != nil {
  313. // logger.Error("EndregexMatch", as)
  314. match := as[msi[0]:msi[1]]
  315. if strings.Index(c.Commands[cmdidx].Regexps[0].Debug, "m") >= 0 || strings.Index(c.Commands[cmdidx].Regexps[0].Debug, "1") >= 0 {
  316. logger.Trace("match end:", "'"+match+"'")
  317. }
  318. as = "" // 全清,开始新的命令
  319. cmdidx++
  320. if cmdidx >= len(c.Commands) {
  321. return nil
  322. }
  323. if strings.Index(c.Commands[cmdidx].Regexps[0].Debug, "p") >= 0 || strings.Index(c.Commands[cmdidx].Regexps[0].Debug, "1") >= 0 {
  324. logger.Trace("command:", c.Commands[cmdidx].Cmd)
  325. }
  326. // stdout.Write([]byte(c.Commands[cmdidx].Cmd + "\n"))
  327. cmdin.Write([]byte(c.Commands[cmdidx].Cmd + "\n"))
  328. continue
  329. }
  330. for _, regex := range c.Commands[cmdidx].Regexps {
  331. if regex == nil {
  332. continue
  333. }
  334. if regex.Regexp == nil {
  335. // like match all
  336. if regex.Output != "" {
  337. // stdout.Write([]byte(regex.Output))
  338. cmdin.Write([]byte(regex.Output))
  339. }
  340. } else if msi := regex.Regexp.FindStringSubmatchIndex(as); msi != nil {
  341. match := as[msi[0]:msi[1]]
  342. if len(msi) >= 4 {
  343. match = as[msi[2]:msi[3]]
  344. as = as[msi[3]:] // 清除已处理完的内容
  345. } else {
  346. as = as[msi[1]:] // 清除已处理完的内容
  347. }
  348. if strings.Index(regex.Debug, "m") >= 0 || strings.Index(regex.Debug, "1") >= 0 {
  349. logger.Trace("match regex:", "'"+match+"'")
  350. }
  351. if regex.Output != "" {
  352. // stdout.Write([]byte(regex.Output))
  353. cmdin.Write([]byte(regex.Output))
  354. }
  355. }
  356. }
  357. }
  358. if msi := regxyesno.FindStringSubmatchIndex(as); msi != nil {
  359. match := as[msi[0]:msi[1]]
  360. if len(msi) >= 4 {
  361. as = as[msi[3]:] // 清除已处理完的内容
  362. } else {
  363. as = as[msi[1]:] // 清除已处理完的内容
  364. }
  365. if strings.Index(c.Commands[0].Regexps[0].Debug, "m") >= 0 || strings.Index(c.Commands[0].Regexps[0].Debug, "1") >= 0 ||
  366. cmdidx < len(c.Commands) &&
  367. (strings.Index(c.Commands[cmdidx].Regexps[0].Debug, "m") >= 0 || strings.Index(c.Commands[cmdidx].Regexps[0].Debug, "1") >= 0) {
  368. logger.Trace("match yesno:", "'"+match+"'")
  369. }
  370. // stdout.Write([]byte("yes\n"))
  371. cmdin.Write([]byte("yes\n"))
  372. }
  373. if msi := regxpassword.FindStringSubmatchIndex(as); msi != nil {
  374. // logger.Error(as)
  375. match := as[msi[0]:msi[1]]
  376. if len(msi) >= 4 {
  377. as = as[msi[3]:] // 清除已处理完的内容
  378. } else {
  379. as = as[msi[1]:] // 清除已处理完的内容
  380. }
  381. p := c.Commands[0].Password
  382. if cmdidx < len(c.Commands) {
  383. p = c.Commands[cmdidx].Password
  384. }
  385. if strings.Index(c.Commands[0].Regexps[0].Debug, "m") >= 0 || strings.Index(c.Commands[0].Regexps[0].Debug, "1") >= 0 ||
  386. cmdidx < len(c.Commands) &&
  387. (strings.Index(c.Commands[cmdidx].Regexps[0].Debug, "m") >= 0 || strings.Index(c.Commands[cmdidx].Regexps[0].Debug, "1") >= 0) {
  388. logger.Trace("match password:", "'"+match+"'")
  389. }
  390. if p != "" {
  391. if p[0:1] == "=" {
  392. p = p[1:]
  393. } else {
  394. x, e := base64.RawStdEncoding.DecodeString(p)
  395. if e == nil {
  396. p = string(x)
  397. }
  398. // else 不是Base64编码,保持原值
  399. }
  400. // don't echo password
  401. if c.Commands[0].Regexps[0].Debug != "" || cmdidx < len(c.Commands) && c.Commands[cmdidx].Regexps[0].Debug != "" {
  402. stdout.Write([]byte(p + "\n"))
  403. }
  404. cmdin.Write([]byte(p + "\n"))
  405. }
  406. }
  407. if len(as) > 1024 {
  408. as = as[len(as)-1024:]
  409. }
  410. }
  411. }
  412. }
  413. type BTReader struct {
  414. *bufio.Reader
  415. flag string
  416. chbs chan []byte
  417. cher chan error
  418. timeout time.Duration
  419. sizeout int
  420. err error
  421. }
  422. func NewBTReader(flag string, reader io.Reader, timeout time.Duration, sizeout int) (me *BTReader) {
  423. me = &BTReader{Reader: bufio.NewReader(reader),
  424. flag: flag,
  425. chbs: make(chan []byte),
  426. cher: make(chan error),
  427. timeout: timeout,
  428. sizeout: sizeout,
  429. }
  430. go func() {
  431. bs := make([]byte, me.Size())
  432. for {
  433. n, err := me.Reader.Read(bs[:1])
  434. if err != nil {
  435. me.cher <- err
  436. return
  437. }
  438. x, err := me.Reader.Read(bs[1 : me.Reader.Buffered()+1])
  439. if err != nil {
  440. me.cher <- err
  441. return
  442. }
  443. n += x
  444. abs := make([]byte, n)
  445. copy(abs, bs[:n])
  446. me.chbs <- abs
  447. }
  448. }()
  449. return
  450. }
  451. // 指定时间内没有新数据进入,且有积累数据,或积累数据超过指定数量,即返回
  452. func (me *BTReader) Read() (rbs []byte, err error) {
  453. if me.err != nil {
  454. return nil, me.err
  455. }
  456. for {
  457. t := time.NewTimer(me.timeout)
  458. select {
  459. case me.err = <-me.cher:
  460. if len(rbs) > 0 {
  461. // 返回最后的数据,下次读时才返回错误
  462. return rbs, nil
  463. }
  464. return nil, me.err
  465. case abs := <-me.chbs:
  466. rbs = append(rbs, abs...)
  467. if len(rbs) > me.sizeout {
  468. return
  469. }
  470. t.Stop()
  471. t.Reset(me.timeout)
  472. case <-t.C:
  473. if len(rbs) == 0 {
  474. t.Stop()
  475. t.Reset(me.timeout)
  476. }
  477. return
  478. }
  479. }
  480. }