service.py 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374
  1. # -*- coding: utf-8 -*-
  2. import json
  3. import uuid
  4. from typing import NewType
  5. from fastapi import Request
  6. from redis.asyncio.client import Redis
  7. from sqlalchemy.ext.asyncio import AsyncSession
  8. from datetime import datetime, timedelta
  9. from user_agents import parse
  10. from app.common.enums import RedisInitKeyConfig
  11. from app.utils.common_util import get_random_character
  12. from app.utils.captcha_util import CaptchaUtil
  13. from app.utils.ip_local_util import IpLocalUtil
  14. from app.utils.hash_bcrpy_util import PwdUtil
  15. from app.core.redis_crud import RedisCURD
  16. from app.core.exceptions import CustomException
  17. from app.core.logger import log
  18. from app.config.setting import settings
  19. from app.core.security import (
  20. CustomOAuth2PasswordRequestForm,
  21. create_access_token,
  22. decode_access_token
  23. )
  24. from app.api.v1.module_monitor.online.schema import OnlineOutSchema
  25. from ..user.crud import UserCRUD
  26. from ..user.model import UserModel
  27. from .schema import (
  28. JWTPayloadSchema,
  29. JWTOutSchema,
  30. AuthSchema,
  31. CaptchaOutSchema,
  32. LogoutPayloadSchema,
  33. RefreshTokenPayloadSchema
  34. )
  35. CaptchaKey = NewType('CaptchaKey', str)
  36. CaptchaBase64 = NewType('CaptchaBase64', str)
  37. class LoginService:
  38. """登录认证服务"""
  39. @classmethod
  40. async def authenticate_user_service(cls, request: Request, redis: Redis, login_form: CustomOAuth2PasswordRequestForm, db: AsyncSession) -> JWTOutSchema:
  41. """
  42. 用户认证
  43. 参数:
  44. - request (Request): FastAPI请求对象
  45. - login_form (CustomOAuth2PasswordRequestForm): 登录表单数据
  46. - db (AsyncSession): 数据库会话对象
  47. 返回:
  48. - JWTOutSchema: 包含访问令牌和刷新令牌的响应模型
  49. 异常:
  50. - CustomException: 认证失败时抛出异常。
  51. """
  52. # 判断是否来自API文档
  53. referer = request.headers.get('referer', '')
  54. request_from_docs = referer.endswith(('docs', 'redoc'))
  55. # 验证码校验
  56. if settings.CAPTCHA_ENABLE and not request_from_docs:
  57. if not login_form.captcha_key or not login_form.captcha:
  58. raise CustomException(msg="验证码不能为空")
  59. await CaptchaService.check_captcha_service(redis=redis, key=login_form.captcha_key, captcha=login_form.captcha)
  60. # 用户认证
  61. auth = AuthSchema(db=db)
  62. user = await UserCRUD(auth).get_by_username_crud(username=login_form.username)
  63. if not user:
  64. raise CustomException(msg="用户不存在")
  65. if not PwdUtil.verify_password(plain_password=login_form.password, password_hash=user.password):
  66. raise CustomException(msg="账号或密码错误")
  67. if not user.status:
  68. raise CustomException(msg="用户已被停用")
  69. # 更新最后登录时间
  70. user = await UserCRUD(auth).update_last_login_crud(id=user.id)
  71. if not user:
  72. raise CustomException(msg="用户不存在")
  73. if not login_form.login_type:
  74. raise CustomException(msg="登录类型不能为空")
  75. # 创建token
  76. token = await cls.create_token_service(request=request, redis=redis, user=user, login_type=login_form.login_type)
  77. return token
  78. @classmethod
  79. async def create_token_service(cls, request: Request, redis: Redis, user: UserModel, login_type: str) -> JWTOutSchema:
  80. """
  81. 创建访问令牌和刷新令牌
  82. 参数:
  83. - request (Request): FastAPI请求对象
  84. - redis (Redis): Redis客户端对象
  85. - user (UserModel): 用户模型对象
  86. - login_type (str): 登录类型
  87. 返回:
  88. - JWTOutSchema: 包含访问令牌和刷新令牌的响应模型
  89. 异常:
  90. - CustomException: 创建令牌失败时抛出异常。
  91. """
  92. # 生成会话编号
  93. session_id = str(uuid.uuid4())
  94. request.scope["session_id"] = session_id
  95. user_agent = parse(request.headers.get("user-agent"))
  96. request_ip = None
  97. x_forwarded_for = request.headers.get('X-Forwarded-For')
  98. if x_forwarded_for:
  99. # 取第一个 IP 地址,通常为客户端真实 IP
  100. request_ip = x_forwarded_for.split(',')[0].strip()
  101. else:
  102. # 若没有 X-Forwarded-For 头,则使用 request.client.host
  103. if request.client:
  104. request_ip = request.client.host
  105. else:
  106. request_ip = "127.0.0.1"
  107. login_location = await IpLocalUtil.get_ip_location(request_ip)
  108. request.scope["login_location"] = login_location
  109. # 确保在请求上下文中设置用户名和会话ID
  110. request.scope["user_username"] = user.username
  111. access_expires = timedelta(minutes=settings.ACCESS_TOKEN_EXPIRE_MINUTES)
  112. refresh_expires = timedelta(minutes=settings.REFRESH_TOKEN_EXPIRE_MINUTES)
  113. now = datetime.now()
  114. # 记录租户信息到日志
  115. log.info(f"用户ID: {user.id}, 用户名: {user.username} 正在生成JWT令牌")
  116. # 生成会话信息
  117. session_info=OnlineOutSchema(
  118. session_id=session_id,
  119. user_id=user.id,
  120. name=user.name,
  121. user_name=user.username,
  122. ipaddr=request_ip,
  123. login_location=login_location,
  124. os=user_agent.os.family,
  125. browser = user_agent.browser.family,
  126. login_time=user.last_login,
  127. login_type=login_type
  128. ).model_dump_json()
  129. access_token = create_access_token(payload=JWTPayloadSchema(
  130. sub=session_info,
  131. is_refresh=False,
  132. exp=now + access_expires,
  133. ))
  134. refresh_token = create_access_token(payload=JWTPayloadSchema(
  135. sub=session_info,
  136. is_refresh=True,
  137. exp=now + refresh_expires,
  138. ))
  139. # 设置新的token
  140. await RedisCURD(redis).set(
  141. key=f'{RedisInitKeyConfig.ACCESS_TOKEN.key}:{session_id}',
  142. value=access_token,
  143. expire=int(access_expires.total_seconds())
  144. )
  145. await RedisCURD(redis).set(
  146. key=f'{RedisInitKeyConfig.REFRESH_TOKEN.key}:{session_id}',
  147. value=refresh_token,
  148. expire=int(refresh_expires.total_seconds())
  149. )
  150. return JWTOutSchema(
  151. access_token=access_token,
  152. refresh_token=refresh_token,
  153. expires_in=int(access_expires.total_seconds()),
  154. token_type=settings.TOKEN_TYPE
  155. )
  156. @classmethod
  157. async def refresh_token_service(cls, db: AsyncSession, redis: Redis, request: Request, refresh_token: RefreshTokenPayloadSchema) -> JWTOutSchema:
  158. """
  159. 刷新访问令牌
  160. 参数:
  161. - db (AsyncSession): 数据库会话对象
  162. - redis (Redis): Redis客户端对象
  163. - request (Request): FastAPI请求对象
  164. - refresh_token (RefreshTokenPayloadSchema): 刷新令牌数据
  165. 返回:
  166. - JWTOutSchema: 新的令牌对象
  167. 异常:
  168. - CustomException: 刷新令牌无效时抛出异常
  169. """
  170. token_payload: JWTPayloadSchema = decode_access_token(token = refresh_token.refresh_token)
  171. if not token_payload.is_refresh:
  172. raise CustomException(msg="非法凭证,请传入刷新令牌")
  173. # 去 Redis 查完整信息
  174. session_info = json.loads(token_payload.sub)
  175. session_id = session_info.get("session_id")
  176. user_id = session_info.get("user_id")
  177. if not session_id or not user_id:
  178. raise CustomException(msg="非法凭证,无法获取会话编号或用户ID")
  179. # 用户认证
  180. auth = AuthSchema(db=db)
  181. user = await UserCRUD(auth).get_by_id_crud(id=user_id)
  182. if not user:
  183. raise CustomException(msg="刷新token失败,用户不存在")
  184. # 记录刷新令牌时的租户信息
  185. log.info(f"用户ID: {user.id}, 用户名: {user.username} 正在刷新JWT令牌")
  186. # 设置新的 token
  187. access_expires = timedelta(minutes=settings.ACCESS_TOKEN_EXPIRE_MINUTES)
  188. refresh_expires = timedelta(minutes=settings.REFRESH_TOKEN_EXPIRE_MINUTES)
  189. now = datetime.now()
  190. session_info_json = json.dumps(session_info)
  191. access_token = create_access_token(payload=JWTPayloadSchema(
  192. sub=session_info_json,
  193. is_refresh=False,
  194. exp=now + access_expires
  195. ))
  196. refresh_token_new = create_access_token(payload=JWTPayloadSchema(
  197. sub=session_info_json,
  198. is_refresh=True,
  199. exp=now + refresh_expires
  200. ))
  201. # 覆盖写入 Redis
  202. await RedisCURD(redis).set(
  203. key=f'{RedisInitKeyConfig.ACCESS_TOKEN.key}:{session_id}',
  204. value=access_token,
  205. expire=int(access_expires.total_seconds())
  206. )
  207. await RedisCURD(redis).set(
  208. key=f'{RedisInitKeyConfig.REFRESH_TOKEN.key}:{session_id}',
  209. value=refresh_token_new,
  210. expire=int(refresh_expires.total_seconds())
  211. )
  212. return JWTOutSchema(
  213. access_token=access_token,
  214. refresh_token=refresh_token_new,
  215. token_type=settings.TOKEN_TYPE,
  216. expires_in=int(access_expires.total_seconds())
  217. )
  218. @classmethod
  219. async def logout_service(cls, redis: Redis, token: LogoutPayloadSchema) -> bool:
  220. """
  221. 退出登录
  222. 参数:
  223. - redis (Redis): Redis客户端对象
  224. - token (LogoutPayloadSchema): 退出登录令牌数据
  225. 返回:
  226. - bool: 退出成功返回True
  227. 异常:
  228. - CustomException: 令牌无效时抛出异常
  229. """
  230. payload: JWTPayloadSchema = decode_access_token(token=token.token)
  231. session_info = json.loads(payload.sub)
  232. session_id = session_info.get("session_id")
  233. if not session_id:
  234. raise CustomException(msg="非法凭证,无法获取会话编号")
  235. # 删除Redis中的在线用户、访问令牌、刷新令牌
  236. await RedisCURD(redis).delete(f"{RedisInitKeyConfig.ACCESS_TOKEN.key}:{session_id}")
  237. await RedisCURD(redis).delete(f"{RedisInitKeyConfig.REFRESH_TOKEN.key}:{session_id}")
  238. log.info(f"用户退出登录成功,会话编号:{session_id}")
  239. return True
  240. class CaptchaService:
  241. """验证码服务"""
  242. @classmethod
  243. async def get_captcha_service(cls, redis: Redis) -> dict[str, CaptchaKey | CaptchaBase64]:
  244. """
  245. 获取验证码
  246. 参数:
  247. - redis (Redis): Redis客户端对象
  248. 返回:
  249. - dict[str, CaptchaKey | CaptchaBase64]: 包含验证码key和base64图片的字典
  250. 异常:
  251. - CustomException: 验证码服务未启用时抛出异常
  252. """
  253. if not settings.CAPTCHA_ENABLE:
  254. raise CustomException(msg="未开启验证码服务")
  255. # 生成验证码图片和值
  256. captcha_base64, captcha_value = CaptchaUtil.captcha_arithmetic()
  257. captcha_key = get_random_character()
  258. # 保存到Redis并设置过期时间
  259. redis_key = f"{RedisInitKeyConfig.CAPTCHA_CODES.key}:{captcha_key}"
  260. await RedisCURD(redis).set(
  261. key=redis_key,
  262. value=captcha_value,
  263. expire=settings.CAPTCHA_EXPIRE_SECONDS
  264. )
  265. log.info(f"生成验证码成功,验证码:{captcha_value}")
  266. # 返回验证码信息
  267. return CaptchaOutSchema(
  268. enable=settings.CAPTCHA_ENABLE,
  269. key=CaptchaKey(captcha_key),
  270. img_base=CaptchaBase64(f"data:image/png;base64,{captcha_base64}")
  271. ).model_dump()
  272. @classmethod
  273. async def check_captcha_service(cls, redis: Redis, key: str, captcha: str) -> bool:
  274. """
  275. 校验验证码
  276. 参数:
  277. - redis (Redis): Redis客户端对象
  278. - key (str): 验证码key
  279. - captcha (str): 用户输入的验证码
  280. 返回:
  281. - bool: 验证通过返回True
  282. 异常:
  283. - CustomException: 验证码无效或错误时抛出异常
  284. """
  285. if not captcha:
  286. raise CustomException(msg="验证码不能为空")
  287. # 获取Redis中存储的验证码
  288. redis_key = f'{RedisInitKeyConfig.CAPTCHA_CODES.key}:{key}'
  289. captcha_value = await RedisCURD(redis).get(redis_key)
  290. if not captcha_value:
  291. log.error('验证码已过期或不存在')
  292. raise CustomException(msg="验证码已过期")
  293. # 验证码不区分大小写比对
  294. if captcha.lower() != captcha_value.lower():
  295. log.error(f'验证码错误,用户输入:{captcha},正确值:{captcha_value}')
  296. raise CustomException(msg="验证码错误")
  297. # 验证成功后删除验证码,避免重复使用
  298. await RedisCURD(redis).delete(redis_key)
  299. log.info(f'验证码校验成功,key:{key}')
  300. return True