Ingen beskrivning
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

main.go 19KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460
  1. package main
  2. import (
  3. "context"
  4. "embed"
  5. "io/fs"
  6. "log"
  7. "net/http"
  8. "strings"
  9. "git.x2erp.com/qdy/go-base/authbase"
  10. "git.x2erp.com/qdy/go-base/config"
  11. "git.x2erp.com/qdy/go-base/container"
  12. "git.x2erp.com/qdy/go-base/ctx"
  13. "git.x2erp.com/qdy/go-base/graceful"
  14. "git.x2erp.com/qdy/go-base/logger"
  15. "git.x2erp.com/qdy/go-base/model/request/configreq"
  16. "git.x2erp.com/qdy/go-base/model/response"
  17. "git.x2erp.com/qdy/go-base/webx"
  18. "git.x2erp.com/qdy/go-base/webx/router"
  19. "git.x2erp.com/qdy/go-db/factory/database"
  20. "git.x2erp.com/qdy/go-db/sqldef"
  21. "git.x2erp.com/qdy/go-svc-configure/internal/service/admin"
  22. "git.x2erp.com/qdy/go-svc-configure/internal/service/bootconfig"
  23. "git.x2erp.com/qdy/go-svc-configure/internal/service/metamanagement"
  24. "git.x2erp.com/qdy/go-svc-configure/internal/service/project"
  25. "git.x2erp.com/qdy/go-svc-configure/internal/service/projectagent"
  26. "git.x2erp.com/qdy/go-svc-configure/internal/service/projecttree"
  27. "git.x2erp.com/qdy/go-svc-configure/internal/service/register"
  28. "git.x2erp.com/qdy/go-svc-configure/internal/service/role"
  29. "git.x2erp.com/qdy/go-svc-configure/internal/service/servicemanagement"
  30. "git.x2erp.com/qdy/go-svc-configure/internal/service/skill"
  31. "git.x2erp.com/qdy/go-svc-configure/internal/service/startup"
  32. "git.x2erp.com/qdy/go-svc-configure/internal/service/svcstartup"
  33. "git.x2erp.com/qdy/go-svc-configure/internal/service/tenant"
  34. "git.x2erp.com/qdy/go-svc-configure/internal/service/token"
  35. "git.x2erp.com/qdy/go-svc-configure/internal/service/user"
  36. "git.x2erp.com/qdy/go-svc-configure/internal/tables"
  37. _ "git.x2erp.com/qdy/go-svc-configure/internal/tables" // 导入表定义包,触发 init() 函数
  38. )
  39. //go:embed web/dist
  40. var frontendFS embed.FS
  41. var (
  42. appName = "svc-configure"
  43. appVersion = "1"
  44. )
  45. func main() {
  46. // 0. 初始化日志
  47. logBootFactory := logger.InitBootLog()
  48. // 1. 获取配置文件
  49. cfg := config.GetConfig()
  50. cfg.SetAppName(appName)
  51. cfg.SetAppVersion(appVersion)
  52. // 2. 创建关闭容器
  53. ctr := container.NewContainer(cfg)
  54. // 注册日志,实现自动关闭
  55. container.Reg(ctr, logBootFactory)
  56. // 3. 创建数据库工厂
  57. dbFactory := container.Create(ctr, database.CreateDBFactory)
  58. dbFactory.TestConnection()
  59. // 创建表
  60. creteTabel(dbFactory)
  61. //启用运行日志
  62. container.Create(ctr, logger.InitRuntimeLogger)
  63. //建立httpservice
  64. //得到webservice服务工厂
  65. webxFactory := webx.GetWebServiceFactory()
  66. //建立hhtpService服务
  67. webServcie, _ := webxFactory.CreateService(cfg.GetServiceConfig())
  68. //建立路由-api
  69. routerService := router.NewWebService(webServcie.GetRouter())
  70. //注册路由--api
  71. registerRoutes(routerService, dbFactory)
  72. // 注册前端静态文件服务
  73. frontendHandler := ServeFrontend()
  74. webServcie.GetRouter().Handle("/", frontendHandler)
  75. //启动服务
  76. webServcie.Run()
  77. //等待关闭
  78. graceful.WaitForShutdown(cfg.GetServiceConfig().ServiceName, ctr, webServcie.GetServer())
  79. }
  80. // 注册所有路由
  81. func registerRoutes(ws *router.RouterService, dbFactory *database.DBFactory) {
  82. // 建立访问配置中心Token
  83. ws.POST("/api/create/config/token",
  84. func(tokenRequest *configreq.ConfigTokenRequest, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[string], error) {
  85. return token.CreateConfigureToken(tokenRequest, ctx, dbFactory, reqCtx), nil
  86. },
  87. ).Use(authbase.TokenAuth).Desc("用户创建自身token").Register()
  88. // svc
  89. ws.POST("/api/delete/config/startup/{servicename}",
  90. func(servicename string, req *configreq.ConfigRequest, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[int64], error) {
  91. return svcstartup.DeleteSVCtartupConfig(servicename, req, ctx, dbFactory, reqCtx), nil
  92. },
  93. ).Use(authbase.BasicAuth).Register()
  94. ws.POST("/api/create/config/startup/{servicename}",
  95. func(servicename string, req *configreq.ConfigRequest, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[int64], error) {
  96. return svcstartup.CreateStartupSVCConfig(servicename, req, ctx, dbFactory, reqCtx), nil
  97. },
  98. ).Use(authbase.TokenAuth).Register()
  99. // service management
  100. ws.POST("/api/services/list",
  101. func(ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[[]servicemanagement.ServiceInfo], error) {
  102. return servicemanagement.GetServices(ctx, dbFactory, reqCtx), nil
  103. },
  104. ).Use(authbase.TokenAuth).Desc("获取微服务列表").Register()
  105. ws.POST("/api/services/create",
  106. func(req *servicemanagement.CreateServiceRequest, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[int64], error) {
  107. return servicemanagement.CreateService(req, ctx, dbFactory, reqCtx), nil
  108. },
  109. ).Use(authbase.TokenAuth).Desc("创建微服务").Register()
  110. ws.POST("/api/services/configs",
  111. func(serviceName string, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[[]servicemanagement.ServiceConfig], error) {
  112. return servicemanagement.GetServiceConfigs(ctx, dbFactory, serviceName, reqCtx), nil
  113. },
  114. ).Use(authbase.TokenAuth).Desc("获取微服务配置项").Register()
  115. ws.POST("/api/services/configs/add",
  116. func(req *servicemanagement.AddServiceConfigRequest, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[int64], error) {
  117. return servicemanagement.AddServiceConfig(req, ctx, dbFactory, reqCtx), nil
  118. },
  119. ).Use(authbase.TokenAuth).Desc("为微服务添加配置项").Register()
  120. // startup
  121. ws.POST("/api/delete/config/startup",
  122. func(req *configreq.ConfigRequest, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[int64], error) {
  123. return startup.DeleteStartupConfig(req, ctx, dbFactory, reqCtx), nil
  124. },
  125. ).Use(authbase.BasicAuth).Register()
  126. ws.POST("/api/create/config/startup",
  127. func(req *configreq.ConfigRequest, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[int64], error) {
  128. return startup.CreateStartupConfig(req, ctx, dbFactory, reqCtx), nil
  129. },
  130. ).Use(authbase.BasicAuth).Register()
  131. // boot config
  132. ws.POST("/api/boot/configs",
  133. func(ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[[]bootconfig.BootConfig], error) {
  134. return bootconfig.GetBootConfigs(ctx, dbFactory, reqCtx), nil
  135. },
  136. ).Use(authbase.BasicAuth).Desc("获取boot配置项").Register()
  137. ws.POST("/api/boot/configs/search",
  138. func(keyword string, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[[]bootconfig.BootConfig], error) {
  139. return bootconfig.SearchBootConfigs(ctx, dbFactory, keyword, reqCtx), nil
  140. },
  141. ).Use(authbase.BasicAuth).Desc("搜索boot配置项").Register()
  142. //init
  143. ws.POST("/api/init/config/meta",
  144. func(ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[map[string]interface{}], error) {
  145. return register.RegisterConfigMeta(ctx, dbFactory, reqCtx), nil
  146. },
  147. ).Use(authbase.BasicAuth).Register()
  148. // config meta management
  149. ws.POST("/api/config/meta/list",
  150. func(ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[[]tables.ConfigMetaDB], error) {
  151. return metamanagement.ListConfigMeta(ctx, dbFactory, reqCtx), nil
  152. },
  153. ).Use(authbase.BasicAuth).Desc("查询配置元信息列表").Register()
  154. ws.POST("/api/config/meta/search",
  155. func(configName, fieldName, yamlName string, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[[]tables.ConfigMetaDB], error) {
  156. return metamanagement.SearchConfigMeta(ctx, dbFactory, configName, fieldName, yamlName, reqCtx), nil
  157. },
  158. ).Use(authbase.BasicAuth).Desc("搜索配置元信息").Register()
  159. // project skill
  160. ws.POST("/api/create/config/project/skill",
  161. func(req *configreq.ProjectSkillRequest, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[int64], error) {
  162. return skill.CreateProjectSkill(req, ctx, dbFactory, reqCtx), nil
  163. },
  164. ).Use(authbase.BasicAuth).Desc("创建项目Skill").Register()
  165. ws.POST("/api/update/config/project/skill/{skill_id}",
  166. func(skillID string, req *configreq.ProjectSkillRequest, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[int64], error) {
  167. return skill.UpdateProjectSkill(skillID, req, ctx, dbFactory, reqCtx), nil
  168. },
  169. ).Use(authbase.BasicAuth).Desc("更新项目Skill").Register()
  170. ws.POST("/api/delete/config/project/skill/{skill_id}",
  171. func(skillID string, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[int64], error) {
  172. return skill.DeleteProjectSkill(skillID, ctx, dbFactory, reqCtx), nil
  173. },
  174. ).Use(authbase.BasicAuth).Desc("删除项目Skill").Register()
  175. ws.POST("/api/query/config/project/skill/{skill_id}",
  176. func(skillID string, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[tables.ProjectSkillDB], error) {
  177. return skill.GetProjectSkill(skillID, ctx, dbFactory, reqCtx), nil
  178. },
  179. ).Use(authbase.BasicAuth).Desc("查询项目Skill").Register()
  180. ws.POST("/api/query/config/project/skills",
  181. func(ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[[]tables.ProjectSkillDB], error) {
  182. return skill.ListProjectSkills(ctx, dbFactory, reqCtx), nil
  183. },
  184. ).Use(authbase.BasicAuth).Desc("查询项目Skill列表").Register()
  185. // project
  186. ws.POST("/api/create/config/project",
  187. func(req *configreq.ProjectRequest, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[int64], error) {
  188. return project.CreateProject(req, ctx, dbFactory, reqCtx), nil
  189. },
  190. ).Use(authbase.BasicAuth).Desc("创建项目").Register()
  191. ws.POST("/api/update/config/project/{project_id}",
  192. func(projectID string, req *configreq.ProjectRequest, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[int64], error) {
  193. return project.UpdateProject(projectID, req, ctx, dbFactory, reqCtx), nil
  194. },
  195. ).Use(authbase.BasicAuth).Desc("更新项目").Register()
  196. ws.POST("/api/delete/config/project/{project_id}",
  197. func(projectID string, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[int64], error) {
  198. return project.DeleteProject(projectID, ctx, dbFactory, reqCtx), nil
  199. },
  200. ).Use(authbase.BasicAuth).Desc("删除项目").Register()
  201. ws.POST("/api/query/config/project/{project_id}",
  202. func(projectID string, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[tables.ProjectDB], error) {
  203. return project.GetProject(projectID, ctx, dbFactory, reqCtx), nil
  204. },
  205. ).Use(authbase.BasicAuth).Desc("查询项目").Register()
  206. ws.POST("/api/query/config/projects",
  207. func(ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[[]tables.ProjectDB], error) {
  208. return project.ListProjects(ctx, dbFactory, reqCtx), nil
  209. },
  210. ).Use(authbase.BasicAuth).Desc("查询项目列表").Register()
  211. // project tree
  212. ws.GET("/api/projects/tree",
  213. func(ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[[]projecttree.TreeNode], error) {
  214. return projecttree.GetProjectTree(ctx, dbFactory, reqCtx), nil
  215. },
  216. ).Use(authbase.BasicAuth).Desc("获取项目树结构").Register()
  217. // project agent
  218. ws.POST("/api/create/config/project/agent",
  219. func(req *configreq.ProjectAgentRequest, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[int64], error) {
  220. return projectagent.CreateProjectAgent(req, ctx, dbFactory, reqCtx), nil
  221. },
  222. ).Use(authbase.BasicAuth).Desc("创建项目Agent").Register()
  223. ws.POST("/api/update/config/project/agent/{agent_id}",
  224. func(agentID string, req *configreq.ProjectAgentRequest, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[int64], error) {
  225. return projectagent.UpdateProjectAgent(agentID, req, ctx, dbFactory, reqCtx), nil
  226. },
  227. ).Use(authbase.BasicAuth).Desc("更新项目Agent").Register()
  228. ws.POST("/api/delete/config/project/agent/{agent_id}",
  229. func(agentID string, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[int64], error) {
  230. return projectagent.DeleteProjectAgent(agentID, ctx, dbFactory, reqCtx), nil
  231. },
  232. ).Use(authbase.BasicAuth).Desc("删除项目Agent").Register()
  233. ws.POST("/api/query/config/project/agent/{agent_id}",
  234. func(agentID string, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[tables.ProjectAgentDB], error) {
  235. return projectagent.GetProjectAgent(agentID, ctx, dbFactory, reqCtx), nil
  236. },
  237. ).Use(authbase.BasicAuth).Desc("查询项目Agent").Register()
  238. ws.POST("/api/query/config/project/agents",
  239. func(ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[[]tables.ProjectAgentDB], error) {
  240. return projectagent.ListProjectAgents(ctx, dbFactory, reqCtx), nil
  241. },
  242. ).Use(authbase.BasicAuth).Desc("查询项目Agent列表").Register()
  243. // tenant
  244. ws.POST("/api/create/config/tenant",
  245. func(req *configreq.TenantRequest, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[int64], error) {
  246. return tenant.CreateTenant(req, ctx, dbFactory, reqCtx), nil
  247. },
  248. ).Use(authbase.BasicAuth).Desc("创建租户").Register()
  249. ws.POST("/api/update/config/tenant/{tenant_id}",
  250. func(tenantID string, req *configreq.TenantRequest, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[int64], error) {
  251. return tenant.UpdateTenant(tenantID, req, ctx, dbFactory, reqCtx), nil
  252. },
  253. ).Use(authbase.BasicAuth).Desc("更新租户").Register()
  254. ws.POST("/api/delete/config/tenant/{tenant_id}",
  255. func(tenantID string, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[int64], error) {
  256. return tenant.DeleteTenant(tenantID, ctx, dbFactory, reqCtx), nil
  257. },
  258. ).Use(authbase.BasicAuth).Desc("删除租户").Register()
  259. ws.POST("/api/query/config/tenant/{tenant_id}",
  260. func(tenantID string, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[tables.TenantDB], error) {
  261. return tenant.GetTenant(tenantID, ctx, dbFactory, reqCtx), nil
  262. },
  263. ).Use(authbase.BasicAuth).Desc("查询租户").Register()
  264. ws.POST("/api/query/config/tenants",
  265. func(ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[[]tables.TenantDB], error) {
  266. return tenant.ListTenants(ctx, dbFactory, reqCtx), nil
  267. },
  268. ).Use(authbase.BasicAuth).Desc("查询租户列表").Register()
  269. // role
  270. ws.POST("/api/create/config/role",
  271. func(req *configreq.RoleRequest, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[int64], error) {
  272. return role.CreateRole(req, ctx, dbFactory, reqCtx), nil
  273. },
  274. ).Use(authbase.BasicAuth).Desc("创建角色").Register()
  275. ws.POST("/api/update/config/role/{role_id}",
  276. func(roleID string, req *configreq.RoleRequest, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[int64], error) {
  277. return role.UpdateRole(roleID, req, ctx, dbFactory, reqCtx), nil
  278. },
  279. ).Use(authbase.BasicAuth).Desc("更新角色").Register()
  280. ws.POST("/api/delete/config/role/{role_id}",
  281. func(roleID string, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[int64], error) {
  282. return role.DeleteRole(roleID, ctx, dbFactory, reqCtx), nil
  283. },
  284. ).Use(authbase.BasicAuth).Desc("删除角色").Register()
  285. ws.POST("/api/query/config/role/{role_id}",
  286. func(roleID string, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[tables.RoleDB], error) {
  287. return role.GetRole(roleID, ctx, dbFactory, reqCtx), nil
  288. },
  289. ).Use(authbase.BasicAuth).Desc("查询角色").Register()
  290. ws.POST("/api/query/config/roles",
  291. func(ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[[]tables.RoleDB], error) {
  292. return role.ListRoles(ctx, dbFactory, reqCtx), nil
  293. },
  294. ).Use(authbase.BasicAuth).Desc("查询角色列表").Register()
  295. // 用户注册(无需认证)
  296. ws.POST("/api/register/user",
  297. func(req *configreq.UserRegisterRequest, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[int64], error) {
  298. return user.RegisterUser(req, ctx, dbFactory, reqCtx), nil
  299. },
  300. ).Desc("用户注册(需邀请码)").Register()
  301. // 用户登录(无需认证)
  302. ws.POST("/api/login/user",
  303. func(req *configreq.UserLoginRequest, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[string], error) {
  304. return user.GetUserToken(req, ctx, dbFactory, reqCtx), nil
  305. },
  306. ).Desc("用户登录(返回token)").Register()
  307. // 管理员登录(无需认证)
  308. ws.POST("/api/login/admin",
  309. func(req *configreq.UserLoginRequest, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[bool], error) {
  310. return admin.LoginAdmin(req, ctx, reqCtx), nil
  311. },
  312. ).Desc("管理员登录(验证配置文件凭据)").Register()
  313. // 创建租户管理员用户(BasicAuth认证)
  314. ws.POST("/api/create/tenant/admin",
  315. func(req *configreq.CreateTenantAdminRequest, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[int64], error) {
  316. return user.CreateTenantAdmin(req, ctx, dbFactory, reqCtx), nil
  317. },
  318. ).Use(authbase.BasicAuth).Desc("创建租户管理员用户").Register()
  319. // 创建邀请码(TokenAuth认证)
  320. ws.POST("/api/create/invitation/code",
  321. func(req *configreq.CreateInvitationCodeRequest, ctx context.Context, reqCtx *ctx.RequestContext) (*response.QueryResult[string], error) {
  322. return user.CreateInvitationCode(req, ctx, dbFactory, reqCtx), nil
  323. },
  324. ).Use(authbase.TokenAuth).Desc("创建邀请码").Register()
  325. }
  326. func creteTabel(factory *database.DBFactory) {
  327. // 获取数据库连接和类型
  328. db := factory.GetDB()
  329. dbType := factory.GetDBType()
  330. // 创建表同步器
  331. syncer, err := sqldef.NewTableSyncer(db, dbType)
  332. if err != nil {
  333. log.Printf("创建 - 建立器失败: %v", err)
  334. }
  335. // 创建表
  336. if err := syncer.CreateTables(); err != nil {
  337. log.Printf("建表失败: %v", err)
  338. }
  339. log.Println("数据库表建立完成!")
  340. }
  341. // ServeFrontend 返回处理前端静态文件的服务处理器
  342. func ServeFrontend() http.Handler {
  343. // 从嵌入的文件系统中获取 dist 子目录
  344. fsys, err := fs.Sub(frontendFS, "web/dist")
  345. if err != nil {
  346. // 如果失败,回退到空文件系统
  347. return http.FileServer(http.FS(frontendFS))
  348. }
  349. // 创建自定义文件服务器,处理 SPA 路由
  350. return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
  351. // 只处理 GET 和 HEAD 请求
  352. if r.Method != "GET" && r.Method != "HEAD" {
  353. // 对于非 GET/HEAD 请求,返回 404 让 API 路由处理
  354. http.NotFound(w, r)
  355. return
  356. }
  357. // API 路径不处理
  358. if strings.HasPrefix(r.URL.Path, "/api/") {
  359. http.NotFound(w, r)
  360. return
  361. }
  362. // 尝试从文件系统读取请求的文件
  363. filePath := strings.TrimPrefix(r.URL.Path, "/")
  364. if filePath == "" {
  365. filePath = "index.html"
  366. }
  367. // 检查文件是否存在
  368. _, err := fs.Stat(fsys, filePath)
  369. if err != nil {
  370. // 文件不存在,返回 index.html 用于 SPA 路由
  371. indexData, err := frontendFS.ReadFile("web/dist/index.html")
  372. if err != nil {
  373. http.Error(w, "Not Found", http.StatusNotFound)
  374. return
  375. }
  376. w.Header().Set("Content-Type", "text/html; charset=utf-8")
  377. w.Write(indexData)
  378. return
  379. }
  380. // 文件存在,正常提供
  381. http.FileServer(http.FS(fsys)).ServeHTTP(w, r)
  382. })
  383. }