handler.go 3.4 KB

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