server.go 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200
  1. package controller
  2. import (
  3. "fmt"
  4. "net/http"
  5. "regexp"
  6. "time"
  7. "x-ui/web/global"
  8. "x-ui/web/service"
  9. "github.com/gin-gonic/gin"
  10. )
  11. var filenameRegex = regexp.MustCompile(`^[a-zA-Z0-9_\-.]+$`)
  12. type ServerController struct {
  13. BaseController
  14. serverService service.ServerService
  15. lastStatus *service.Status
  16. lastGetStatusTime time.Time
  17. lastVersions []string
  18. lastGetVersionsTime time.Time
  19. }
  20. func NewServerController(g *gin.RouterGroup) *ServerController {
  21. a := &ServerController{
  22. lastGetStatusTime: time.Now(),
  23. }
  24. a.initRouter(g)
  25. a.startTask()
  26. return a
  27. }
  28. func (a *ServerController) initRouter(g *gin.RouterGroup) {
  29. g = g.Group("/server")
  30. g.Use(a.checkLogin)
  31. g.POST("/status", a.status)
  32. g.POST("/getXrayVersion", a.getXrayVersion)
  33. g.POST("/stopXrayService", a.stopXrayService)
  34. g.POST("/restartXrayService", a.restartXrayService)
  35. g.POST("/installXray/:version", a.installXray)
  36. g.POST("/updateGeofile/:fileName", a.updateGeofile)
  37. g.POST("/logs/:count", a.getLogs)
  38. g.POST("/getConfigJson", a.getConfigJson)
  39. g.GET("/getDb", a.getDb)
  40. g.POST("/importDB", a.importDB)
  41. g.POST("/getNewX25519Cert", a.getNewX25519Cert)
  42. }
  43. func (a *ServerController) refreshStatus() {
  44. a.lastStatus = a.serverService.GetStatus(a.lastStatus)
  45. }
  46. func (a *ServerController) startTask() {
  47. webServer := global.GetWebServer()
  48. c := webServer.GetCron()
  49. c.AddFunc("@every 2s", func() {
  50. now := time.Now()
  51. if now.Sub(a.lastGetStatusTime) > time.Minute*3 {
  52. return
  53. }
  54. a.refreshStatus()
  55. })
  56. }
  57. func (a *ServerController) status(c *gin.Context) {
  58. a.lastGetStatusTime = time.Now()
  59. jsonObj(c, a.lastStatus, nil)
  60. }
  61. func (a *ServerController) getXrayVersion(c *gin.Context) {
  62. now := time.Now()
  63. if now.Sub(a.lastGetVersionsTime) <= time.Minute {
  64. jsonObj(c, a.lastVersions, nil)
  65. return
  66. }
  67. versions, err := a.serverService.GetXrayVersions()
  68. if err != nil {
  69. jsonMsg(c, I18nWeb(c, "getVersion"), err)
  70. return
  71. }
  72. a.lastVersions = versions
  73. a.lastGetVersionsTime = time.Now()
  74. jsonObj(c, versions, nil)
  75. }
  76. func (a *ServerController) installXray(c *gin.Context) {
  77. version := c.Param("version")
  78. err := a.serverService.UpdateXray(version)
  79. jsonMsg(c, I18nWeb(c, "pages.index.xraySwitchVersionPopover"), err)
  80. }
  81. func (a *ServerController) updateGeofile(c *gin.Context) {
  82. fileName := c.Param("fileName")
  83. err := a.serverService.UpdateGeofile(fileName)
  84. jsonMsg(c, I18nWeb(c, "pages.index.geofileUpdatePopover"), err)
  85. }
  86. func (a *ServerController) stopXrayService(c *gin.Context) {
  87. a.lastGetStatusTime = time.Now()
  88. err := a.serverService.StopXrayService()
  89. if err != nil {
  90. jsonMsg(c, "", err)
  91. return
  92. }
  93. jsonMsg(c, "Xray stopped", err)
  94. }
  95. func (a *ServerController) restartXrayService(c *gin.Context) {
  96. err := a.serverService.RestartXrayService()
  97. if err != nil {
  98. jsonMsg(c, "", err)
  99. return
  100. }
  101. jsonMsg(c, "Xray restarted", err)
  102. }
  103. func (a *ServerController) getLogs(c *gin.Context) {
  104. count := c.Param("count")
  105. level := c.PostForm("level")
  106. syslog := c.PostForm("syslog")
  107. logs := a.serverService.GetLogs(count, level, syslog)
  108. jsonObj(c, logs, nil)
  109. }
  110. func (a *ServerController) getConfigJson(c *gin.Context) {
  111. configJson, err := a.serverService.GetConfigJson()
  112. if err != nil {
  113. jsonMsg(c, "get config.json", err)
  114. return
  115. }
  116. jsonObj(c, configJson, nil)
  117. }
  118. func (a *ServerController) getDb(c *gin.Context) {
  119. db, err := a.serverService.GetDb()
  120. if err != nil {
  121. jsonMsg(c, "get Database", err)
  122. return
  123. }
  124. filename := "x-ui.db"
  125. if !isValidFilename(filename) {
  126. c.AbortWithError(http.StatusBadRequest, fmt.Errorf("invalid filename"))
  127. return
  128. }
  129. // Set the headers for the response
  130. c.Header("Content-Type", "application/octet-stream")
  131. c.Header("Content-Disposition", "attachment; filename="+filename)
  132. // Write the file contents to the response
  133. c.Writer.Write(db)
  134. }
  135. func isValidFilename(filename string) bool {
  136. // Validate that the filename only contains allowed characters
  137. return filenameRegex.MatchString(filename)
  138. }
  139. func (a *ServerController) importDB(c *gin.Context) {
  140. // Get the file from the request body
  141. file, _, err := c.Request.FormFile("db")
  142. if err != nil {
  143. jsonMsg(c, "Error reading db file", err)
  144. return
  145. }
  146. defer file.Close()
  147. // Always restart Xray before return
  148. defer a.serverService.RestartXrayService()
  149. defer func() {
  150. a.lastGetStatusTime = time.Now()
  151. }()
  152. // Import it
  153. err = a.serverService.ImportDB(file)
  154. if err != nil {
  155. jsonMsg(c, "", err)
  156. return
  157. }
  158. jsonObj(c, "Import DB", nil)
  159. }
  160. func (a *ServerController) getNewX25519Cert(c *gin.Context) {
  161. cert, err := a.serverService.GetNewX25519Cert()
  162. if err != nil {
  163. jsonMsg(c, "get x25519 certificate", err)
  164. return
  165. }
  166. jsonObj(c, cert, nil)
  167. }