1
0

web.go 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423
  1. package web
  2. import (
  3. "context"
  4. "crypto/tls"
  5. "embed"
  6. "github.com/BurntSushi/toml"
  7. "github.com/gin-contrib/sessions"
  8. "github.com/gin-contrib/sessions/cookie"
  9. "github.com/gin-gonic/gin"
  10. "github.com/nicksnyder/go-i18n/v2/i18n"
  11. "github.com/robfig/cron/v3"
  12. "golang.org/x/text/language"
  13. "html/template"
  14. "io"
  15. "io/fs"
  16. "net"
  17. "net/http"
  18. "os"
  19. "strconv"
  20. "strings"
  21. "time"
  22. "x-ui/config"
  23. "x-ui/logger"
  24. "x-ui/util/common"
  25. "x-ui/web/controller"
  26. "x-ui/web/service"
  27. )
  28. //go:embed assets/*
  29. var assetsFS embed.FS
  30. //go:embed html/*
  31. var htmlFS embed.FS
  32. //go:embed translation/*
  33. var i18nFS embed.FS
  34. var startTime = time.Now()
  35. type wrapAssetsFS struct {
  36. embed.FS
  37. }
  38. func (f *wrapAssetsFS) Open(name string) (fs.File, error) {
  39. file, err := f.FS.Open("assets/" + name)
  40. if err != nil {
  41. return nil, err
  42. }
  43. return &wrapAssetsFile{
  44. File: file,
  45. }, nil
  46. }
  47. type wrapAssetsFile struct {
  48. fs.File
  49. }
  50. func (f *wrapAssetsFile) Stat() (fs.FileInfo, error) {
  51. info, err := f.File.Stat()
  52. if err != nil {
  53. return nil, err
  54. }
  55. return &wrapAssetsFileInfo{
  56. FileInfo: info,
  57. }, nil
  58. }
  59. type wrapAssetsFileInfo struct {
  60. fs.FileInfo
  61. }
  62. func (f *wrapAssetsFileInfo) ModTime() time.Time {
  63. return startTime
  64. }
  65. type Server struct {
  66. httpServer *http.Server
  67. listener net.Listener
  68. index *controller.IndexController
  69. server *controller.ServerController
  70. xui *controller.XUIController
  71. xrayService service.XrayService
  72. settingService service.SettingService
  73. inboundService service.InboundService
  74. cron *cron.Cron
  75. ctx context.Context
  76. cancel context.CancelFunc
  77. }
  78. func NewServer() *Server {
  79. ctx, cancel := context.WithCancel(context.Background())
  80. return &Server{
  81. ctx: ctx,
  82. cancel: cancel,
  83. }
  84. }
  85. func (s *Server) getHtmlFiles() ([]string, error) {
  86. files := make([]string, 0)
  87. dir, _ := os.Getwd()
  88. err := fs.WalkDir(os.DirFS(dir), "web/html", func(path string, d fs.DirEntry, err error) error {
  89. if err != nil {
  90. return err
  91. }
  92. if d.IsDir() {
  93. return nil
  94. }
  95. files = append(files, path)
  96. return nil
  97. })
  98. if err != nil {
  99. return nil, err
  100. }
  101. return files, nil
  102. }
  103. func (s *Server) getHtmlTemplate(funcMap template.FuncMap) (*template.Template, error) {
  104. t := template.New("").Funcs(funcMap)
  105. err := fs.WalkDir(htmlFS, "html", func(path string, d fs.DirEntry, err error) error {
  106. if err != nil {
  107. return err
  108. }
  109. if d.IsDir() {
  110. newT, err := t.ParseFS(htmlFS, path+"/*.html")
  111. if err != nil {
  112. // ignore
  113. return nil
  114. }
  115. t = newT
  116. }
  117. return nil
  118. })
  119. if err != nil {
  120. return nil, err
  121. }
  122. return t, nil
  123. }
  124. func (s *Server) initRouter() (*gin.Engine, error) {
  125. if config.IsDebug() {
  126. gin.SetMode(gin.DebugMode)
  127. } else {
  128. gin.DefaultWriter = io.Discard
  129. gin.DefaultErrorWriter = io.Discard
  130. gin.SetMode(gin.ReleaseMode)
  131. }
  132. engine := gin.Default()
  133. secret, err := s.settingService.GetSecret()
  134. if err != nil {
  135. return nil, err
  136. }
  137. basePath, err := s.settingService.GetBasePath()
  138. if err != nil {
  139. return nil, err
  140. }
  141. assetsBasePath := basePath + "assets/"
  142. store := cookie.NewStore(secret)
  143. engine.Use(sessions.Sessions("session", store))
  144. engine.Use(func(c *gin.Context) {
  145. c.Set("base_path", basePath)
  146. })
  147. engine.Use(func(c *gin.Context) {
  148. uri := c.Request.RequestURI
  149. if strings.HasPrefix(uri, assetsBasePath) {
  150. c.Header("Cache-Control", "max-age=31536000")
  151. }
  152. })
  153. err = s.initI18n(engine)
  154. if err != nil {
  155. return nil, err
  156. }
  157. if config.IsDebug() {
  158. // for develop
  159. files, err := s.getHtmlFiles()
  160. if err != nil {
  161. return nil, err
  162. }
  163. engine.LoadHTMLFiles(files...)
  164. engine.StaticFS(basePath+"assets", http.FS(os.DirFS("web/assets")))
  165. } else {
  166. // for prod
  167. t, err := s.getHtmlTemplate(engine.FuncMap)
  168. if err != nil {
  169. return nil, err
  170. }
  171. engine.SetHTMLTemplate(t)
  172. engine.StaticFS(basePath+"assets", http.FS(&wrapAssetsFS{FS: assetsFS}))
  173. }
  174. g := engine.Group(basePath)
  175. s.index = controller.NewIndexController(g)
  176. s.server = controller.NewServerController(g)
  177. s.xui = controller.NewXUIController(g)
  178. return engine, nil
  179. }
  180. func (s *Server) initI18n(engine *gin.Engine) error {
  181. bundle := i18n.NewBundle(language.SimplifiedChinese)
  182. bundle.RegisterUnmarshalFunc("toml", toml.Unmarshal)
  183. err := fs.WalkDir(i18nFS, "translation", func(path string, d fs.DirEntry, err error) error {
  184. if err != nil {
  185. return err
  186. }
  187. if d.IsDir() {
  188. return nil
  189. }
  190. data, err := i18nFS.ReadFile(path)
  191. if err != nil {
  192. return err
  193. }
  194. _, err = bundle.ParseMessageFileBytes(data, path)
  195. return err
  196. })
  197. if err != nil {
  198. return err
  199. }
  200. findI18nParamNames := func(key string) []string {
  201. names := make([]string, 0)
  202. keyLen := len(key)
  203. for i := 0; i < keyLen-1; i++ {
  204. if key[i:i+2] == "{{" { // 判断开头 "{{"
  205. j := i + 2
  206. isFind := false
  207. for ; j < keyLen-1; j++ {
  208. if key[j:j+2] == "}}" { // 结尾 "}}"
  209. isFind = true
  210. break
  211. }
  212. }
  213. if isFind {
  214. names = append(names, key[i+3:j])
  215. }
  216. }
  217. }
  218. return names
  219. }
  220. var localizer *i18n.Localizer
  221. engine.FuncMap["i18n"] = func(key string, params ...string) (string, error) {
  222. names := findI18nParamNames(key)
  223. if len(names) != len(params) {
  224. return "", common.NewError("find names:", names, "---------- params:", params, "---------- num not equal")
  225. }
  226. templateData := map[string]interface{}{}
  227. for i := range names {
  228. templateData[names[i]] = params[i]
  229. }
  230. return localizer.Localize(&i18n.LocalizeConfig{
  231. MessageID: key,
  232. TemplateData: templateData,
  233. })
  234. }
  235. engine.Use(func(c *gin.Context) {
  236. accept := c.GetHeader("Accept-Language")
  237. localizer = i18n.NewLocalizer(bundle, accept)
  238. c.Set("localizer", localizer)
  239. c.Next()
  240. })
  241. return nil
  242. }
  243. func (s *Server) startTask() {
  244. err := s.xrayService.RestartXray()
  245. if err != nil {
  246. logger.Warning("start xray failed:", err)
  247. }
  248. var checkTime = 0
  249. // 每 30 秒检查一次 xray 是否在运行
  250. s.cron.AddFunc("@every 30s", func() {
  251. if s.xrayService.IsXrayRunning() {
  252. checkTime = 0
  253. return
  254. }
  255. checkTime++
  256. if checkTime < 2 {
  257. return
  258. }
  259. s.xrayService.SetToNeedRestart()
  260. })
  261. go func() {
  262. time.Sleep(time.Second * 5)
  263. // 每 10 秒统计一次流量,首次启动延迟 5 秒,与重启 xray 的时间错开
  264. s.cron.AddFunc("@every 10s", func() {
  265. if !s.xrayService.IsXrayRunning() {
  266. return
  267. }
  268. traffics, err := s.xrayService.GetXrayTraffic()
  269. if err != nil {
  270. logger.Warning("get xray traffic failed:", err)
  271. return
  272. }
  273. err = s.inboundService.AddTraffic(traffics)
  274. if err != nil {
  275. logger.Warning("add traffic failed:", err)
  276. }
  277. })
  278. }()
  279. // 每 30 秒检查一次 inbound 流量超出情况
  280. s.cron.AddFunc("@every 30s", func() {
  281. count, err := s.inboundService.DisableInvalidInbounds()
  282. if err != nil {
  283. logger.Warning("disable invalid inbounds err:", err)
  284. } else if count > 0 {
  285. logger.Debugf("disabled %v inbounds", count)
  286. s.xrayService.SetToNeedRestart()
  287. }
  288. })
  289. }
  290. func (s *Server) Start() (err error) {
  291. defer func() {
  292. if err != nil {
  293. s.Stop()
  294. }
  295. }()
  296. loc, err := s.settingService.GetTimeLocation()
  297. if err != nil {
  298. return err
  299. }
  300. s.cron = cron.New(cron.WithLocation(loc), cron.WithSeconds())
  301. s.cron.Start()
  302. engine, err := s.initRouter()
  303. if err != nil {
  304. return err
  305. }
  306. certFile, err := s.settingService.GetCertFile()
  307. if err != nil {
  308. return err
  309. }
  310. keyFile, err := s.settingService.GetKeyFile()
  311. if err != nil {
  312. return err
  313. }
  314. listen, err := s.settingService.GetListen()
  315. if err != nil {
  316. return err
  317. }
  318. port, err := s.settingService.GetPort()
  319. if err != nil {
  320. return err
  321. }
  322. listenAddr := net.JoinHostPort(listen, strconv.Itoa(port))
  323. var listener net.Listener
  324. if certFile != "" || keyFile != "" {
  325. var cert tls.Certificate
  326. cert, err = tls.LoadX509KeyPair(certFile, keyFile)
  327. if err != nil {
  328. return err
  329. }
  330. c := &tls.Config{
  331. Certificates: []tls.Certificate{cert},
  332. }
  333. listener, err = tls.Listen("tcp", listenAddr, c)
  334. } else {
  335. listener, err = net.Listen("tcp", listenAddr)
  336. }
  337. if err != nil {
  338. return err
  339. }
  340. if certFile != "" || keyFile != "" {
  341. logger.Info("web server run https on", listener.Addr())
  342. } else {
  343. logger.Info("web server run http on", listener.Addr())
  344. }
  345. s.listener = listener
  346. s.startTask()
  347. s.httpServer = &http.Server{
  348. Handler: engine,
  349. }
  350. go s.httpServer.Serve(listener)
  351. return nil
  352. }
  353. func (s *Server) Stop() error {
  354. s.cancel()
  355. s.xrayService.StopXray()
  356. if s.cron != nil {
  357. s.cron.Stop()
  358. }
  359. var err1 error
  360. var err2 error
  361. if s.httpServer != nil {
  362. err1 = s.httpServer.Shutdown(s.ctx)
  363. }
  364. if s.listener != nil {
  365. err2 = s.listener.Close()
  366. }
  367. return common.Combine(err1, err2)
  368. }
  369. func (s *Server) GetCtx() context.Context {
  370. return s.ctx
  371. }
  372. func (s *Server) GetCron() *cron.Cron {
  373. return s.cron
  374. }