api.ts 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143
  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 advertisementRoutes from './api/advertisements/index'
  9. import advertisementTypeRoutes from './api/advertisement-types/index'
  10. import goodsCategoryRoutes from './api/goods-categories/index'
  11. import goodsRoutes from './api/goods/index'
  12. import cityRoutes from './api/cities/index'
  13. import configRoutes from './api/configs/index'
  14. import expressCompanyRoutes from './api/express-companies/index'
  15. import organizationRoutes from './api/organizations/index'
  16. import supplierRoutes from './api/suppliers/index'
  17. import cardRoutes from './api/cards/index'
  18. import { AuthContext } from './types/context'
  19. import { AppDataSource } from './data-source'
  20. import { Hono } from 'hono'
  21. const app = new Hono();
  22. const api = new OpenAPIHono<AuthContext>()
  23. api.onError(errorHandler)
  24. // Rate limiting
  25. api.use('/api/v1/*', async (c, next) => {
  26. const ip = c.req.header('x-forwarded-for') || c.req.header('cf-connecting-ip')
  27. // 实现速率限制逻辑
  28. await next()
  29. })
  30. // 数据库初始化中间件
  31. api.use('/api/v1/*', async (c, next) => {
  32. if(!AppDataSource.isInitialized) {
  33. await AppDataSource.initialize();
  34. }
  35. await next();
  36. })
  37. // 注册Bearer认证方案
  38. api.openAPIRegistry.registerComponent('securitySchemes','bearerAuth',{
  39. type:'http',
  40. scheme:'bearer',
  41. bearerFormat:'JWT',
  42. description:'使用JWT进行认证'
  43. })
  44. // OpenAPI documentation endpoint
  45. // !import.meta.env.PROD
  46. if(1){
  47. // api.doc31('/doc', {
  48. // openapi: '3.1.0',
  49. // info: {
  50. // title: 'API Documentation',
  51. // version: '1.0.0'
  52. // },
  53. // security: [{
  54. // bearerAuth: []
  55. // }]
  56. // // servers: [{ url: '/api/v1' }]
  57. // })
  58. api.get('/doc', (c) => {
  59. const config = {
  60. openapi: '3.1.0',
  61. info: {
  62. title: 'API Documentation',
  63. version: '1.0.0'
  64. },
  65. security: [{
  66. bearerAuth: []
  67. }]
  68. // servers: [{ url: '/api/v1' }]
  69. };
  70. try {
  71. const document = api.getOpenAPI31Document(config);
  72. return c.json(document);
  73. } catch (e: any) {
  74. return c.json({
  75. error: 'OpenAPI document generation failed',
  76. message: e.message || 'Unknown error',
  77. stack: e.stack
  78. }, 500)
  79. }
  80. })
  81. app.get('/ui', swaggerUI({
  82. url: '/doc',
  83. persistAuthorization: true,
  84. manuallySwaggerUIHtml: (asset) => `
  85. <div>
  86. <div id="swagger-ui"></div>
  87. <link rel="stylesheet" href="https://ai-oss.d8d.fun/swagger-ui-dist/swagger-ui.css" />
  88. <script src="https://ai-oss.d8d.fun/swagger-ui-dist/swagger-ui-bundle.js" crossorigin="anonymous"></script>
  89. <script>
  90. window.onload = () => {
  91. window.ui = SwaggerUIBundle({
  92. dom_id: '#swagger-ui',
  93. url: '/doc',
  94. persistAuthorization: true
  95. })
  96. }
  97. </script>
  98. </div>
  99. `
  100. }))
  101. }
  102. const userRoutes = api.route('/api/v1/users', usersRouter)
  103. const authRoutes = api.route('/api/v1/auth', authRoute)
  104. const roleRoutes = api.route('/api/v1/roles', rolesRoute)
  105. const fileApiRoutes = api.route('/api/v1/files', fileRoutes)
  106. const advertisementApiRoutes = api.route('/api/v1/advertisements', advertisementRoutes)
  107. const advertisementTypeApiRoutes = api.route('/api/v1/advertisement-types', advertisementTypeRoutes)
  108. const goodsCategoryApiRoutes = api.route('/api/v1/goods-categories', goodsCategoryRoutes)
  109. const goodsApiRoutes = api.route('/api/v1/goods', goodsRoutes)
  110. const cityApiRoutes = api.route('/api/v1/cities', cityRoutes)
  111. const configApiRoutes = api.route('/api/v1/configs', configRoutes)
  112. const expressCompanyApiRoutes = api.route('/api/v1/express-companies', expressCompanyRoutes)
  113. const organizationApiRoutes = api.route('/api/v1/organizations', organizationRoutes)
  114. const supplierApiRoutes = api.route('/api/v1/suppliers', supplierRoutes)
  115. const cardApiRoutes = api.route('/api/v1/cards', cardRoutes)
  116. export type AuthRoutes = typeof authRoutes
  117. export type UserRoutes = typeof userRoutes
  118. export type RoleRoutes = typeof roleRoutes
  119. export type FileRoutes = typeof fileApiRoutes
  120. export type AdvertisementRoutes = typeof advertisementApiRoutes
  121. export type AdvertisementTypeRoutes = typeof advertisementTypeApiRoutes
  122. export type GoodsCategoryRoutes = typeof goodsCategoryApiRoutes
  123. export type GoodsRoutes = typeof goodsApiRoutes
  124. export type CityRoutes = typeof cityApiRoutes
  125. export type ConfigRoutes = typeof configApiRoutes
  126. export type ExpressCompanyRoutes = typeof expressCompanyApiRoutes
  127. export type OrganizationRoutes = typeof organizationApiRoutes
  128. export type SupplierRoutes = typeof supplierApiRoutes
  129. export type CardRoutes = typeof cardApiRoutes
  130. app.route('/', api)
  131. export default app