controller.py 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286
  1. from typing import Annotated
  2. from fastapi import APIRouter, Depends, Request
  3. from fastapi.responses import JSONResponse
  4. from redis.asyncio.client import Redis
  5. from sqlalchemy.ext.asyncio import AsyncSession
  6. from app.common.response import ErrorResponse, SuccessResponse
  7. from app.config.setting import settings
  8. from app.core.dependencies import db_getter, get_current_user, redis_getter
  9. from app.core.logger import log
  10. from app.core.router_class import OperationLogRoute
  11. from app.core.security import CustomOAuth2PasswordRequestForm
  12. from .schema import (
  13. AutoLoginTokenSchema,
  14. AutoLoginUserSchema,
  15. CaptchaOutSchema,
  16. JWTOutSchema,
  17. LoginMiniRequestSchema,
  18. LogoutPayloadSchema,
  19. RefreshTokenPayloadSchema, SmsCodeSchema,
  20. )
  21. from .service import AutoLoginService, CaptchaService, LoginService, SmsCodeService
  22. AuthRouter = APIRouter(route_class=OperationLogRoute, prefix="/auth", tags=["认证授权"])
  23. @AuthRouter.post(
  24. '/sms-code',
  25. summary="发送短信验证码",
  26. description="发送短信验证码",
  27. )
  28. async def sms_code_controller(
  29. sms_code: SmsCodeSchema,
  30. redis: Annotated[Redis, Depends(redis_getter)],
  31. ) -> JSONResponse:
  32. """
  33. 发送短信验证码
  34. 参数:
  35. - smsCode (SmsCodeSchema): 短信验证码请求模型
  36. - redis (Redis): Redis 客户端对象
  37. - db (AsyncSession): 数据库会话对象
  38. 返回:
  39. - JSONResponse | dict: 包含短信验证码的响应模型
  40. 异常:
  41. - CustomException: 验证码发送失败时抛出异常。
  42. """
  43. await SmsCodeService.send_sms_code_service(
  44. sms_code=sms_code, redis=redis
  45. )
  46. return SuccessResponse(data={}, msg="短信验证码发送成功")
  47. @AuthRouter.post(
  48. "/login/mini",
  49. summary="小程序登录",
  50. description="小程序登录",
  51. response_model=JWTOutSchema,
  52. )
  53. async def login_mini_controller(
  54. request: Request,
  55. redis: Annotated[Redis, Depends(redis_getter)],
  56. login_form: LoginMiniRequestSchema,
  57. db: Annotated[AsyncSession, Depends(db_getter)],
  58. ) -> JSONResponse | dict:
  59. login_token = await LoginService.authenticate_mini_user_service(
  60. request=request, redis=redis, login_form=login_form, db=db
  61. )
  62. return SuccessResponse(data=login_token.model_dump(), msg="登录成功")
  63. @AuthRouter.post(
  64. "/login",
  65. summary="登录",
  66. description="登录",
  67. response_model=JWTOutSchema,
  68. )
  69. async def login_for_access_token_controller(
  70. request: Request,
  71. redis: Annotated[Redis, Depends(redis_getter)],
  72. login_form: Annotated[CustomOAuth2PasswordRequestForm, Depends()],
  73. db: Annotated[AsyncSession, Depends(db_getter)],
  74. ) -> JSONResponse | dict:
  75. """
  76. 用户登录
  77. 参数:
  78. - request (Request): FastAPI请求对象
  79. - redis (Redis): Redis 客户端对象
  80. - login_form (CustomOAuth2PasswordRequestForm): 登录表单数据
  81. - db (AsyncSession): 数据库会话对象
  82. 返回:
  83. - JWTOutSchema: 包含访问令牌和刷新令牌的响应模型
  84. 异常:
  85. - CustomException: 认证失败时抛出异常。
  86. """
  87. login_token = await LoginService.authenticate_user_service(
  88. request=request, redis=redis, login_form=login_form, db=db
  89. )
  90. log.info(f"用户{login_form.username}登录成功")
  91. # 如果是文档请求,则不记录日志:http://localhost:8000/api/v1/docs
  92. if settings.DOCS_URL in request.headers.get("referer", ""):
  93. return login_token.model_dump()
  94. return SuccessResponse(data=login_token.model_dump(), msg="登录成功")
  95. @AuthRouter.post(
  96. "/token/refresh",
  97. summary="刷新token",
  98. description="刷新token",
  99. response_model=JWTOutSchema,
  100. dependencies=[Depends(get_current_user)],
  101. )
  102. async def get_new_token_controller(
  103. request: Request,
  104. payload: RefreshTokenPayloadSchema,
  105. db: Annotated[AsyncSession, Depends(db_getter)],
  106. redis: Annotated[Redis, Depends(redis_getter)],
  107. ) -> JSONResponse:
  108. """
  109. 刷新token
  110. 参数:
  111. - request (Request): FastAPI请求对象
  112. - payload (RefreshTokenPayloadSchema): 刷新令牌负载模型
  113. - db (AsyncSession): 数据库会话对象
  114. - redis (Redis): Redis 客户端对象
  115. 返回:
  116. - JWTOutSchema: 包含新的访问令牌和刷新令牌的响应模型
  117. 异常:
  118. - CustomException: 刷新令牌失败时抛出异常。
  119. """
  120. # 解析当前的访问Token以获取用户名
  121. new_token = await LoginService.refresh_token_service(
  122. db=db, request=request, redis=redis, refresh_token=payload
  123. )
  124. token_dict = new_token.model_dump()
  125. log.info(f"刷新token成功: {token_dict}")
  126. return SuccessResponse(data=token_dict, msg="刷新成功")
  127. @AuthRouter.get(
  128. "/captcha/get",
  129. summary="获取验证码",
  130. description="获取登录验证码",
  131. response_model=CaptchaOutSchema,
  132. )
  133. async def get_captcha_for_login_controller(
  134. redis: Annotated[Redis, Depends(redis_getter)],
  135. ) -> JSONResponse:
  136. """
  137. 获取登录验证码
  138. 参数:
  139. - redis (Redis): Redis客户端对象
  140. 返回:
  141. - CaptchaOutSchema: 包含验证码图片和key的响应模型
  142. 异常:
  143. - CustomException: 获取验证码失败时抛出异常。
  144. """
  145. # 获取验证码
  146. captcha = await CaptchaService.get_captcha_service(redis=redis)
  147. log.info("获取验证码成功")
  148. return SuccessResponse(data=captcha, msg="获取验证码成功")
  149. @AuthRouter.post(
  150. "/logout",
  151. summary="退出登录",
  152. description="退出登录",
  153. dependencies=[Depends(get_current_user)],
  154. )
  155. async def logout_controller(
  156. payload: LogoutPayloadSchema,
  157. redis: Annotated[Redis, Depends(redis_getter)],
  158. ) -> JSONResponse:
  159. """
  160. 退出登录
  161. 参数:
  162. - payload (LogoutPayloadSchema): 退出登录负载模型
  163. - redis (Redis): Redis客户端对象
  164. 返回:
  165. - JSONResponse: 包含退出登录结果的响应模型
  166. 异常:
  167. - CustomException: 退出登录失败时抛出异常。
  168. """
  169. if await LoginService.logout_service(redis=redis, token=payload):
  170. log.info("退出成功")
  171. return SuccessResponse(msg="退出成功")
  172. return ErrorResponse(msg="退出失败")
  173. @AuthRouter.get(
  174. "/auto-login/users",
  175. summary="获取免登录用户列表",
  176. description="获取可用于免登录快速登录的用户列表",
  177. response_model=list[AutoLoginUserSchema],
  178. )
  179. async def get_auto_login_users_controller(
  180. db: Annotated[AsyncSession, Depends(db_getter)],
  181. ) -> JSONResponse:
  182. """
  183. 获取免登录用户列表
  184. 参数:
  185. - db (AsyncSession): 数据库会话对象
  186. 返回:
  187. - list[AutoLoginUserSchema]: 免登录用户列表
  188. """
  189. users = await AutoLoginService.get_auto_login_users_service(db=db)
  190. return SuccessResponse(data=users, msg="获取成功")
  191. @AuthRouter.post(
  192. "/auto-login/token",
  193. summary="获取免登录Token",
  194. description="根据用户ID生成免登录Token",
  195. response_model=AutoLoginTokenSchema,
  196. )
  197. async def get_auto_login_token_controller(
  198. redis: Annotated[Redis, Depends(redis_getter)],
  199. db: Annotated[AsyncSession, Depends(db_getter)],
  200. user_id: int,
  201. ) -> JSONResponse:
  202. """
  203. 获取免登录Token
  204. 参数:
  205. - redis (Redis): Redis客户端对象
  206. - db (AsyncSession): 数据库会话对象
  207. - user_id (int): 用户ID
  208. 返回:
  209. - AutoLoginTokenSchema: 免登录Token和用户信息
  210. """
  211. result = await AutoLoginService.create_auto_login_token_service(
  212. redis=redis, db=db, user_id=user_id
  213. )
  214. return SuccessResponse(data=result, msg="获取成功")
  215. @AuthRouter.post(
  216. "/auto-login",
  217. summary="免登录",
  218. description="使用免登录Token快速登录",
  219. response_model=JWTOutSchema,
  220. )
  221. async def auto_login_controller(
  222. request: Request,
  223. redis: Annotated[Redis, Depends(redis_getter)],
  224. db: Annotated[AsyncSession, Depends(db_getter)],
  225. token: str,
  226. ) -> JSONResponse:
  227. """
  228. 免登录
  229. 参数:
  230. - request (Request): FastAPI请求对象
  231. - redis (Redis): Redis客户端对象
  232. - db (AsyncSession): 数据库会话对象
  233. - token (str): 免登录Token
  234. 返回:
  235. - JWTOutSchema: JWT令牌信息
  236. """
  237. login_token = await AutoLoginService.auto_login_service(
  238. request=request, redis=redis, db=db, token=token
  239. )
  240. log.info("用户免登录成功")
  241. return SuccessResponse(data=login_token.model_dump(), msg="登录成功")