web.go 7.6 KB

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