controller.py 7.4 KB

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