Files
X-Financial/server/src/app/services/expense_claim_pagination.py
caoxiaozhu 1f4681f486 feat(claim): 重构报销审批流并收敛风险标记
- 直属领导兼任部门 P8 预算审批人时合并预算审批,直接流转至财务审批
- 预算超过警戒值时强制要求预算管理者填写审批意见
- 新增风险标记去重工具,消除各审核阶段重复风险卡片
- 新增工作流修复 Mixin,纠正重复预算审批阶段的历史数据
- 收紧单据删除权限至 admin,放宽预算分析可见范围至当前审核人
- 提交校验放宽已上传票据条目的 OCR 字段缺失并忽略尾部占位条目
2026-06-17 14:38:07 +08:00

68 lines
2.4 KiB
Python

from __future__ import annotations
from sqlalchemy import select
from sqlalchemy.orm import selectinload
from app.api.deps import CurrentUserContext
from app.models.employee import Employee
from app.models.financial_record import ExpenseClaim
from app.services.pagination import PageResult, paginate_select
class ExpenseClaimPaginationMixin:
def _claim_list_stmt(self):
return select(ExpenseClaim).options(
selectinload(ExpenseClaim.items),
selectinload(ExpenseClaim.employee).selectinload(Employee.manager),
selectinload(ExpenseClaim.employee).selectinload(Employee.roles),
)
def list_claims_page(
self,
current_user: CurrentUserContext,
*,
page: int | None,
page_size: int | None,
) -> PageResult[ExpenseClaim]:
stmt = self._claim_list_stmt().order_by(
ExpenseClaim.created_at.desc(),
ExpenseClaim.occurred_at.desc(),
)
stmt = self._access_policy.apply_claim_scope(stmt, current_user)
result = paginate_select(self.db, stmt, page=page, page_size=page_size)
self._repair_duplicate_budget_approval_stages(result.items)
self._access_policy.attach_budget_approval_snapshots(result.items)
return result
def list_approval_claims_page(
self,
current_user: CurrentUserContext,
*,
page: int | None,
page_size: int | None,
) -> PageResult[ExpenseClaim]:
stmt = self._claim_list_stmt().order_by(
ExpenseClaim.submitted_at.desc(),
ExpenseClaim.created_at.desc(),
)
stmt = self._access_policy.apply_approval_claim_scope(stmt, current_user)
result = paginate_select(self.db, stmt, page=page, page_size=page_size)
self._repair_duplicate_budget_approval_stages(result.items)
self._access_policy.attach_budget_approval_snapshots(result.items)
return result
def list_archived_claims_page(
self,
current_user: CurrentUserContext,
*,
page: int | None,
page_size: int | None,
) -> PageResult[ExpenseClaim]:
stmt = self._claim_list_stmt().order_by(
ExpenseClaim.updated_at.desc(),
ExpenseClaim.submitted_at.desc(),
ExpenseClaim.created_at.desc(),
)
stmt = self._access_policy.apply_archived_claim_scope(stmt, current_user)
return paginate_select(self.db, stmt, page=page, page_size=page_size)