Нет описания
Вы не можете выбрать более 25 тем Темы должны начинаться с буквы или цифры, могут содержать дефисы(-) и должны содержать не более 35 символов.

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208
  1. package opencode
  2. import (
  3. "bufio"
  4. "fmt"
  5. "io"
  6. "net"
  7. "net/http"
  8. "os"
  9. "os/exec"
  10. "strconv"
  11. "sync"
  12. "time"
  13. "git.x2erp.com/qdy/go-base/logger"
  14. )
  15. // Process 表示 opencode 进程
  16. type Process struct {
  17. cmd *exec.Cmd
  18. port int
  19. logChan chan string
  20. mu sync.Mutex
  21. running bool
  22. }
  23. // GetPort 返回 opencode 服务的端口
  24. func (p *Process) GetPort() int {
  25. return p.port
  26. }
  27. // GetLogs 返回日志通道
  28. func (p *Process) GetLogs() <-chan string {
  29. return p.logChan
  30. }
  31. // Close 停止 opencode 进程(实现 io.Closer 接口)
  32. func (p *Process) Close() error {
  33. return p.Stop()
  34. }
  35. // Stop 停止 opencode 进程
  36. func (p *Process) Stop() error {
  37. p.mu.Lock()
  38. defer p.mu.Unlock()
  39. if !p.running || p.cmd == nil {
  40. return nil
  41. }
  42. logger.Info("正在停止 opencode 进程...")
  43. // 尝试优雅停止
  44. if err := p.cmd.Process.Signal(os.Interrupt); err != nil {
  45. logger.Warn(fmt.Sprintf("发送中断信号失败: %v", err))
  46. // 强制终止
  47. if err := p.cmd.Process.Kill(); err != nil {
  48. logger.Error(fmt.Sprintf("强制终止进程失败: %v", err))
  49. }
  50. }
  51. // 等待进程退出
  52. done := make(chan error, 1)
  53. go func() {
  54. done <- p.cmd.Wait()
  55. }()
  56. select {
  57. case err := <-done:
  58. p.running = false
  59. if err != nil {
  60. logger.Debug(fmt.Sprintf("opencode 进程退出: %v", err))
  61. } else {
  62. logger.Debug("opencode 进程已停止")
  63. }
  64. case <-time.After(5 * time.Second):
  65. logger.Warn("opencode 进程在5秒内未退出,强制终止")
  66. if err := p.cmd.Process.Kill(); err != nil {
  67. logger.Error(fmt.Sprintf("最终强制终止失败: %v", err))
  68. }
  69. p.running = false
  70. }
  71. close(p.logChan)
  72. return nil
  73. }
  74. // Start 启动 opencode 进程
  75. func Start(port int) (*Process, error) {
  76. cmd := exec.Command("opencode", "serve",
  77. "--hostname", "127.0.0.1",
  78. "--port", strconv.Itoa(port),
  79. "--log-level", "INFO",
  80. )
  81. logChan := make(chan string, 100)
  82. // 捕获标准输出
  83. stdout, err := cmd.StdoutPipe()
  84. if err != nil {
  85. return nil, fmt.Errorf("获取标准输出管道失败: %w", err)
  86. }
  87. // 捕获标准错误
  88. stderr, err := cmd.StderrPipe()
  89. if err != nil {
  90. return nil, fmt.Errorf("获取标准错误管道失败: %w", err)
  91. }
  92. // 启动进程
  93. if err := cmd.Start(); err != nil {
  94. return nil, fmt.Errorf("启动 opencode 进程失败: %w", err)
  95. }
  96. process := &Process{
  97. cmd: cmd,
  98. port: port,
  99. logChan: logChan,
  100. running: true,
  101. }
  102. // 读取日志
  103. go readLogs(stdout, logChan, "stdout")
  104. go readLogs(stderr, logChan, "stderr")
  105. // 监控进程退出
  106. go func() {
  107. err := cmd.Wait()
  108. process.mu.Lock()
  109. process.running = false
  110. process.mu.Unlock()
  111. if err != nil {
  112. select {
  113. case logChan <- fmt.Sprintf("进程异常退出: %v", err):
  114. default:
  115. }
  116. } else {
  117. select {
  118. case logChan <- "进程正常退出":
  119. default:
  120. }
  121. }
  122. close(logChan)
  123. }()
  124. logger.Debug(fmt.Sprintf("opencode 进程已启动,端口: %d,PID: %d", port, cmd.Process.Pid))
  125. return process, nil
  126. }
  127. // readLogs 读取进程输出并发送到日志通道
  128. func readLogs(reader io.Reader, logChan chan<- string, source string) {
  129. scanner := bufio.NewScanner(reader)
  130. for scanner.Scan() {
  131. line := scanner.Text()
  132. select {
  133. case logChan <- fmt.Sprintf("[%s] %s", source, line):
  134. default:
  135. // 如果通道满,丢弃旧日志(创建临时通道)
  136. tempChan := make(chan string, 1)
  137. select {
  138. case tempChan <- fmt.Sprintf("[%s] %s", source, line):
  139. // 无法发送,直接丢弃
  140. default:
  141. }
  142. }
  143. }
  144. if err := scanner.Err(); err != nil {
  145. select {
  146. case logChan <- fmt.Sprintf("[%s] 读取错误: %v", source, err):
  147. default:
  148. }
  149. }
  150. }
  151. // GetAvailablePort 获取可用端口
  152. func GetAvailablePort() (int, error) {
  153. // 使用简化的端口获取,无需解析TCP地址
  154. listener, err := net.Listen("tcp", "127.0.0.1:0")
  155. if err != nil {
  156. return 0, err
  157. }
  158. defer listener.Close()
  159. addr := listener.Addr().(*net.TCPAddr)
  160. return addr.Port, nil
  161. }
  162. // WaitForReady 等待 opencode 服务就绪
  163. func (p *Process) WaitForReady(timeout time.Duration) error {
  164. url := fmt.Sprintf("http://127.0.0.1:%d/global/health", p.port)
  165. client := &http.Client{Timeout: 5 * time.Second}
  166. start := time.Now()
  167. for time.Since(start) < timeout {
  168. resp, err := client.Get(url)
  169. if err == nil && resp.StatusCode == 200 {
  170. resp.Body.Close()
  171. return nil
  172. }
  173. if resp != nil {
  174. resp.Body.Close()
  175. }
  176. time.Sleep(100 * time.Millisecond)
  177. }
  178. return fmt.Errorf("opencode 服务在 %v 内未就绪", timeout)
  179. }