api.ts 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134
  1. import { OpenAPIHono } from '@hono/zod-openapi'
  2. import { swaggerUI } from '@hono/swagger-ui'
  3. import { errorHandler } from './utils/errorHandler'
  4. import usersRouter from './api/users/index'
  5. import authRoute from './api/auth/index'
  6. import rolesRoute from './api/roles/index'
  7. import fileRoute from './api/files/index'
  8. import membershipPlanRoute from './api/membership-plans/index'
  9. import paymentRoute from './api/payments/index'
  10. import templateRoute from './api/templates/index'
  11. import publicTemplateRoute from './api/public/templates/index'
  12. import publicSettingsRoute from './api/public/settings/index'
  13. import settingsRoute from './api/settings/index'
  14. import documentsRoute from './api/documents/index'
  15. import { AuthContext } from './types/context'
  16. import { AppDataSource } from './data-source'
  17. import { Hono } from 'hono'
  18. const app = new Hono();
  19. const api = new OpenAPIHono<AuthContext>()
  20. api.onError(errorHandler)
  21. // Rate limiting
  22. api.use('/api/v1/*', async (c, next) => {
  23. const ip = c.req.header('x-forwarded-for') || c.req.header('cf-connecting-ip')
  24. // 实现速率限制逻辑
  25. await next()
  26. })
  27. // 数据库初始化中间件
  28. api.use('/api/v1/*', async (c, next) => {
  29. if(!AppDataSource.isInitialized) {
  30. await AppDataSource.initialize();
  31. }
  32. await next();
  33. })
  34. // 注册Bearer认证方案
  35. api.openAPIRegistry.registerComponent('securitySchemes','bearerAuth',{
  36. type:'http',
  37. scheme:'bearer',
  38. bearerFormat:'JWT',
  39. description:'使用JWT进行认证'
  40. })
  41. // OpenAPI documentation endpoint
  42. // !import.meta.env.PROD
  43. if(1){
  44. // api.doc31('/doc', {
  45. // openapi: '3.1.0',
  46. // info: {
  47. // title: 'API Documentation',
  48. // version: '1.0.0'
  49. // },
  50. // security: [{
  51. // bearerAuth: []
  52. // }]
  53. // // servers: [{ url: '/api/v1' }]
  54. // })
  55. api.get('/doc', (c) => {
  56. const config = {
  57. openapi: '3.1.0',
  58. info: {
  59. title: 'API Documentation',
  60. version: '1.0.0'
  61. },
  62. security: [{
  63. bearerAuth: []
  64. }]
  65. // servers: [{ url: '/api/v1' }]
  66. };
  67. try {
  68. const document = api.getOpenAPI31Document(config);
  69. return c.json(document);
  70. } catch (e: any) {
  71. return c.json({
  72. error: 'OpenAPI document generation failed',
  73. message: e.message || 'Unknown error',
  74. stack: e.stack
  75. }, 500)
  76. }
  77. })
  78. app.get('/ui', swaggerUI({
  79. url: '/doc',
  80. persistAuthorization: true,
  81. manuallySwaggerUIHtml: (asset) => `
  82. <div>
  83. <div id="swagger-ui"></div>
  84. <link rel="stylesheet" href="https://ai-oss.d8d.fun/swagger-ui-dist/swagger-ui.css" />
  85. <script src="https://ai-oss.d8d.fun/swagger-ui-dist/swagger-ui-bundle.js" crossorigin="anonymous"></script>
  86. <script>
  87. window.onload = () => {
  88. window.ui = SwaggerUIBundle({
  89. dom_id: '#swagger-ui',
  90. url: '/doc',
  91. persistAuthorization: true
  92. })
  93. }
  94. </script>
  95. </div>
  96. `
  97. }))
  98. }
  99. const userRoutes = api.route('/api/v1/users', usersRouter)
  100. const authRoutes = api.route('/api/v1/auth', authRoute)
  101. const roleRoutes = api.route('/api/v1/roles', rolesRoute)
  102. const fileRoutes = api.route('/api/v1/files', fileRoute)
  103. const membershipPlanRoutes = api.route('/api/v1/membership-plans', membershipPlanRoute)
  104. const paymentRoutes = api.route('/api/v1/payments', paymentRoute)
  105. const templateRoutes = api.route('/api/v1/templates', templateRoute)
  106. const publicTemplateRoutes = api.route('/api/v1/public/templates', publicTemplateRoute)
  107. const publicSettingsRoutes = api.route('/api/v1/public/settings', publicSettingsRoute)
  108. const settingsRoutes = api.route('/api/v1/settings', settingsRoute)
  109. const documentsRoutes = api.route('/api/v1/documents', documentsRoute)
  110. export type AuthRoutes = typeof authRoutes
  111. export type UserRoutes = typeof userRoutes
  112. export type RoleRoutes = typeof roleRoutes
  113. export type FileRoutes = typeof fileRoutes
  114. export type MembershipPlanRoutes = typeof membershipPlanRoutes
  115. export type PaymentRoutes = typeof paymentRoutes
  116. export type TemplateRoutes = typeof templateRoutes
  117. export type PublicTemplateRoutes = typeof publicTemplateRoutes
  118. export type PublicSettingsRoutes = typeof publicSettingsRoutes
  119. export type SettingsRoutes = typeof settingsRoutes
  120. export type DocumentsRoutes = typeof documentsRoutes
  121. app.route('/', api)
  122. export default app