Input: Design documents from /specs/010-order-invoice/
Prerequisites: plan.md ✅, spec.md ✅, research.md ✅, data-model.md ✅, contracts/api.md ✅, quickstart.md ✅
Tests: 轻量。仅对「开票金额拆分 + ezPay 回应判读」加可选单测(mock EzPay.issueInvoice),其余按 quickstart.md 手测。项目无强制 TDD。
范围说明: 客户端前端 UI 不在本期——本期交付后端开票 API(applyInvoice/getInvoice,供客户端团队后续对接)+ 管理端(商家端 + 平台后台)查看/重试/作废。客户端 UI 与其 i18n 由客户端团队另行实现,不在本 tasks 内。
Organization: 按 spec.md 五个 user story 组织;US1–US3 为后端开票链路,US4–US5 为管理端。
[ID] [P?] [Story] Descriptionfoodie_server):
ruoyi-system/src/main/java/com/ruoyi/system/domain/(含 dto/、vo/)ruoyi-system/src/main/java/com/ruoyi/system/mapper/ruoyi-system/src/main/resources/mapper/chanting/ruoyi-admin/src/main/java/com/ruoyi/app/order/(本期放 admin,因需调 EzPay)ruoyi-admin/src/main/java/com/ruoyi/app/order/UserOrderController.java(已有,新增接口)ruoyi-admin/src/main/java/com/ruoyi/app/mendian/(新增 PosOrderInvoiceController)ruoyi-admin/src/main/java/com/ruoyi/app/utils/ezPay/PosStoreEzpay / PosStoreEzpayMapperupdatesql/sql.mdE:\QtwCode\foodie\foodie-admin-vue):src/api/chanting/、src/views/mendian/、src/api/language/(CRLF 换行,用 Python 脚本改)E:\QtwCode\foodie\foodie-store):src/lang/、订单详情视图(CRLF 换行,用 Python 脚本改)Purpose: SQL 变更与权限菜单(交开发者手动执行)
updatesql/sql.md 追加 2026-06-16 段落:① 按 data-model.md 创建 pos_order_invoice 表(含 uk_order_id 唯一键、idx_status、idx_store 索引);② sys_menu 插入「订单发票管理」(C, chanting:orderInvoice:list) + 按钮权限 chanting:orderInvoice:query/retry/invalid(参考 009 菜单 SQL 写法,挂在门店/订单相关父菜单下)Purpose: 数据模型地基,所有 user story 都依赖
⚠️ CRITICAL: 本阶段完成前不得开始任何 user story
PosOrderInvoice 实体于 ruoyi-system/src/main/java/com/ruoyi/system/domain/PosOrderInvoice.java(MyBatis-Plus @TableName("pos_order_invoice")、@TableId(type=IdType.AUTO),字段见 data-model.md:id/orderId/orderNo/storeId/invoiceCategory/buyerName/buyerUbn/buyerEmail/carrierType/carrierNum/invoiceNumber/randomNum/invoiceStatus/totalAmt/salesAmt/taxAmt/failReason/invoiceUrl/applyTime/issueTime/invalidTime/createTime/updateTime/createBy/updateBy,lombok @Data)ApplyInvoiceDto 于 ruoyi-system/src/main/java/com/ruoyi/system/domain/dto/ApplyInvoiceDto.java(字段:orderId/category/buyerName/buyerUbn/buyerEmail/carrierType/carrierNum;校验注解:orderId 必填、category 必填∈{B2C,B2B}、buyerName 必填、buyerEmail 邮箱格式、buyerUbn 长度 8 数字)PosOrderInvoiceVo 于 ruoyi-system/src/main/java/com/ruoyi/system/domain/vo/PosOrderInvoiceVo.java(列表/详情字段:订单号/门店名/发票类型/买方/发票号/状态/金额/各时间 + 查询筛选字段 invoiceStatus/orderNo/storeId/invoiceCategory/日期范围)PosOrderInvoiceMapper 接口于 ruoyi-system/src/main/java/com/ruoyi/system/mapper/PosOrderInvoiceMapper.java,继承 MyBatis-Plus BaseMapper<PosOrderInvoice>,声明 selectInvoiceList(PosOrderInvoiceVo query) 分页查询 + selectInvoiceDetail(@Param("orderId") Long) 单行PosOrderInvoiceMapper.xml 于 ruoyi-system/src/main/resources/mapper/chanting/PosOrderInvoiceMapper.xml:selectInvoiceList(pos_order_invoice i LEFT JOIN pos_order o LEFT JOIN pos_store s,取订单号/门店名/发票字段,支持 invoiceStatus/orderNo/storeId/invoiceCategory/日期范围筛选)、selectInvoiceDetail(单行,含买方/载具/金额/凭证详情)OrderInvoiceService 于 ruoyi-admin/src/main/java/com/ruoyi/app/order/OrderInvoiceService.java(@Service,注入 EzPay、PosOrderMapper(或 PosOrderService)、PosStoreEzpayMapper、PosStoreMapper、PosOrderInvoiceMapper),声明占位方法 applyInvoice/ getInvoice/ list/ detail/ retry/ invalid,具体业务在后续 story 实现。说明:本期 service 放 admin(需调 EzPay),与 009 不同Checkpoint: 表结构 + 实体 + DTO/VO + mapper + service 骨架就绪,user story 可开始
Goal: 提供客户端可调用的开票 API——客户对已完成订单申请 B2C 个人发票(邮箱),系统调 ezPay 即时开立、记录发票号,并可查询
Independent Test: 已完成订单(所属门店 ezPay 已开通启用)→ 调 POST /system/userOrder/applyInvoice(B2C+邮箱)→ 开立成功 → 调 GET /getInvoice 返回发票号
OrderInvoiceService.applyInvoice(ApplyInvoiceDto dto, Long currentUserId)(核心开票链路):① 校验订单归属(pos_order.user_id == currentUserId)、state==3(已完成)、pay_status==1(已支付);② 校验门店可开票:pos_store_ezpay.ezpay_status==2 && is_enabled==1 且 pos_store.invoice_exempt==0,否则抛「门店暂不支持开票」;③ 校验订单未开票(invoice_status != 1);④ 金额拆分(research D1):invoiceTotal = amount - freight、sales = round(invoiceTotal/1.05)、tax = invoiceTotal - sales;⑤ 组装 ezPay issue 参数:Category=B2C、PrintFlag=Y、BuyerName、BuyerEmail、TaxType=1、TaxRate=5、Amt/TaxAmt/TotalAmt、MerchantOrderNo=ddId,逐商品明细从 food JSON 解析(ItemName/ItemPrice=price+otherPrice 含税/ItemCount=number/ItemAmt),优惠按比例分摊到各商品(research D3 方案 B);⑥ EzPayConfig 由门店 pos_store_ezpay 凭证构造 → ezPay.issueInvoice(baseUrl, cfg, postData);⑦ 判读(research D9):Status==SUCCESS && InvoiceNumber 非空 → 落库 invoice_number/random_num/invoice_url/total/sales/tax/invoice_status=1/issue_time;否则 invoice_status=2 + fail_reason;⑧ 网络异常 → invoice_status=2 + "ezPay 服务暂不可用"、订单状态不变;⑨ uk_order_id 防重复(catch DuplicateKeyException)。依赖 T002、T005、T007OrderInvoiceService.getInvoice(Long orderId, Long currentUserId):校验订单归属,返回发票 VO(invoiceStatus/invoiceNumber/invoiceUrl/category);门店不可开票时 invoiceStatus=null(客户端据此隐藏入口)。依赖 T008UserOrderController(/system/userOrder)新增:POST /applyInvoice(@RequestBody ApplyInvoiceDto,从 JWT/当前用户取 userId 调 applyInvoice)、GET /getInvoice?orderId=(调 getInvoice)。依赖 T008、T009Checkpoint: US1 可独立验收——开票/查询 API 可用,curl/Postman 跑通 B2C 开票
Goal: 开票 API 支持 B2B 公司发票(统编),提交前校验统编格式
Independent Test: applyInvoice B2B + 合法统编 → 开立含买方统编;非法统编 → 调 ezPay 前被拦截
applyInvoice 增加 B2B 分支:Category=B2B、BuyerUBN=统编(8码)、PrintFlag=Y、BuyerName=公司名;B2B 时 buyerUbn 必填且 8 位数字校验(service 内强校验,非法抛「统编格式不正确」不调 ezPay)。依赖 T008Checkpoint: US2 可独立验收——B2B 开票 API + 统编校验双分支
Goal: 开票 API 支持选择载具(手机条码/自然人凭证/会员载具),发票存入载具
Independent Test: applyInvoice 选手机条码载具 → 校验后开立、PrintFlag=N、发票存入载具
applyInvoice 增加载具分支:carrierType 非空时传 CarrierType(0/1/2)+CarrierNum、PrintFlag=N、LoveCode 空;校验载具号非空(手机条码格式 /XXXX+/ 可选校验)。依赖 T008Checkpoint: US3 可独立验收——载具开票 API
Goal: 商家端/平台后台查看订单开票状态与发票号;运营对失败单重新开票
Independent Test: 一笔开票失败订单 → 平台后台点「重新开票」 → 成功、状态变已开
OrderInvoiceService 的管理端方法:list(query)(分页,带门店/状态筛选)、detail(orderId)、retry(orderId)(校验 invoice_status ∈ {2 失败, 3 作废} → 重置为未开 → 复用 T008 开票逻辑重开,运营身份不校验客户归属)。依赖 T008PosOrderInvoiceController 于 ruoyi-admin/src/main/java/com/ruoyi/app/mendian/PosOrderInvoiceController.java,@RequestMapping("/system/orderInvoice"):GET /list(@PreAuthorize("@ss.hasPermi('chanting:orderInvoice:list')") + startPage() + TableDataInfo)、GET /{orderId}(chanting:orderInvoice:query)、PUT /retry/{orderId}(chanting:orderInvoice:retry + @Log)。依赖 T013E:\QtwCode\foodie\foodie-admin-vue\src\api\chanting\orderInvoice.js(list/detail/retry 封装,走 /system/orderInvoice/*)与页面 src\views\mendian\orderInvoice\index.vue(Element 表格:订单号/门店/类型/买方/发票号/状态/金额/开立时间 + 顶部筛选 + 失败/作废行的「重新开票」按钮,v-hasPermi 权限控制)。依赖 T014foodie-admin-vue\src\api\language\ 的 zh_CN/zh_TW/en_US/vi.js 新增 orderInvoice:{} 对象层级(标题/表头/状态文案/按钮,四文件 key 一致)。依赖 T015foodie-store:订单详情显示发票状态/发票号(复用现有商家订单详情视图,加展示)。依赖 T014(接口)foodie-store\src\lang\ 的 zh/tw/en/vi.js 加发票状态相关 key(四语言一致,CRLF 用 Python 脚本改)。依赖 T017Checkpoint: US4 可独立验收——三端查看 + 失败重试
Goal: 已开立发票可作废(ezPay invalid),作废后订单可重开
Independent Test: 作废一张本月开具的发票 → ezPay invalid 成功 → 状态作废 → 重新开票生成新号
OrderInvoiceService.invalid(Long orderId, String reason):校验 invoice_status==1(已开),构造 {InvoiceNumber, InvalidReason, RespondType:JSON, Version:1.0} 调 ezPay.doPost(baseUrl + EzPay.URL_INVALID, cfg, postData);判读成功 → invoice_status=3/invalid_time;失败 → 记 fail_reason;不可作废以 ezPay 返回为准(不做本地时限计算)。依赖 T014PosOrderInvoiceController 加 PUT /invalid/{orderId}(入参可选 {invalidReason},@PreAuthorize('chanting:orderInvoice:invalid') + @Log)。依赖 T019index.vue 对「已开」行加「作废」按钮 + 作废原因输入弹窗(调 invalid,成功刷新)。依赖 T015、T020orderInvoice:{} 下补充作废相关 key(作废/作废原因/不可作废提示)。依赖 T021Checkpoint: US5 可独立验收——作废 + 重开
quickstart.md 跑通手测(通过 curl/Postman 调开票 API + 平台后台 UI):执行 SQL → ezPay 测试环境真凭证 → B2C 邮箱 / B2B 统编 / 载具开票 → 失败重试 → 作废重开 → 优惠对账(方案 B 分摊),逐条核对 spec.md 的 SC-001~SC-006OrderInvoiceServiceTest(mock EzPay.issueInvoice)验证 ① 金额拆分(invoiceTotal=amount-freight、sales+tax=total);② 回应判读两条分支(SUCCESS→已开 / 其它→失败 status 不变)。按项目测试习惯放 ruoyi-admin/src/test/...C:\Users\qmj\.claude\projects\E--QtwCode-foodie-foodie-server\memory\MEMORY.md 加一行指向 specs/010-order-invoice/spec.md(参考 008/009 的写法)orderInvoice/index.vue 跨 US4/US5 同文件 → 顺序追加applyInvoice(B2C 邮箱)→ 开票成功、getInvoice 返回发票号/system/userOrder/applyInvoice、/getInvoice),客户端团队后续对接 UI;US1–US3 验收以 API 手测(curl/Postman)为准EzPay/EzPayConfig/EzPayEncryptUtil 已实现并经官方数据验证,本期不改其内部,仅业务层调用(issue 开票 / doPost 作废)invoiceTotal = amount - freight(research D1),各类优惠已含在 amount 内无需逐项区分ruoyi-admin.app.order(非 ruoyi-system),因需调 EzPay;domain/mapper 放 ruoyi-systemupdatesql/sql.md 由开发者手动跑