server.go 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315
  1. package controller
  2. import (
  3. "fmt"
  4. "net/http"
  5. "regexp"
  6. "strconv"
  7. "time"
  8. "x-ui/web/global"
  9. "x-ui/web/service"
  10. "github.com/gin-gonic/gin"
  11. )
  12. var filenameRegex = regexp.MustCompile(`^[a-zA-Z0-9_\-.]+$`)
  13. type ServerController struct {
  14. BaseController
  15. serverService service.ServerService
  16. settingService service.SettingService
  17. lastStatus *service.Status
  18. lastVersions []string
  19. lastGetVersionsTime int64 // unix seconds
  20. }
  21. func NewServerController(g *gin.RouterGroup) *ServerController {
  22. a := &ServerController{}
  23. a.initRouter(g)
  24. a.startTask()
  25. return a
  26. }
  27. func (a *ServerController) initRouter(g *gin.RouterGroup) {
  28. g.GET("/status", a.status)
  29. g.GET("/cpuHistory/:bucket", a.getCpuHistoryBucket)
  30. g.GET("/getXrayVersion", a.getXrayVersion)
  31. g.GET("/getConfigJson", a.getConfigJson)
  32. g.GET("/getDb", a.getDb)
  33. g.GET("/getNewUUID", a.getNewUUID)
  34. g.GET("/getNewX25519Cert", a.getNewX25519Cert)
  35. g.GET("/getNewmldsa65", a.getNewmldsa65)
  36. g.GET("/getNewmlkem768", a.getNewmlkem768)
  37. g.GET("/getNewVlessEnc", a.getNewVlessEnc)
  38. g.POST("/stopXrayService", a.stopXrayService)
  39. g.POST("/restartXrayService", a.restartXrayService)
  40. g.POST("/installXray/:version", a.installXray)
  41. g.POST("/updateGeofile", a.updateGeofile)
  42. g.POST("/updateGeofile/:fileName", a.updateGeofile)
  43. g.POST("/logs/:count", a.getLogs)
  44. g.POST("/xraylogs/:count", a.getXrayLogs)
  45. g.POST("/importDB", a.importDB)
  46. g.POST("/getNewEchCert", a.getNewEchCert)
  47. }
  48. func (a *ServerController) refreshStatus() {
  49. a.lastStatus = a.serverService.GetStatus(a.lastStatus)
  50. // collect cpu history when status is fresh
  51. if a.lastStatus != nil {
  52. a.serverService.AppendCpuSample(time.Now(), a.lastStatus.Cpu)
  53. }
  54. }
  55. func (a *ServerController) startTask() {
  56. webServer := global.GetWebServer()
  57. c := webServer.GetCron()
  58. c.AddFunc("@every 2s", func() {
  59. // Always refresh to keep CPU history collected continuously.
  60. // Sampling is lightweight and capped to ~6 hours in memory.
  61. a.refreshStatus()
  62. })
  63. }
  64. func (a *ServerController) status(c *gin.Context) { jsonObj(c, a.lastStatus, nil) }
  65. func (a *ServerController) getCpuHistoryBucket(c *gin.Context) {
  66. bucketStr := c.Param("bucket")
  67. bucket, err := strconv.Atoi(bucketStr)
  68. if err != nil || bucket <= 0 {
  69. jsonMsg(c, "invalid bucket", fmt.Errorf("bad bucket"))
  70. return
  71. }
  72. allowed := map[int]bool{
  73. 2: true, // Real-time view
  74. 30: true, // 30s intervals
  75. 60: true, // 1m intervals
  76. 120: true, // 2m intervals
  77. 180: true, // 3m intervals
  78. 300: true, // 5m intervals
  79. }
  80. if !allowed[bucket] {
  81. jsonMsg(c, "invalid bucket", fmt.Errorf("unsupported bucket"))
  82. return
  83. }
  84. points := a.serverService.AggregateCpuHistory(bucket, 60)
  85. jsonObj(c, points, nil)
  86. }
  87. func (a *ServerController) getXrayVersion(c *gin.Context) {
  88. now := time.Now().Unix()
  89. if now-a.lastGetVersionsTime <= 60 { // 1 minute cache
  90. jsonObj(c, a.lastVersions, nil)
  91. return
  92. }
  93. versions, err := a.serverService.GetXrayVersions()
  94. if err != nil {
  95. jsonMsg(c, I18nWeb(c, "getVersion"), err)
  96. return
  97. }
  98. a.lastVersions = versions
  99. a.lastGetVersionsTime = now
  100. jsonObj(c, versions, nil)
  101. }
  102. func (a *ServerController) installXray(c *gin.Context) {
  103. version := c.Param("version")
  104. err := a.serverService.UpdateXray(version)
  105. jsonMsg(c, I18nWeb(c, "pages.index.xraySwitchVersionPopover"), err)
  106. }
  107. func (a *ServerController) updateGeofile(c *gin.Context) {
  108. fileName := c.Param("fileName")
  109. err := a.serverService.UpdateGeofile(fileName)
  110. jsonMsg(c, I18nWeb(c, "pages.index.geofileUpdatePopover"), err)
  111. }
  112. func (a *ServerController) stopXrayService(c *gin.Context) {
  113. err := a.serverService.StopXrayService()
  114. if err != nil {
  115. jsonMsg(c, I18nWeb(c, "pages.xray.stopError"), err)
  116. return
  117. }
  118. jsonMsg(c, I18nWeb(c, "pages.xray.stopSuccess"), err)
  119. }
  120. func (a *ServerController) restartXrayService(c *gin.Context) {
  121. err := a.serverService.RestartXrayService()
  122. if err != nil {
  123. jsonMsg(c, I18nWeb(c, "pages.xray.restartError"), err)
  124. return
  125. }
  126. jsonMsg(c, I18nWeb(c, "pages.xray.restartSuccess"), err)
  127. }
  128. func (a *ServerController) getLogs(c *gin.Context) {
  129. count := c.Param("count")
  130. level := c.PostForm("level")
  131. syslog := c.PostForm("syslog")
  132. logs := a.serverService.GetLogs(count, level, syslog)
  133. jsonObj(c, logs, nil)
  134. }
  135. func (a *ServerController) getXrayLogs(c *gin.Context) {
  136. count := c.Param("count")
  137. filter := c.PostForm("filter")
  138. showDirect := c.PostForm("showDirect")
  139. showBlocked := c.PostForm("showBlocked")
  140. showProxy := c.PostForm("showProxy")
  141. var freedoms []string
  142. var blackholes []string
  143. //getting tags for freedom and blackhole outbounds
  144. config, err := a.settingService.GetDefaultXrayConfig()
  145. if err == nil && config != nil {
  146. if cfgMap, ok := config.(map[string]interface{}); ok {
  147. if outbounds, ok := cfgMap["outbounds"].([]interface{}); ok {
  148. for _, outbound := range outbounds {
  149. if obMap, ok := outbound.(map[string]interface{}); ok {
  150. switch obMap["protocol"] {
  151. case "freedom":
  152. if tag, ok := obMap["tag"].(string); ok {
  153. freedoms = append(freedoms, tag)
  154. }
  155. case "blackhole":
  156. if tag, ok := obMap["tag"].(string); ok {
  157. blackholes = append(blackholes, tag)
  158. }
  159. }
  160. }
  161. }
  162. }
  163. }
  164. }
  165. if len(freedoms) == 0 {
  166. freedoms = []string{"direct"}
  167. }
  168. if len(blackholes) == 0 {
  169. blackholes = []string{"blocked"}
  170. }
  171. logs := a.serverService.GetXrayLogs(count, filter, showDirect, showBlocked, showProxy, freedoms, blackholes)
  172. jsonObj(c, logs, nil)
  173. }
  174. func (a *ServerController) getConfigJson(c *gin.Context) {
  175. configJson, err := a.serverService.GetConfigJson()
  176. if err != nil {
  177. jsonMsg(c, I18nWeb(c, "pages.index.getConfigError"), err)
  178. return
  179. }
  180. jsonObj(c, configJson, nil)
  181. }
  182. func (a *ServerController) getDb(c *gin.Context) {
  183. db, err := a.serverService.GetDb()
  184. if err != nil {
  185. jsonMsg(c, I18nWeb(c, "pages.index.getDatabaseError"), err)
  186. return
  187. }
  188. filename := "x-ui.db"
  189. if !isValidFilename(filename) {
  190. c.AbortWithError(http.StatusBadRequest, fmt.Errorf("invalid filename"))
  191. return
  192. }
  193. // Set the headers for the response
  194. c.Header("Content-Type", "application/octet-stream")
  195. c.Header("Content-Disposition", "attachment; filename="+filename)
  196. // Write the file contents to the response
  197. c.Writer.Write(db)
  198. }
  199. func isValidFilename(filename string) bool {
  200. // Validate that the filename only contains allowed characters
  201. return filenameRegex.MatchString(filename)
  202. }
  203. func (a *ServerController) importDB(c *gin.Context) {
  204. // Get the file from the request body
  205. file, _, err := c.Request.FormFile("db")
  206. if err != nil {
  207. jsonMsg(c, I18nWeb(c, "pages.index.readDatabaseError"), err)
  208. return
  209. }
  210. defer file.Close()
  211. // Always restart Xray before return
  212. defer a.serverService.RestartXrayService()
  213. // lastGetStatusTime removed; no longer needed
  214. // Import it
  215. err = a.serverService.ImportDB(file)
  216. if err != nil {
  217. jsonMsg(c, I18nWeb(c, "pages.index.importDatabaseError"), err)
  218. return
  219. }
  220. jsonObj(c, I18nWeb(c, "pages.index.importDatabaseSuccess"), nil)
  221. }
  222. func (a *ServerController) getNewX25519Cert(c *gin.Context) {
  223. cert, err := a.serverService.GetNewX25519Cert()
  224. if err != nil {
  225. jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.getNewX25519CertError"), err)
  226. return
  227. }
  228. jsonObj(c, cert, nil)
  229. }
  230. func (a *ServerController) getNewmldsa65(c *gin.Context) {
  231. cert, err := a.serverService.GetNewmldsa65()
  232. if err != nil {
  233. jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.getNewmldsa65Error"), err)
  234. return
  235. }
  236. jsonObj(c, cert, nil)
  237. }
  238. func (a *ServerController) getNewEchCert(c *gin.Context) {
  239. sni := c.PostForm("sni")
  240. cert, err := a.serverService.GetNewEchCert(sni)
  241. if err != nil {
  242. jsonMsg(c, "get ech certificate", err)
  243. return
  244. }
  245. jsonObj(c, cert, nil)
  246. }
  247. func (a *ServerController) getNewVlessEnc(c *gin.Context) {
  248. out, err := a.serverService.GetNewVlessEnc()
  249. if err != nil {
  250. jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.getNewVlessEncError"), err)
  251. return
  252. }
  253. jsonObj(c, out, nil)
  254. }
  255. func (a *ServerController) getNewUUID(c *gin.Context) {
  256. uuidResp, err := a.serverService.GetNewUUID()
  257. if err != nil {
  258. jsonMsg(c, "Failed to generate UUID", err)
  259. return
  260. }
  261. jsonObj(c, uuidResp, nil)
  262. }
  263. func (a *ServerController) getNewmlkem768(c *gin.Context) {
  264. out, err := a.serverService.GetNewmlkem768()
  265. if err != nil {
  266. jsonMsg(c, "Failed to generate mlkem768 keys", err)
  267. return
  268. }
  269. jsonObj(c, out, nil)
  270. }