2
0

api.ts 3.4 KB

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