auth.schema.mt.ts 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. import { z } from '@hono/zod-openapi';
  2. export const LoginSchema = z.object({
  3. username: z.string().min(3).openapi({
  4. example: 'admin',
  5. description: '用户名'
  6. }),
  7. password: z.string().min(6).openapi({
  8. example: 'admin123',
  9. description: '密码'
  10. })
  11. });
  12. export const RegisterSchema = z.object({
  13. username: z.string().min(3).openapi({
  14. example: 'john_doe',
  15. description: '用户名'
  16. }),
  17. password: z.string().min(6).openapi({
  18. example: 'password123',
  19. description: '密码'
  20. }),
  21. email: z.string().email().openapi({
  22. example: 'john@example.com',
  23. description: '邮箱'
  24. }).optional()
  25. });
  26. export const MiniLoginSchema = z.object({
  27. code: z.string().openapi({
  28. example: '08123456789012345678901234567890',
  29. description: '小程序登录code'
  30. }),
  31. userInfo: z.object({
  32. nickName: z.string().optional(),
  33. avatarUrl: z.string().optional()
  34. }).optional()
  35. });
  36. export const UserResponseSchema = z.object({
  37. id: z.number().int().positive().openapi({ description: '用户ID' }),
  38. tenantId: z.number().int().positive().openapi({ description: '租户ID' }),
  39. username: z.string().min(3, '用户名至少3个字符').max(255, '用户名最多255个字符').openapi({
  40. example: 'admin',
  41. description: '用户名,3-255个字符'
  42. }),
  43. phone: z.string().max(255, '手机号最多255个字符').nullable().openapi({
  44. example: '13800138000',
  45. description: '手机号'
  46. }),
  47. email: z.string().email('请输入正确的邮箱格式').max(255, '邮箱最多255个字符').nullable().openapi({
  48. example: 'user@example.com',
  49. description: '邮箱'
  50. }),
  51. nickname: z.string().max(255, '昵称最多255个字符').nullable().openapi({
  52. example: '昵称',
  53. description: '用户昵称'
  54. }),
  55. name: z.string().max(255, '姓名最多255个字符').nullable().openapi({
  56. example: '张三',
  57. description: '真实姓名'
  58. }),
  59. avatarFileId: z.number().int().positive().nullable().openapi({
  60. example: 1,
  61. description: '头像文件ID'
  62. }),
  63. avatarFile: z.object({
  64. id: z.number().int().positive().openapi({ description: '文件ID' }),
  65. name: z.string().max(255).openapi({ description: '文件名', example: 'avatar.jpg' }),
  66. fullUrl: z.string().openapi({ description: '文件完整URL', example: 'https://example.com/avatar.jpg' }),
  67. type: z.string().nullable().openapi({ description: '文件类型', example: 'image/jpeg' }),
  68. size: z.number().nullable().openapi({ description: '文件大小(字节)', example: 102400 })
  69. }).nullable().optional().openapi({
  70. description: '头像文件信息'
  71. }),
  72. openid: z.string().max(255).nullable().optional().openapi({
  73. example: 'oABCDEFGH123456789',
  74. description: '微信小程序openid'
  75. }),
  76. unionid: z.string().max(255).nullable().optional().openapi({
  77. example: 'unionid123456789',
  78. description: '微信unionid'
  79. }),
  80. registrationSource: z.string().max(20).default('web').openapi({
  81. example: 'miniapp',
  82. description: '注册来源: web, miniapp'
  83. }),
  84. isDisabled: z.number().int().min(0).max(1).default(0).openapi({
  85. example: 0,
  86. description: '是否禁用(0:启用,1:禁用)'
  87. }),
  88. isDeleted: z.number().int().min(0).max(1).default(0).openapi({
  89. example: 0,
  90. description: '是否删除(0:未删除,1:已删除)'
  91. }),
  92. roles: z.array(z.object({
  93. id: z.number().int().positive().openapi({ description: '角色ID' }),
  94. tenantId: z.number().int().positive().openapi({ description: '租户ID' }),
  95. name: z.string().max(255).openapi({ description: '角色名称', example: 'admin' }),
  96. description: z.string().max(255).nullable().openapi({ description: '角色描述', example: '管理员' }),
  97. permissions: z.array(z.string()).openapi({ description: '权限列表', example: ['user:create'] }),
  98. createdAt: z.coerce.date().openapi({ description: '创建时间' }),
  99. updatedAt: z.coerce.date().openapi({ description: '更新时间' })
  100. })).optional().openapi({
  101. example: [
  102. {
  103. id: 1,
  104. tenantId: 1,
  105. name: 'admin',
  106. description: '管理员',
  107. permissions: ['user:create'],
  108. createdAt: new Date(),
  109. updatedAt: new Date()
  110. }
  111. ],
  112. description: '用户角色列表'
  113. }),
  114. createdAt: z.coerce.date().openapi({ description: '创建时间' }),
  115. updatedAt: z.coerce.date().openapi({ description: '更新时间' })
  116. });
  117. export const TokenResponseSchema = z.object({
  118. token: z.string().openapi({
  119. example: 'jwt.token.here',
  120. description: 'JWT Token'
  121. }),
  122. user: UserResponseSchema
  123. });
  124. export const MiniLoginResponseSchema = z.object({
  125. token: z.string().openapi({
  126. example: 'jwt.token.here',
  127. description: 'JWT Token'
  128. }),
  129. user: z.object({
  130. id: z.number(),
  131. username: z.string(),
  132. nickname: z.string().nullable(),
  133. phone: z.string().nullable(),
  134. email: z.string().nullable(),
  135. avatarFileId: z.number().nullable(),
  136. registrationSource: z.string()
  137. }),
  138. isNewUser: z.boolean().openapi({
  139. example: true,
  140. description: '是否为新注册用户'
  141. })
  142. });
  143. export const SuccessSchema = z.object({
  144. message: z.string().openapi({ example: '登出成功' })
  145. });
  146. export const PhoneDecryptSchema = z.object({
  147. encryptedData: z.string().openapi({
  148. example: 'encrypted_phone_data_here',
  149. description: '微信小程序加密的手机号数据'
  150. }),
  151. iv: z.string().openapi({
  152. example: 'encryption_iv_here',
  153. description: '加密算法的初始向量'
  154. })
  155. });
  156. export const PhoneDecryptResponseSchema = z.object({
  157. phoneNumber: z.string().openapi({
  158. example: '13800138000',
  159. description: '解密后的手机号'
  160. }),
  161. user: z.object({
  162. id: z.number(),
  163. username: z.string(),
  164. nickname: z.string().nullable(),
  165. phone: z.string().nullable(),
  166. email: z.string().nullable(),
  167. avatarFileId: z.number().nullable(),
  168. registrationSource: z.string()
  169. })
  170. });