service.py 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393
  1. # -*- coding: utf-8 -*-
  2. import json
  3. from redis.asyncio.client import Redis
  4. from fastapi import UploadFile
  5. from redis.asyncio.client import Redis
  6. from app.common.enums import RedisInitKeyConfig
  7. from app.core.database import async_db_session
  8. from app.core.redis_crud import RedisCURD
  9. from app.utils.excel_util import ExcelUtil
  10. from app.utils.upload_util import UploadUtil
  11. from app.core.base_schema import UploadResponseSchema
  12. from app.core.exceptions import CustomException
  13. from app.core.logger import log
  14. from ..auth.schema import AuthSchema
  15. from .schema import ParamsOutSchema, ParamsUpdateSchema, ParamsCreateSchema, ParamsQueryParam
  16. from .crud import ParamsCRUD
  17. class ParamsService:
  18. """
  19. 配置管理模块服务层
  20. """
  21. @classmethod
  22. async def get_obj_detail_service(cls, auth: AuthSchema, id: int) -> dict:
  23. """
  24. 获取配置详情
  25. 参数:
  26. - auth (AuthSchema): 认证信息模型
  27. - id (int): 配置管理型ID
  28. 返回:
  29. - dict: 配置管理型模型实例字典表示
  30. """
  31. obj = await ParamsCRUD(auth).get_obj_by_id_crud(id=id)
  32. return ParamsOutSchema.model_validate(obj).model_dump()
  33. @classmethod
  34. async def get_obj_by_key_service(cls, auth: AuthSchema, config_key: str) -> dict:
  35. """
  36. 根据配置键获取配置详情
  37. 参数:
  38. - auth (AuthSchema): 认证信息模型
  39. - config_key (str): 配置管理型key
  40. 返回:
  41. - Dict: 配置管理型模型实例字典表示
  42. """
  43. obj = await ParamsCRUD(auth).get_obj_by_key_crud(key=config_key)
  44. if not obj:
  45. raise CustomException(msg=f'配置键 {config_key} 不存在')
  46. return ParamsOutSchema.model_validate(obj).model_dump()
  47. @classmethod
  48. async def get_config_value_by_key_service(cls, auth: AuthSchema, config_key: str) -> str | None:
  49. """
  50. 根据配置键获取配置值
  51. 参数:
  52. - auth (AuthSchema): 认证信息模型
  53. - config_key (str): 配置管理型key
  54. 返回:
  55. - str | None: 配置值字符串或None
  56. """
  57. obj = await ParamsCRUD(auth).get_obj_by_key_crud(key=config_key)
  58. if not obj:
  59. raise CustomException(msg=f'配置键 {config_key} 不存在')
  60. return obj.config_value
  61. @classmethod
  62. async def get_obj_list_service(cls, auth: AuthSchema, search: ParamsQueryParam | None = None, order_by: list[dict] | None = None) -> list[dict]:
  63. """
  64. 获取配置管理型列表
  65. 参数:
  66. - auth (AuthSchema): 认证信息模型
  67. - search (ParamsQueryParam | None): 查询参数对象
  68. - order_by (list[dict] | None): 排序参数列表
  69. 返回:
  70. - list[dict]: 配置管理型模型实例字典列表表示
  71. """
  72. obj_list = None
  73. if search:
  74. obj_list = await ParamsCRUD(auth).get_obj_list_crud(search=search.__dict__, order_by=order_by)
  75. else:
  76. obj_list = await ParamsCRUD(auth).get_obj_list_crud()
  77. return [ParamsOutSchema.model_validate(obj).model_dump() for obj in obj_list]
  78. @classmethod
  79. async def create_obj_service(cls, auth: AuthSchema, redis: Redis, data: ParamsCreateSchema) -> dict:
  80. """
  81. 创建配置管理型
  82. 参数:
  83. - auth (AuthSchema): 认证信息模型
  84. - redis (Redis): Redis 客户端实例
  85. - data (ParamsCreateSchema): 配置管理型创建模型
  86. 返回:
  87. - dict: 新创建的配置管理型模型实例字典表示
  88. """
  89. exist_obj = await ParamsCRUD(auth).get(config_key=data.config_key)
  90. if exist_obj:
  91. raise CustomException(msg='创建失败,该配置key已存在')
  92. obj = await ParamsCRUD(auth).create_obj_crud(data=data)
  93. new_obj_dict = ParamsOutSchema.model_validate(obj).model_dump()
  94. # 同步redis
  95. redis_key = f"{RedisInitKeyConfig.SYSTEM_CONFIG.key}:{data.config_key}"
  96. try:
  97. result = await RedisCURD(redis).set(
  98. key=redis_key,
  99. value="",
  100. )
  101. if not result:
  102. log.error(f"同步配置到缓存失败: {new_obj_dict}")
  103. raise CustomException(msg="同步配置到缓存失败")
  104. except Exception as e:
  105. log.error(f"创建字典类型失败: {e}")
  106. raise CustomException(msg=f"创建字典类型失败 {e}")
  107. return new_obj_dict
  108. @classmethod
  109. async def update_obj_service(cls, auth: AuthSchema, redis: Redis, id:int, data: ParamsUpdateSchema) -> dict:
  110. """
  111. 更新配置管理型
  112. 参数:
  113. - auth (AuthSchema): 认证信息模型
  114. - redis (Redis): Redis 客户端实例
  115. - id (int): 配置管理型ID
  116. - data (ParamsUpdateSchema): 配置管理型更新模型
  117. 返回:
  118. - Dict: 更新后的配置管理型模型实例字典表示
  119. """
  120. exist_obj = await ParamsCRUD(auth).get_obj_by_id_crud(id=id)
  121. if not exist_obj:
  122. raise CustomException(msg='更新失败,该数系统配置不存在')
  123. if exist_obj.config_key != data.config_key:
  124. raise CustomException(msg='更新失败,系统配置key不允许修改')
  125. new_obj = await ParamsCRUD(auth).update_obj_crud(id=id, data=data)
  126. if not new_obj:
  127. raise CustomException(msg='更新失败,系统配置不存在')
  128. new_obj_dict = ParamsOutSchema.model_validate(new_obj).model_dump()
  129. # 同步redis
  130. redis_key = f"{RedisInitKeyConfig.SYSTEM_CONFIG.key}:{new_obj.config_key}"
  131. try:
  132. value = json.dumps(new_obj_dict, ensure_ascii=False)
  133. result = await RedisCURD(redis).set(
  134. key=redis_key,
  135. value=value,
  136. )
  137. if not result:
  138. log.error(f"同步配置到缓存失败: {new_obj_dict}")
  139. raise CustomException(msg="同步配置到缓存失败")
  140. except Exception as e:
  141. log.error(f"更新系统配置失败: {e}")
  142. raise CustomException(msg="更新系统配置失败")
  143. return new_obj_dict
  144. @classmethod
  145. async def delete_obj_service(cls, auth: AuthSchema, redis: Redis, ids: list[int]) -> None:
  146. """
  147. 删除配置管理型
  148. 参数:
  149. - auth (AuthSchema): 认证信息模型
  150. - redis (Redis): Redis 客户端实例
  151. - ids (list[int]): 配置管理型ID列表
  152. 返回:
  153. - None
  154. """
  155. if len(ids) < 1:
  156. raise CustomException(msg='删除失败,删除对象不能为空')
  157. for id in ids:
  158. exist_obj = await ParamsCRUD(auth).get_obj_by_id_crud(id=id)
  159. if not exist_obj:
  160. raise CustomException(msg='删除失败,该数据字典类型不存在')
  161. # 检查是否是否初始化类型
  162. if exist_obj.config_type:
  163. # 如果有字典数据,不能删除
  164. raise CustomException(msg=f'{exist_obj.config_name} 删除失败,系统初始化配置不可以删除')
  165. await ParamsCRUD(auth).delete_obj_crud(ids=ids)
  166. # 同步删除Redis缓存
  167. for id in ids:
  168. exist_obj = await ParamsCRUD(auth).get_obj_by_id_crud(id=id)
  169. if not exist_obj:
  170. continue
  171. redis_key = f"{RedisInitKeyConfig.SYSTEM_CONFIG.key}:{exist_obj.config_key}"
  172. try:
  173. await RedisCURD(redis).delete(redis_key)
  174. log.info(f"删除系统配置成功: {id}")
  175. except Exception as e:
  176. log.error(f"删除系统配置失败: {e}")
  177. raise CustomException(msg="删除字典类型失败")
  178. @classmethod
  179. async def export_obj_service(cls, data_list: list[dict]) -> bytes:
  180. """
  181. 导出系统配置列表
  182. 参数:
  183. - data_list (list[dict]): 系统配置模型实例字典列表表示
  184. 返回:
  185. - bytes: Excel文件二进制数据
  186. """
  187. mapping_dict = {
  188. 'id': '编号',
  189. 'config_name': '参数名称',
  190. 'config_key': '参数键名',
  191. 'config_value': '参数键值',
  192. 'config_type': '系统内置((True:是 False:否))',
  193. 'description': '备注',
  194. 'created_time': '创建时间',
  195. 'updated_time': '更新时间',
  196. 'created_id': '创建者ID',
  197. 'updated_id': '更新者ID',
  198. }
  199. # 复制数据并转换状态
  200. data = data_list.copy()
  201. for item in data:
  202. # 处理状态
  203. item['config_type'] = '是' if item.get('config_type') else '否'
  204. item['creator'] = item.get('creator', {}).get('name', '未知') if isinstance(item.get('creator'), dict) else '未知'
  205. return ExcelUtil.export_list2excel(list_data=data, mapping_dict=mapping_dict)
  206. @classmethod
  207. async def upload_service(cls, base_url: str, file: UploadFile) -> dict:
  208. """
  209. 上传文件
  210. 参数:
  211. - base_url (str): 基础URL
  212. - file (UploadFile): 上传的文件对象
  213. 返回:
  214. - dict: 上传文件的响应模型实例字典表示
  215. """
  216. filename, filepath, file_url = await UploadUtil.upload_file(file=file, base_url=base_url)
  217. return UploadResponseSchema(
  218. file_path=f'{filepath}',
  219. file_name=filename,
  220. origin_name=file.filename,
  221. file_url=f'{file_url}',
  222. ).model_dump()
  223. @classmethod
  224. async def init_config_service(cls, redis: Redis) -> None:
  225. """
  226. 初始化系统配置
  227. 参数:
  228. - redis (Redis): Redis 客户端实例
  229. 返回:
  230. - None
  231. """
  232. async with async_db_session() as session:
  233. async with session.begin():
  234. # 在初始化过程中,不需要检查数据权限
  235. auth = AuthSchema(db=session, check_data_scope=False)
  236. config_obj = await ParamsCRUD(auth).get_obj_list_crud()
  237. if not config_obj:
  238. raise CustomException(msg="系统配置不存在")
  239. try:
  240. # 保存到Redis并设置过期时间
  241. for config in config_obj:
  242. redis_key = (f"{RedisInitKeyConfig.SYSTEM_CONFIG.key}:{config.config_key}")
  243. config_obj_dict = ParamsOutSchema.model_validate(config).model_dump()
  244. value = json.dumps(config_obj_dict, ensure_ascii=False)
  245. result = await RedisCURD(redis).set(
  246. key=redis_key,
  247. value=value,
  248. )
  249. log.info(f"✅ 系统配置缓存成功: {config.config_key}")
  250. if not result:
  251. log.error(f"❌️ 初始化系统配置失败: {config_obj_dict}")
  252. raise CustomException(msg="初始化系统配置失败")
  253. except Exception as e:
  254. log.error(f"❌️ 初始化系统配置失败: {e}")
  255. raise CustomException(msg="初始化系统配置失败")
  256. @classmethod
  257. async def get_init_config_service(cls, redis: Redis) -> list[dict]:
  258. """
  259. 获取系统配置
  260. 参数:
  261. - redis (Redis): Redis 客户端实例
  262. 返回:
  263. - list[dict]: 系统配置模型实例字典列表表示
  264. """
  265. redis_keys = await RedisCURD(redis).get_keys(f"{RedisInitKeyConfig.SYSTEM_CONFIG.key}:*")
  266. redis_configs = await RedisCURD(redis).mget(redis_keys)
  267. configs = []
  268. for config in redis_configs:
  269. if not config:
  270. continue
  271. try:
  272. new_config = json.loads(config)
  273. configs.append(new_config)
  274. except Exception as e:
  275. log.error(f"解析系统配置数据失败: {e}")
  276. continue
  277. return configs
  278. @classmethod
  279. async def get_system_config_for_middleware(cls, redis: Redis) -> dict:
  280. """
  281. 获取中间件所需的系统配置
  282. 参数:
  283. - redis (Redis): Redis 客户端实例
  284. 返回:
  285. - dict: 包含演示模式、IP白名单、API白名单和IP黑名单的配置字典
  286. """
  287. # 定义需要获取的配置键
  288. config_keys = [
  289. f"{RedisInitKeyConfig.SYSTEM_CONFIG.key}:demo_enable",
  290. f"{RedisInitKeyConfig.SYSTEM_CONFIG.key}:ip_white_list",
  291. f"{RedisInitKeyConfig.SYSTEM_CONFIG.key}:white_api_list_path",
  292. f"{RedisInitKeyConfig.SYSTEM_CONFIG.key}:ip_black_list"
  293. ]
  294. # 批量获取配置
  295. config_values = await RedisCURD(redis).mget(config_keys)
  296. # 初始化默认配置
  297. config_result = {
  298. "demo_enable": False,
  299. "ip_white_list": [],
  300. "white_api_list_path": [],
  301. "ip_black_list": []
  302. }
  303. # 解析演示模式配置
  304. if config_values[0]:
  305. try:
  306. demo_config = json.loads(config_values[0])
  307. config_result["demo_enable"] = demo_config.get("config_value", False) if isinstance(demo_config, dict) else False
  308. except json.JSONDecodeError:
  309. log.error(f"解析演示模式配置失败")
  310. # 解析IP白名单配置
  311. if config_values[1]:
  312. try:
  313. ip_white_config = json.loads(config_values[1])
  314. # 确保是列表类型
  315. config_result["ip_white_list"] = json.loads(ip_white_config.get("config_value", []))
  316. except json.JSONDecodeError:
  317. log.error(f"解析IP白名单配置失败")
  318. # 解析IP黑名单
  319. # 解析API路径白名单
  320. if config_values[2]:
  321. try:
  322. white_api_config = json.loads(config_values[2])
  323. # 确保是列表类型
  324. config_result["white_api_list_path"] = json.loads(white_api_config.get("config_value", []))
  325. except json.JSONDecodeError:
  326. log.error(f"解析API白名单配置失败")
  327. # 解析IP黑名单
  328. if config_values[3]:
  329. try:
  330. black_ip_config = json.loads(config_values[3])
  331. # 确保是列表类型
  332. config_result["ip_black_list"] = json.loads(black_ip_config.get("config_value", []))
  333. except json.JSONDecodeError:
  334. log.error(f"解析IP黑名单配置失败")
  335. return config_result