api.ts 4.2 KB

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