controller.py 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341
  1. from typing import Annotated
  2. import uuid
  3. from fastapi import APIRouter, Depends, Path, Query
  4. from fastapi.responses import JSONResponse
  5. from app.api.v1.module_system.auth.schema import AuthSchema
  6. from app.common.response import ResponseSchema, SuccessResponse
  7. from app.core.dependencies import AuthPermission
  8. from app.core.logger import log
  9. from app.core.router_class import OperationLogRoute
  10. from app.plugin.module_payment.expense.institution.schema import InstitutionListOutSchema
  11. from .service import InstitutionService, InstitutionScopeService, IssueruleService
  12. from alipay.aop.api.domain.AlipayEbppInvoiceInstitutionCreateModel import (
  13. AlipayEbppInvoiceInstitutionCreateModel,
  14. )
  15. from alipay.aop.api.response.AlipayEbppInvoiceInstitutionCreateResponse import (
  16. AlipayEbppInvoiceInstitutionCreateResponse,
  17. )
  18. from alipay.aop.api.domain.AlipayEbppInvoiceInstitutionDeleteModel import (
  19. AlipayEbppInvoiceInstitutionDeleteModel,
  20. )
  21. from alipay.aop.api.response.AlipayEbppInvoiceInstitutionDeleteResponse import (
  22. AlipayEbppInvoiceInstitutionDeleteResponse,
  23. )
  24. from alipay.aop.api.domain.AlipayEbppInvoiceInstitutionModifyModel import (
  25. AlipayEbppInvoiceInstitutionModifyModel,
  26. )
  27. from alipay.aop.api.response.AlipayEbppInvoiceInstitutionModifyResponse import (
  28. AlipayEbppInvoiceInstitutionModifyResponse,
  29. )
  30. InstitutionRouter = APIRouter(
  31. route_class=OperationLogRoute,
  32. prefix="/institution",
  33. tags=["费控制度"],
  34. )
  35. @InstitutionRouter.post(
  36. "",
  37. summary="创建费控制度",
  38. description="创建费控制度。支持串联调用:创建制度→设置成员→创建发放规则",
  39. )
  40. async def create_institution_controller(
  41. data: dict,
  42. auth: Annotated[AuthSchema, Depends(AuthPermission(["module_payment:expense:institution:create"]))],
  43. ) -> JSONResponse:
  44. """创建费控制度(含完整串联流程)"""
  45. enterprise_id = data.get("enterprise_id", "")
  46. if not enterprise_id:
  47. from app.plugin.module_payment.enterprise.model import EnterpriseModel
  48. from sqlalchemy import select
  49. tenant_id = auth.user.tenant_id if auth.user and auth.user.tenant_id else auth.tenant_id
  50. log.info(f"推导 enterprise_id: tenant_id={tenant_id}, user_tenant_id={getattr(auth.user, 'tenant_id', None)}")
  51. stmt = select(EnterpriseModel).where(EnterpriseModel.tenant_id == tenant_id).limit(1)
  52. result = await auth.db.execute(stmt)
  53. enterprise = result.scalar_one_or_none()
  54. log.info(f"查询 enterprise 结果: {enterprise.enterprise_id if enterprise else 'None'}")
  55. enterprise_id = enterprise.enterprise_id if enterprise else ""
  56. if enterprise_id:
  57. data["enterprise_id"] = enterprise_id
  58. # 字段映射:前端 name → Alipay institution_name
  59. if data.get("name") and not data.get("institution_name"):
  60. data["institution_name"] = data["name"]
  61. institution_create_model = AlipayEbppInvoiceInstitutionCreateModel.from_alipay_dict(data)
  62. # 解析适用成员数据
  63. scope_data = None
  64. adapter_type = data.get("applicable_scope")
  65. if adapter_type and adapter_type != "NONE":
  66. scope_data = {
  67. "adapter_type": adapter_type,
  68. "owner_type": data.get("scope_owner_type", "EMPLOYEE"),
  69. "add_owner_id_list": data.get("scope_owner_id_list"),
  70. }
  71. # 解析发放规则数据
  72. issuerule_data = None
  73. if data.get("grant_mode") == "period":
  74. period_type_raw = data.get("period_type", "monthly")
  75. # 映射前端period_type到支付宝枚举
  76. ISSUE_TYPE_MAP = {
  77. "daily": "ISSUE_DAY",
  78. "weekly": "ISSUE_WEEK",
  79. "monthly": "ISSUE_MONTH",
  80. "quarterly": "ISSUE_QUARTER",
  81. "yearly": "ISSUE_YEAR",
  82. }
  83. issue_type = ISSUE_TYPE_MAP.get(period_type_raw, "ISSUE_MONTH")
  84. amount = data.get("amount", 0)
  85. # 有效时间配置
  86. effective_time_type = data.get("effective_time_type", "unlimited")
  87. if effective_time_type == "unlimited":
  88. effective_period = '{"all": true}'
  89. elif effective_time_type == "workday":
  90. workday_start = data.get("workday_start_time", "00:00")
  91. workday_end = data.get("workday_end_time", "23:59")
  92. effective_period = f'{{"regular":{{"workday":[["{workday_start}","{workday_end}"]]}}}}'
  93. else:
  94. effective_period = '{"all": true}'
  95. issuerule_data = {
  96. "quota_type": "CAP",
  97. "issue_type": issue_type,
  98. "issue_amount_value": str(amount),
  99. "issue_rule_name": data.get("name", "") + "-发放规则",
  100. "effective_period": effective_period,
  101. "invalid_mode": 1 if data.get("effective_time_type") == "unlimited" else 0,
  102. "share_mode": 0,
  103. "outer_source_id": data.get("outer_source_id") or str(uuid.uuid4()),
  104. }
  105. result = await InstitutionService.create_institution_full_flow(
  106. auth=auth,
  107. institution_model=institution_create_model,
  108. enterprise_id=enterprise_id,
  109. scope_data=scope_data,
  110. issuerule_data=issuerule_data,
  111. )
  112. log.info(f"创建费控制度成功: institution_id={result.get('institution_id')}")
  113. return SuccessResponse(data=result, msg="创建费控制度成功")
  114. @InstitutionRouter.get(
  115. "",
  116. summary="查询费控制度列表",
  117. description="分页查询费控制度列表",
  118. response_model=ResponseSchema[InstitutionListOutSchema],
  119. )
  120. async def list_institution_controller(
  121. auth: Annotated[AuthSchema, Depends(AuthPermission(["module_payment:expense:institution:list"]))],
  122. page_no: Annotated[int, Query(description="页码")] = 1,
  123. page_size: Annotated[int, Query(description="每页数量")] = 20,
  124. enterprise_id: Annotated[str | None, Query(description="企业ID")] = None,
  125. name: Annotated[str | None, Query(description="制度名称")] = None,
  126. expense_type: Annotated[str | None, Query(description="费用类型")] = None,
  127. status: Annotated[str | None, Query(description="状态")] = None,
  128. ) -> JSONResponse:
  129. """查询费控制度列表"""
  130. search = {}
  131. if enterprise_id:
  132. search["enterprise_id"] = enterprise_id
  133. if name:
  134. search["name"] = name
  135. if expense_type:
  136. search["expense_type"] = expense_type
  137. if status:
  138. search["status"] = status
  139. result = await InstitutionService.list_service(
  140. auth=auth, page_no=page_no, page_size=page_size, search=search
  141. )
  142. return SuccessResponse(data=result, msg="查询费控制度列表成功")
  143. @InstitutionRouter.get(
  144. "/{institution_id}",
  145. summary="查询费控制度详情",
  146. description="查询费控制度详情 (alipay.ebpp.invoice.institution.detailinfo.query),失败时降级到本地DB",
  147. )
  148. async def detail_institution_controller(
  149. institution_id: Annotated[str, Path(description="制度ID")],
  150. auth: Annotated[AuthSchema, Depends(AuthPermission(["module_payment:expense:institution:detail"]))],
  151. enterprise_id: Annotated[str | None, Query(description="企业ID")] = None,
  152. ) -> JSONResponse:
  153. """查询费控制度详情"""
  154. if not enterprise_id:
  155. return SuccessResponse(data=None, msg="企业ID不能为空")
  156. result = await InstitutionService.detailinfo_query_service(
  157. auth=auth,
  158. institution_id=institution_id,
  159. enterprise_id=enterprise_id,
  160. )
  161. if result is None:
  162. return SuccessResponse(data=None, msg="制度不存在")
  163. return SuccessResponse(data=result, msg="查询费控制度详情成功")
  164. @InstitutionRouter.delete(
  165. "",
  166. summary="删除费控制度",
  167. description="删除费控制度 (alipay.ebpp.invoice.institution.delete)",
  168. )
  169. async def delete_institution_controller(
  170. data: dict,
  171. auth: Annotated[AuthSchema, Depends(AuthPermission(["module_payment:expense:institution:delete"]))],
  172. ) -> JSONResponse:
  173. """删除费控制度"""
  174. institution_delete_model = AlipayEbppInvoiceInstitutionDeleteModel(**data)
  175. result = await InstitutionService.delete_institution_service(auth=auth, data=institution_delete_model)
  176. log.info(f"删除费控制度成功: institution_id={institution_delete_model.institution_id}, enterprise_id={institution_delete_model.enterprise_id}")
  177. return SuccessResponse(data=result, msg="删除费控制度成功")
  178. @InstitutionRouter.post(
  179. "/modify",
  180. summary="编辑费控制度",
  181. description="编辑费控制度 (alipay.ebpp.invoice.institution.modify)",
  182. )
  183. async def modify_institution_controller(
  184. data: dict,
  185. auth: Annotated[AuthSchema, Depends(AuthPermission(["module_payment:expense:institution:modify"]))],
  186. ) -> JSONResponse:
  187. """编辑费控制度"""
  188. institution_modify_model = AlipayEbppInvoiceInstitutionModifyModel(**data)
  189. result = await InstitutionService.modify_institution_service(auth=auth, data=institution_modify_model)
  190. log.info(f"编辑费控制度成功: institution_id={institution_modify_model.institution_id}")
  191. return SuccessResponse(data=result, msg="编辑费控制度成功")
  192. # ========== 制度成员范围管理 ==========
  193. @InstitutionRouter.get(
  194. "/{institution_id}/scope",
  195. summary="查询制度成员范围",
  196. description="查询制度下成员范围 (alipay.ebpp.invoice.institution.scopepageinfo.query)",
  197. )
  198. async def list_scope_controller(
  199. institution_id: Annotated[str, Path(description="制度ID")],
  200. auth: Annotated[AuthSchema, Depends(AuthPermission(["module_payment:expense:institution:scope:list"]))],
  201. enterprise_id: Annotated[str | None, Query(description="企业ID")] = None,
  202. owner_type: Annotated[str | None, Query(description="适配ID类型")] = None,
  203. page_num: Annotated[int, Query(description="页码")] = 1,
  204. page_size: Annotated[int, Query(description="每页条数")] = 20,
  205. ) -> JSONResponse:
  206. """查询制度成员"""
  207. result = await InstitutionScopeService.scopepageinfo_query_service(
  208. auth=auth,
  209. institution_id=institution_id,
  210. enterprise_id=enterprise_id,
  211. page_num=page_num,
  212. page_size=page_size,
  213. owner_type=owner_type,
  214. )
  215. return SuccessResponse(data=result, msg="查询成功")
  216. @InstitutionRouter.post(
  217. "/{institution_id}/scope",
  218. summary="设置制度成员范围",
  219. description="设置/修改制度成员范围 (alipay.ebpp.invoice.institution.scope.modify)",
  220. )
  221. async def modify_scope_controller(
  222. institution_id: Annotated[str, Path(description="制度ID")],
  223. data: dict,
  224. auth: Annotated[AuthSchema, Depends(AuthPermission(["module_payment:expense:institution:scope:modify"]))],
  225. ) -> JSONResponse:
  226. """设置制度成员"""
  227. result = await InstitutionScopeService.scope_modify_service(
  228. auth=auth,
  229. institution_id=institution_id,
  230. data=data,
  231. )
  232. log.info(f"设置制度成员成功: institution_id={institution_id}, adapter_type={data.get('adapter_type')}")
  233. return SuccessResponse(data=result, msg="设置成功")
  234. # ========== 自动额度发放规则管理 ==========
  235. @InstitutionRouter.post(
  236. "/{institution_id}/issuerule",
  237. summary="创建自动发放规则",
  238. description="创建自动额度发放规则 (alipay.ebpp.invoice.issuerule.create)",
  239. )
  240. async def create_issuerule_controller(
  241. institution_id: Annotated[str, Path(description="制度ID")],
  242. data: dict,
  243. auth: Annotated[AuthSchema, Depends(AuthPermission(["module_payment:expense:institution:issuerule:create"]))],
  244. ) -> JSONResponse:
  245. """创建自动发放规则"""
  246. result = await IssueruleService.create_issuerule_service(
  247. auth=auth,
  248. institution_id=institution_id,
  249. enterprise_id=data.get("enterprise_id", ""),
  250. quota_type=data.get("quota_type", "CAP"),
  251. issue_type=data.get("issue_type", "ISSUE_MONTH"),
  252. issue_amount_value=data.get("issue_amount_value", "0"),
  253. outer_source_id=data.get("outer_source_id"),
  254. issue_rule_name=data.get("issue_rule_name"),
  255. effective_period=data.get("effective_period"),
  256. invalid_mode=data.get("invalid_mode"),
  257. share_mode=data.get("share_mode"),
  258. )
  259. log.info(f"创建自动发放规则成功: institution_id={institution_id}")
  260. return SuccessResponse(data=result, msg="创建自动发放规则成功")
  261. @InstitutionRouter.put(
  262. "/{institution_id}/issuerule/{issue_rule_id}",
  263. summary="编辑自动发放规则",
  264. description="编辑自动额度发放规则 (alipay.ebpp.invoice.issuerule.modify)",
  265. )
  266. async def modify_issuerule_controller(
  267. institution_id: Annotated[str, Path(description="制度ID")],
  268. issue_rule_id: Annotated[str, Path(description="发放规则ID")],
  269. data: dict,
  270. auth: Annotated[AuthSchema, Depends(AuthPermission(["module_payment:expense:institution:issuerule:modify"]))],
  271. ) -> JSONResponse:
  272. result = await IssueruleService.modify_issuerule_service(
  273. auth=auth,
  274. institution_id=institution_id,
  275. issue_rule_id=issue_rule_id,
  276. enterprise_id=data.get("enterprise_id", ""),
  277. quota_type=data.get("quota_type"),
  278. issue_type=data.get("issue_type"),
  279. issue_amount_value=data.get("issue_amount_value"),
  280. issue_rule_name=data.get("issue_rule_name"),
  281. effective=data.get("effective"),
  282. effective_period=data.get("effective_period"),
  283. invalid_mode=data.get("invalid_mode"),
  284. share_mode=data.get("share_mode"),
  285. )
  286. log.info(f"编辑自动发放规则成功: issue_rule_id={issue_rule_id}")
  287. return SuccessResponse(data=result, msg="编辑自动发放规则成功")
  288. @InstitutionRouter.delete(
  289. "/{institution_id}/issuerule",
  290. summary="删除自动发放规则",
  291. description="删除自动额度发放规则 (alipay.ebpp.invoice.issuerule.delete)",
  292. )
  293. async def delete_issuerule_controller(
  294. institution_id: Annotated[str, Path(description="制度ID")],
  295. data: dict,
  296. auth: Annotated[AuthSchema, Depends(AuthPermission(["module_payment:expense:institution:issuerule:delete"]))],
  297. ) -> JSONResponse:
  298. result = await IssueruleService.delete_issuerule_service(
  299. auth=auth,
  300. institution_id=institution_id,
  301. issue_rule_id_list=data.get("issue_rule_id_list", []),
  302. enterprise_id=data.get("enterprise_id", ""),
  303. )
  304. log.info(f"删除自动发放规则成功: institution_id={institution_id}")
  305. return SuccessResponse(data=result, msg="删除自动发放规则成功")