handler.go 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146
  1. package agent
  2. import (
  3. "errors"
  4. "fmt"
  5. "github.com/labstack/gommon/log"
  6. "io"
  7. "net/http"
  8. "nginx-ui/server/constants"
  9. "nginx-ui/server/models"
  10. nginx2 "nginx-ui/server/nginx"
  11. "nginx-ui/server/utils"
  12. "os"
  13. "strings"
  14. "time"
  15. )
  16. var OnAgentMessageHandler func(agent *Agent, message *models.AgentData) (interface{}, error)
  17. // OnNginxUpdated 不管是Agent注册,还是服务端修改Nginx信息,均会调用该方法
  18. func OnNginxUpdated(c *Agent, data *models.AgentData) (interface{}, error) {
  19. nginx := &models.Nginx{}
  20. err := data.ReadData(nginx)
  21. if err != nil {
  22. return nil, err
  23. }
  24. c.Nginx = nginx
  25. return make(map[string]interface{}), nil
  26. }
  27. func OnServerConnected(c *Agent, data *models.AgentData) (interface{}, error) {
  28. log.Printf("OnServerConnected: %v,%v\n", c.Token, data)
  29. if c.Token == "" {
  30. log.Warn("missing token")
  31. return nil, errors.New("missing token")
  32. }
  33. nginx := &models.Nginx{
  34. Name: "Agent",
  35. IsServer: false,
  36. Proxy: true,
  37. IsLocal: true,
  38. Password: "",
  39. Remark: "agent",
  40. IpAddr: utils.GetNetIP(),
  41. DataDir: "/data/nginx",
  42. NginxPath: "/usr/sbin/nginx",
  43. NginxDir: "/etc/nginx",
  44. }
  45. request := &models.AgentData{
  46. Type: models.RegisterNginxType,
  47. Data: nginx,
  48. }
  49. res, err := c.Send(request, 20*time.Second)
  50. if err != nil {
  51. log.Warn("register fail for: %v\n", err)
  52. return nil, err
  53. }
  54. if res != nil {
  55. err = res.ReadData(nginx)
  56. if err != nil {
  57. log.Warn("register fail: %v\n", err)
  58. return nil, nil
  59. }
  60. c.Nginx = nginx
  61. }
  62. return c.Nginx, nil
  63. }
  64. // OnCMD 执行Nginx命令
  65. func OnCMD(c *Agent, data *models.AgentData) (interface{}, error) {
  66. nginx := c.Nginx
  67. if nginx == nil {
  68. go func() {
  69. _, _ = OnServerConnected(c, data)
  70. }()
  71. return nil, errors.New("nginx agent not register")
  72. }
  73. request := &models.AgentCMD{}
  74. err := data.ReadData(request)
  75. if err != nil {
  76. return nil, err
  77. }
  78. instance := nginx2.NewLocalNginx(nginx)
  79. res, err := instance.Run(request.Cmd)
  80. return res, err
  81. }
  82. // OnSendFile 服务端复制文件到客户端
  83. // 先使用Http接口下载文件,然后本地解压复制
  84. func OnSendFile(c *Agent, data *models.AgentData) (interface{}, error) {
  85. nginx := c.Nginx
  86. if nginx == nil {
  87. ngx, err := OnServerConnected(c, data)
  88. if err != nil {
  89. return nil, err
  90. }
  91. if ngx == nil {
  92. return nil, errors.New("nginx agent not register")
  93. }
  94. nginx = ngx.(*models.Nginx)
  95. }
  96. request := &models.AgentSendFile{}
  97. err := data.ReadData(request)
  98. if err != nil {
  99. log.Warn("send file fail for: %v\n", nginx)
  100. return nil, err
  101. }
  102. protocol := "http"
  103. if c.SSL == "Y" {
  104. protocol = "https"
  105. }
  106. url := fmt.Sprintf("%v://%v%v", protocol, c.Url, constants.DownloadFileUrl)
  107. url = strings.Replace(url, ":nginxId", fmt.Sprintf("%d", nginx.Id), 1)
  108. url = strings.Replace(url, ":fileName", request.FileName, 1)
  109. log.Printf("download file: %s\n", url)
  110. out, err := os.Create(request.Dst)
  111. if err != nil {
  112. log.Warn("create file fail for: %v\n", nginx)
  113. return nil, err
  114. }
  115. defer out.Close()
  116. req, err := http.NewRequest("GET", url, nil)
  117. if err != nil {
  118. log.Warn("create file fail for: %v\n", nginx)
  119. return nil, err
  120. }
  121. req.Header.Set("Token", c.Token)
  122. client := &http.Client{}
  123. resp, err := client.Do(req)
  124. if err != nil {
  125. log.Warn("download file fail for: %v\n", nginx)
  126. return nil, err
  127. }
  128. defer resp.Body.Close()
  129. _, err = io.Copy(out, resp.Body)
  130. if err != nil {
  131. log.Warn("download file fail for: %v\n", nginx)
  132. return nil, err
  133. }
  134. return make(map[string]interface{}), nil
  135. }