cmd_Task.go 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160
  1. package utils
  2. import (
  3. "bytes"
  4. "fmt"
  5. "io"
  6. "os"
  7. "os/exec"
  8. "path/filepath"
  9. "runtime"
  10. "sync"
  11. )
  12. type RunTask interface {
  13. AddTask()
  14. RunTask()
  15. }
  16. // T: Task任务
  17. type T struct {
  18. sync.Mutex
  19. // ch: 获取事件channel
  20. ch chan struct{}
  21. closeChan chan struct{}
  22. // 记录process对象
  23. p *os.Process
  24. // f: 执行任务
  25. f func(chan struct{}) error
  26. }
  27. // NewT: 实例化方法
  28. func NewT() *T {
  29. return newT(nil)
  30. }
  31. func newT(f func(chan struct{}) error) *T {
  32. t := &T{
  33. Mutex: sync.Mutex{},
  34. ch: make(chan struct{}, 1),
  35. closeChan: make(chan struct{}),
  36. f: f,
  37. }
  38. if f == nil {
  39. t.f = t.DefaultF
  40. }
  41. return t
  42. }
  43. func (t *T) AddTask() {
  44. t.Lock()
  45. defer t.Unlock()
  46. if len(t.ch) == 1 {
  47. // 代表已经有任务了
  48. // 直接丢弃这次任务
  49. return
  50. }
  51. t.ch <- struct{}{}
  52. }
  53. func (t *T) RunTask() {
  54. fmt.Println("进入")
  55. // 这里做的make 是用于关闭上一个执行的任务
  56. ch := make(chan struct{})
  57. // 先run服务
  58. go t.f(ch)
  59. for {
  60. _, ok := <-t.ch
  61. ch <- struct{}{}
  62. if !ok {
  63. return
  64. }
  65. // 等待上一个关闭
  66. <-t.closeChan
  67. go t.f(ch)
  68. }
  69. }
  70. // DefaultF: 默认的StartFunction
  71. func (t *T) DefaultF(ch chan struct{}) error {
  72. var buildCmd *exec.Cmd
  73. var cmd *exec.Cmd
  74. // 判断是否有makefile
  75. _, err := os.Stat(filepath.Join("Makefile"))
  76. if runtime.GOOS != "windows" && err == nil {
  77. _, err := exec.LookPath("make")
  78. if err == nil {
  79. cmd = exec.Command("makefile")
  80. goto makefile
  81. }
  82. }
  83. // 检测系统是否有编译环境
  84. _, err = exec.LookPath("go")
  85. if err != nil {
  86. return err
  87. }
  88. // build
  89. switch runtime.GOOS {
  90. case "windows":
  91. buildCmd = exec.Command("go", "build", "-o", "gva.exe", "main.go")
  92. default:
  93. buildCmd = exec.Command("go", "build", "-o", "gva", "main.go")
  94. }
  95. err = buildCmd.Run()
  96. if err != nil {
  97. return err
  98. }
  99. fmt.Println("build 执行完成")
  100. // 执行
  101. switch runtime.GOOS {
  102. case "windows":
  103. cmd = exec.Command("gva.exe")
  104. default:
  105. cmd = exec.Command("./gva")
  106. }
  107. makefile:
  108. // 开始执行任务
  109. t.echo(cmd)
  110. <-ch
  111. // 回收资源
  112. err = cmd.Process.Kill()
  113. fmt.Println("kill err", err)
  114. // 发送关闭完成信号
  115. t.closeChan <- struct{}{}
  116. return err
  117. }
  118. // echo: 封装回显
  119. func (t *T) echo(cmd *exec.Cmd) error {
  120. var stdoutBuf bytes.Buffer
  121. stdoutIn, _ := cmd.StdoutPipe()
  122. var errStdout, errStderr error
  123. stdout := io.MultiWriter(os.Stdout, &stdoutBuf)
  124. err := cmd.Start()
  125. if err != nil {
  126. return err
  127. }
  128. go func() {
  129. _, errStdout = io.Copy(stdout, stdoutIn)
  130. }()
  131. t.p = cmd.Process
  132. fmt.Println("pid", t.p.Pid)
  133. go func() {
  134. _ = cmd.Wait()
  135. if errStdout != nil || errStderr != nil {
  136. fmt.Printf("failed to capture stdout or stderr\n")
  137. }
  138. outStr := string(stdoutBuf.Bytes())
  139. fmt.Printf("\nout:\n%s\n", outStr)
  140. }()
  141. return nil
  142. }