service.py 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731
  1. import json
  2. import uuid
  3. from datetime import datetime, timedelta
  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 user_agents import parse
  9. from app.api.v1.module_monitor.online.schema import OnlineOutSchema
  10. from app.api.v1.module_system.user.crud import UserCRUD
  11. from app.api.v1.module_system.user.model import UserModel
  12. from app.common.enums import RedisInitKeyConfig
  13. from app.config.setting import settings
  14. from app.core.exceptions import CustomException
  15. from app.core.logger import log
  16. from app.core.redis_crud import RedisCURD
  17. from app.core.security import (
  18. CustomOAuth2PasswordRequestForm,
  19. create_access_token,
  20. decode_access_token,
  21. )
  22. from app.utils.captcha_util import CaptchaUtil
  23. from app.utils.common_util import get_random_character, generate_random_code
  24. from app.utils.hash_bcrpy_util import PwdUtil
  25. from app.utils.ip_local_util import IpLocalUtil
  26. from app.utils.sms_util import SendSmsRequest, SmsTemplateEnum, SmsSender
  27. from .schema import (
  28. AuthSchema,
  29. AutoLoginTokenSchema,
  30. AutoLoginUserSchema,
  31. CaptchaOutSchema,
  32. JWTOutSchema,
  33. JWTPayloadSchema,
  34. LoginMiniRequestSchema,
  35. LogoutPayloadSchema,
  36. RefreshTokenPayloadSchema,
  37. SmsCodeSchema,
  38. SmsLoginRequestSchema,
  39. )
  40. CaptchaKey = NewType("CaptchaKey", str)
  41. CaptchaBase64 = NewType("CaptchaBase64", str)
  42. class LoginService:
  43. """登录认证服务"""
  44. @classmethod
  45. async def authenticate_mini_user_service(
  46. cls,
  47. request: Request,
  48. redis: Redis,
  49. login_form: LoginMiniRequestSchema,
  50. db: AsyncSession,
  51. ) -> JWTOutSchema:
  52. # 小程序用户认证
  53. auth = AuthSchema(db=db)
  54. user = await UserCRUD(auth).get_by_username_crud(username=login_form.username)
  55. if not user:
  56. raise CustomException(msg="用户不存在")
  57. if not PwdUtil.verify_password(
  58. plain_password=login_form.password, password_hash=user.password
  59. ):
  60. raise CustomException(msg="账号或密码错误")
  61. if user.status == "1":
  62. raise CustomException(msg="用户已被停用")
  63. # 更新最后登录时间
  64. user = await UserCRUD(auth).update_last_login_crud(id=user.id)
  65. if not user:
  66. raise CustomException(msg="用户不存在")
  67. if not login_form.login_type:
  68. raise CustomException(msg="登录类型不能为空")
  69. # 创建token
  70. token = await cls.create_token_service(
  71. request=request,
  72. redis=redis,
  73. user=user,
  74. login_type=login_form.login_type,
  75. )
  76. return token
  77. @classmethod
  78. async def authenticate_user_service(
  79. cls,
  80. request: Request,
  81. redis: Redis,
  82. login_form: CustomOAuth2PasswordRequestForm,
  83. db: AsyncSession,
  84. ) -> JWTOutSchema:
  85. """
  86. 用户认证
  87. 参数:
  88. - request (Request): FastAPI请求对象
  89. - login_form (CustomOAuth2PasswordRequestForm): 登录表单数据
  90. - db (AsyncSession): 数据库会话对象
  91. 返回:
  92. - JWTOutSchema: 包含访问令牌和刷新令牌的响应模型
  93. 异常:
  94. - CustomException: 认证失败时抛出异常。
  95. """
  96. # 判断是否来自API文档
  97. referer = request.headers.get("referer", "")
  98. request_from_docs = referer.endswith(("docs", "redoc"))
  99. # 验证码校验
  100. if settings.CAPTCHA_ENABLE and not request_from_docs:
  101. if not login_form.captcha_key or not login_form.captcha:
  102. raise CustomException(msg="验证码不能为空")
  103. await CaptchaService.check_captcha_service(
  104. redis=redis,
  105. key=login_form.captcha_key,
  106. captcha=login_form.captcha,
  107. )
  108. # 用户认证
  109. auth = AuthSchema(db=db)
  110. user = await UserCRUD(auth).get_by_username_crud(username=login_form.username)
  111. if not user:
  112. raise CustomException(msg="用户不存在")
  113. if not PwdUtil.verify_password(
  114. plain_password=login_form.password, password_hash=user.password
  115. ):
  116. raise CustomException(msg="账号或密码错误")
  117. if user.status == "1":
  118. raise CustomException(msg="用户已被停用")
  119. # 更新最后登录时间
  120. user = await UserCRUD(auth).update_last_login_crud(id=user.id)
  121. if not user:
  122. raise CustomException(msg="用户不存在")
  123. if not login_form.login_type:
  124. raise CustomException(msg="登录类型不能为空")
  125. # 创建token
  126. token = await cls.create_token_service(
  127. request=request,
  128. redis=redis,
  129. user=user,
  130. login_type=login_form.login_type,
  131. )
  132. return token
  133. @classmethod
  134. async def authenticate_sms_user_service(
  135. cls,
  136. request: Request,
  137. redis: Redis,
  138. login_form: SmsLoginRequestSchema,
  139. db: AsyncSession,
  140. ) -> JWTOutSchema:
  141. """
  142. 短信验证码登录
  143. 参数:
  144. - request (Request): FastAPI请求对象
  145. - redis (Redis): Redis客户端对象
  146. - login_form (SmsLoginRequestSchema): 短信登录表单数据
  147. - db (AsyncSession): 数据库会话对象
  148. 返回:
  149. - JWTOutSchema: 包含访问令牌和刷新令牌的响应模型
  150. 异常:
  151. - CustomException: 认证失败时抛出异常。
  152. """
  153. mobile = login_form.mobile
  154. code = login_form.code
  155. # 验证验证码
  156. # verify_result = await SmsCodeService.verify_sms_code_service(login_form, redis)
  157. # if not verify_result:
  158. # raise CustomException(msg="验证码已过期或错误")
  159. pass # 测试阶段跳过短信验证码校验
  160. # 根据手机号查找用户
  161. auth = AuthSchema(db=db)
  162. user = await UserCRUD(auth).get_by_mobile_crud(mobile=mobile)
  163. if not user:
  164. raise CustomException(msg="用户不存在")
  165. if user.status == "1":
  166. raise CustomException(msg="用户已被停用")
  167. # 更新最后登录时间
  168. user = await UserCRUD(auth).update_last_login_crud(id=user.id)
  169. if not user:
  170. raise CustomException(msg="用户不存在")
  171. # 创建token
  172. token = await cls.create_token_service(
  173. request=request,
  174. redis=redis,
  175. user=user,
  176. login_type="sms",
  177. )
  178. return token
  179. @classmethod
  180. async def create_token_service(
  181. cls, request: Request, redis: Redis, user: UserModel, login_type: str
  182. ) -> JWTOutSchema:
  183. """
  184. 创建访问令牌和刷新令牌
  185. 参数:
  186. - request (Request): FastAPI请求对象
  187. - redis (Redis): Redis客户端对象
  188. - user (UserModel): 用户模型对象
  189. - login_type (str): 登录类型
  190. 返回:
  191. - JWTOutSchema: 包含访问令牌和刷新令牌的响应模型
  192. 异常:
  193. - CustomException: 创建令牌失败时抛出异常。
  194. """
  195. # 生成会话编号
  196. session_id = str(uuid.uuid4())
  197. request.scope["session_id"] = session_id
  198. user_agent = parse(request.headers.get("user-agent"))
  199. request_ip = None
  200. x_forwarded_for = request.headers.get("X-Forwarded-For")
  201. if x_forwarded_for:
  202. # 取第一个 IP 地址,通常为客户端真实 IP
  203. request_ip = x_forwarded_for.split(",")[0].strip()
  204. else:
  205. # 若没有 X-Forwarded-For 头,则使用 request.client.host
  206. request_ip = request.client.host if request.client else "127.0.0.1"
  207. login_location = await IpLocalUtil.resolve_location_for_log(request_ip)
  208. request.scope["login_location"] = login_location
  209. # 确保在请求上下文中设置用户名和会话ID
  210. request.scope["user_username"] = user.username
  211. access_expires = timedelta(seconds=settings.ACCESS_TOKEN_EXPIRE_MINUTES)
  212. refresh_expires = timedelta(seconds=settings.REFRESH_TOKEN_EXPIRE_MINUTES)
  213. now = datetime.now()
  214. # 记录租户信息到日志
  215. log.info(f"用户ID: {user.id}, 用户名: {user.username} 正在生成JWT令牌")
  216. # 生成会话信息
  217. session_info = OnlineOutSchema(
  218. session_id=session_id,
  219. user_id=user.id,
  220. name=user.name,
  221. user_name=user.username,
  222. ipaddr=request_ip,
  223. login_location=login_location,
  224. os=user_agent.os.family,
  225. browser=user_agent.browser.family,
  226. login_time=user.last_login,
  227. login_type=login_type,
  228. ).model_dump_json()
  229. access_token = create_access_token(
  230. payload=JWTPayloadSchema(
  231. sub=session_info,
  232. is_refresh=False,
  233. exp=now + access_expires,
  234. )
  235. )
  236. refresh_token = create_access_token(
  237. payload=JWTPayloadSchema(
  238. sub=session_info,
  239. is_refresh=True,
  240. exp=now + refresh_expires,
  241. )
  242. )
  243. # 设置新的token
  244. await RedisCURD(redis).set(
  245. key=f"{RedisInitKeyConfig.ACCESS_TOKEN.key}:{session_id}",
  246. value=access_token,
  247. expire=int(access_expires.total_seconds()),
  248. )
  249. await RedisCURD(redis).set(
  250. key=f"{RedisInitKeyConfig.REFRESH_TOKEN.key}:{session_id}",
  251. value=refresh_token,
  252. expire=int(refresh_expires.total_seconds()),
  253. )
  254. return JWTOutSchema(
  255. access_token=access_token,
  256. refresh_token=refresh_token,
  257. expires_in=int(access_expires.total_seconds()),
  258. token_type=settings.TOKEN_TYPE,
  259. )
  260. @classmethod
  261. async def refresh_token_service(
  262. cls,
  263. db: AsyncSession,
  264. redis: Redis,
  265. request: Request,
  266. refresh_token: RefreshTokenPayloadSchema,
  267. ) -> JWTOutSchema:
  268. """
  269. 刷新访问令牌
  270. 参数:
  271. - db (AsyncSession): 数据库会话对象
  272. - redis (Redis): Redis客户端对象
  273. - request (Request): FastAPI请求对象
  274. - refresh_token (RefreshTokenPayloadSchema): 刷新令牌数据
  275. 返回:
  276. - JWTOutSchema: 新的令牌对象
  277. 异常:
  278. - CustomException: 刷新令牌无效时抛出异常
  279. """
  280. token_payload: JWTPayloadSchema = decode_access_token(token=refresh_token.refresh_token)
  281. if not token_payload.is_refresh:
  282. raise CustomException(msg="非法凭证,请传入刷新令牌")
  283. # 去 Redis 查完整信息
  284. session_info = json.loads(token_payload.sub)
  285. session_id = session_info.get("session_id")
  286. user_id = session_info.get("user_id")
  287. if not session_id or not user_id:
  288. raise CustomException(msg="非法凭证,无法获取会话编号或用户ID")
  289. # 用户认证
  290. auth = AuthSchema(db=db)
  291. user = await UserCRUD(auth).get_by_id_crud(id=user_id)
  292. if not user:
  293. raise CustomException(msg="刷新token失败,用户不存在")
  294. # 记录刷新令牌时的租户信息
  295. log.info(f"用户ID: {user.id}, 用户名: {user.username} 正在刷新JWT令牌")
  296. # 设置新的 token
  297. access_expires = timedelta(seconds=settings.ACCESS_TOKEN_EXPIRE_MINUTES)
  298. refresh_expires = timedelta(seconds=settings.REFRESH_TOKEN_EXPIRE_MINUTES)
  299. now = datetime.now()
  300. session_info_json = json.dumps(session_info)
  301. access_token = create_access_token(
  302. payload=JWTPayloadSchema(
  303. sub=session_info_json,
  304. is_refresh=False,
  305. exp=now + access_expires,
  306. )
  307. )
  308. refresh_token_new = create_access_token(
  309. payload=JWTPayloadSchema(
  310. sub=session_info_json,
  311. is_refresh=True,
  312. exp=now + refresh_expires,
  313. )
  314. )
  315. # 覆盖写入 Redis
  316. await RedisCURD(redis).set(
  317. key=f"{RedisInitKeyConfig.ACCESS_TOKEN.key}:{session_id}",
  318. value=access_token,
  319. expire=int(access_expires.total_seconds()),
  320. )
  321. await RedisCURD(redis).set(
  322. key=f"{RedisInitKeyConfig.REFRESH_TOKEN.key}:{session_id}",
  323. value=refresh_token_new,
  324. expire=int(refresh_expires.total_seconds()),
  325. )
  326. return JWTOutSchema(
  327. access_token=access_token,
  328. refresh_token=refresh_token_new,
  329. token_type=settings.TOKEN_TYPE,
  330. expires_in=int(access_expires.total_seconds()),
  331. )
  332. @classmethod
  333. async def logout_service(cls, redis: Redis, token: LogoutPayloadSchema) -> bool:
  334. """
  335. 退出登录
  336. 参数:
  337. - redis (Redis): Redis客户端对象
  338. - token (LogoutPayloadSchema): 退出登录令牌数据
  339. 返回:
  340. - bool: 退出成功返回True
  341. 异常:
  342. - CustomException: 令牌无效时抛出异常
  343. """
  344. payload: JWTPayloadSchema = decode_access_token(token=token.token)
  345. session_info = json.loads(payload.sub)
  346. session_id = session_info.get("session_id")
  347. if not session_id:
  348. raise CustomException(msg="非法凭证,无法获取会话编号")
  349. # 删除Redis中的在线用户、访问令牌、刷新令牌
  350. await RedisCURD(redis).delete(f"{RedisInitKeyConfig.ACCESS_TOKEN.key}:{session_id}")
  351. await RedisCURD(redis).delete(f"{RedisInitKeyConfig.REFRESH_TOKEN.key}:{session_id}")
  352. log.info(f"用户退出登录成功,会话编号:{session_id}")
  353. return True
  354. class CaptchaService:
  355. """验证码服务"""
  356. @classmethod
  357. async def get_captcha_service(cls, redis: Redis) -> dict[str, CaptchaKey | CaptchaBase64]:
  358. """
  359. 获取验证码
  360. 参数:
  361. - redis (Redis): Redis客户端对象
  362. 返回:
  363. - dict[str, CaptchaKey | CaptchaBase64]: 包含验证码key和base64图片的字典
  364. 异常:
  365. - CustomException: 验证码服务未启用时抛出异常
  366. """
  367. if not settings.CAPTCHA_ENABLE:
  368. raise CustomException(msg="未开启验证码服务")
  369. # 生成验证码图片和值
  370. captcha_base64, captcha_value = CaptchaUtil.captcha_arithmetic()
  371. captcha_key = get_random_character()
  372. # 保存到Redis并设置过期时间
  373. redis_key = f"{RedisInitKeyConfig.CAPTCHA_CODES.key}:{captcha_key}"
  374. await RedisCURD(redis).set(
  375. key=redis_key,
  376. value=captcha_value,
  377. expire=settings.CAPTCHA_EXPIRE_SECONDS,
  378. )
  379. log.info(f"生成验证码成功,验证码:{captcha_value}")
  380. # 返回验证码信息
  381. return CaptchaOutSchema(
  382. enable=settings.CAPTCHA_ENABLE,
  383. key=CaptchaKey(captcha_key),
  384. img_base=CaptchaBase64(f"data:image/png;base64,{captcha_base64}"),
  385. ).model_dump()
  386. @classmethod
  387. async def check_captcha_service(cls, redis: Redis, key: str, captcha: str) -> bool:
  388. """
  389. 校验验证码
  390. 参数:
  391. - redis (Redis): Redis客户端对象
  392. - key (str): 验证码key
  393. - captcha (str): 用户输入的验证码
  394. 返回:
  395. - bool: 验证通过返回True
  396. 异常:
  397. - CustomException: 验证码无效或错误时抛出异常
  398. """
  399. if not captcha:
  400. raise CustomException(msg="验证码不能为空")
  401. # 获取Redis中存储的验证码
  402. redis_key = f"{RedisInitKeyConfig.CAPTCHA_CODES.key}:{key}"
  403. captcha_value = await RedisCURD(redis).get(redis_key)
  404. if not captcha_value:
  405. log.error("验证码已过期或不存在")
  406. raise CustomException(msg="验证码已过期")
  407. # 验证码不区分大小写比对
  408. if captcha.lower() != captcha_value.lower():
  409. log.error(f"验证码错误,用户输入:{captcha},正确值:{captcha_value}")
  410. raise CustomException(msg="验证码错误")
  411. # 验证成功后删除验证码,避免重复使用
  412. await RedisCURD(redis).delete(redis_key)
  413. log.info(f"验证码校验成功,key:{key}")
  414. return True
  415. class AutoLoginService:
  416. """免登录服务"""
  417. # 免登录Token前缀
  418. AUTO_LOGIN_PREFIX = "fastapiadmin:auto_login:"
  419. # Token有效期(秒) - 5分钟
  420. TOKEN_EXPIRE = 300
  421. @classmethod
  422. async def get_auto_login_users_service(cls, db: AsyncSession) -> list[AutoLoginUserSchema]:
  423. """
  424. 获取免登录用户列表
  425. 参数:
  426. - db (AsyncSession): 数据库会话对象
  427. 返回:
  428. - list[AutoLoginUserSchema]: 用户列表
  429. """
  430. from sqlalchemy import select
  431. from app.api.v1.module_system.user.model import UserModel
  432. # 查询所有启用的用户
  433. stmt = select(UserModel).where(UserModel.status == "0").order_by(UserModel.id)
  434. result = await db.execute(stmt)
  435. users = result.scalars().all()
  436. return [
  437. AutoLoginUserSchema(
  438. id=user.id,
  439. username=user.username,
  440. name=user.name,
  441. avatar=user.avatar,
  442. )
  443. for user in users
  444. ]
  445. @classmethod
  446. async def create_auto_login_token_service(
  447. cls, redis: Redis, db: AsyncSession, user_id: int
  448. ) -> AutoLoginTokenSchema:
  449. """
  450. 创建免登录Token
  451. 参数:
  452. - request (Request): FastAPI请求对象
  453. - redis (Redis): Redis客户端对象
  454. - db (AsyncSession): 数据库会话对象
  455. - user_id (int): 用户ID
  456. 返回:
  457. - AutoLoginTokenSchema: 免登录Token和用户信息
  458. 异常:
  459. - CustomException: 用户不存在或已停用时抛出异常
  460. """
  461. from sqlalchemy import select
  462. from app.api.v1.module_system.user.model import UserModel
  463. # 查询用户
  464. stmt = select(UserModel).where(UserModel.id == user_id)
  465. result = await db.execute(stmt)
  466. user = result.scalar_one_or_none()
  467. if not user:
  468. raise CustomException(msg="用户不存在")
  469. if user.status == "1":
  470. raise CustomException(msg="用户已被停用")
  471. # 生成免登录Token
  472. import uuid
  473. token = str(uuid.uuid4())
  474. token_key = f"{cls.AUTO_LOGIN_PREFIX}{token}"
  475. # 存储到Redis,设置5分钟过期
  476. token_data = {
  477. "user_id": user.id,
  478. "username": user.username,
  479. "created_at": datetime.now().isoformat(),
  480. }
  481. await RedisCURD(redis).set(
  482. key=token_key,
  483. value=json.dumps(token_data),
  484. expire=cls.TOKEN_EXPIRE,
  485. )
  486. log.info(f"创建免登录Token成功,用户:{user.username}")
  487. return AutoLoginTokenSchema(
  488. token=token,
  489. user=AutoLoginUserSchema(
  490. id=user.id,
  491. username=user.username,
  492. name=user.name,
  493. avatar=user.avatar,
  494. ),
  495. )
  496. @classmethod
  497. async def auto_login_service(
  498. cls, request: Request, redis: Redis, db: AsyncSession, token: str
  499. ) -> JWTOutSchema:
  500. """
  501. 免登录
  502. 参数:
  503. - request (Request): FastAPI请求对象
  504. - redis (Redis): Redis客户端对象
  505. - db (AsyncSession): 数据库会话对象
  506. - token (str): 免登录Token
  507. 返回:
  508. - JWTOutSchema: JWT令牌信息
  509. 异常:
  510. - CustomException: Token无效或过期时抛出异常
  511. """
  512. from sqlalchemy import select
  513. from app.api.v1.module_system.user.model import UserModel
  514. # 验证Token
  515. token_key = f"{cls.AUTO_LOGIN_PREFIX}{token}"
  516. token_data_str = await RedisCURD(redis).get(token_key)
  517. if not token_data_str:
  518. raise CustomException(msg="免登录Token已过期或无效")
  519. token_data = json.loads(token_data_str)
  520. user_id = token_data.get("user_id")
  521. # 查询用户
  522. stmt = select(UserModel).where(UserModel.id == user_id)
  523. result = await db.execute(stmt)
  524. user = result.scalar_one_or_none()
  525. if not user:
  526. raise CustomException(msg="用户不存在")
  527. if user.status == "1":
  528. raise CustomException(msg="用户已被停用")
  529. # 删除已使用的Token
  530. await RedisCURD(redis).delete(token_key)
  531. # 使用LoginService创建token
  532. jwt_token = await LoginService.create_token_service(
  533. request=request, redis=redis, user=user, login_type="PC端"
  534. )
  535. log.info(f"用户{user.username}免登录成功")
  536. return jwt_token
  537. class SmsCodeService:
  538. """短信验证码服务"""
  539. SMS_CODE_PREFIX = "sms_code"
  540. SMS_CODE_EXPIRE = 60 * 10
  541. @classmethod
  542. async def send_sms_code_service(
  543. cls, sms_code: SmsCodeSchema, redis: Redis
  544. ) -> bool:
  545. """
  546. 发送短信验证码
  547. 参数:
  548. - smsCode (SmsCodeSchema): 短信验证码请求模型
  549. - redis (Redis): Redis客户端对象
  550. 异常:
  551. - CustomException: 验证码发送失败时抛出异常。
  552. """
  553. template = SmsTemplateEnum.get_template_by_name(sms_code.template_name)
  554. code = generate_random_code(6)
  555. redis_key = f"{cls.SMS_CODE_PREFIX}:{sms_code.template_name}:{sms_code.mobile}"
  556. await RedisCURD(redis).set(
  557. key=redis_key,
  558. value=code,
  559. expire=cls.SMS_CODE_EXPIRE,
  560. )
  561. request = SendSmsRequest(
  562. phone_numbers=sms_code.mobile,
  563. template_code=template.template_code,
  564. template_param=template.template_param_fn(code=code),
  565. )
  566. return await SmsSender.send_sms(request)
  567. @classmethod
  568. async def verify_sms_code_service(
  569. cls, sms_code: SmsCodeSchema, redis: Redis
  570. ) -> bool:
  571. """
  572. 验证短信验证码
  573. 参数:
  574. - smsCode (SmsCodeSchema): 短信验证码请求模型
  575. - redis (Redis): Redis客户端对象
  576. 返回:
  577. - bool: 验证结果
  578. 异常:
  579. - CustomException: 验证码验证失败时抛出异常。
  580. """
  581. redis_key = f"{cls.SMS_CODE_PREFIX}:{sms_code.template_name}:{sms_code.mobile}"
  582. code = await RedisCURD(redis).get(redis_key)
  583. if not code or code != sms_code.code:
  584. return False
  585. await RedisCURD(redis).delete(redis_key)
  586. return True