Files
junhong_cmp_fiber/openspec/specs/client-asset-info/spec.md
huang b9733c4913
All checks were successful
构建并部署到测试环境(无 SSH) / build-and-deploy (push) Successful in 7m12s
fix: 修正零售价架构错误 + 清理旧微信配置 + 归档提案 + 前端接口文档
1. 修正 retail_price 架构:
   - 删除 batch-pricing 接口的 pricing_target 字段和 retail_price 分支
     (上级只能改下级成本价,不能改零售价)
   - 新增 PATCH /api/admin/packages/:id/retail-price 接口
     (代理自己改自己的零售价,校验 retail_price >= cost_price)

2. 清理旧微信 YAML 配置(已全部迁移到数据库 tb_wechat_config):
   - 删除 config.yaml 中 wechat.official_account 配置节
   - 删除 NewOfficialAccountApp() 旧工厂函数
   - 清理 personal_customer service 中的死代码(旧登录/绑定微信方法)
   - 清理 docker-compose.prod.yml 中旧微信环境变量和证书挂载注释

3. 归档四个已完成提案到 openspec/changes/archive/

4. 新增前端接口变更说明文档(docs/前端接口变更说明.md)

5. 修正归档提案和 specs 中关于 pricing_target 的错误描述
2026-03-19 17:39:43 +08:00

3.7 KiB
Raw Blame History

Capability: 客户端资产信息

ADDED Requirements

Requirement: B1 资产基本信息查询接口

系统 SHALL 提供 GET /api/c/v1/asset/info?identifier=xxx,并且 MUST 要求个人客户认证C 端 Token。接口 MUST 复用 asset.Service.Resolve() 解析标识符,并在调用时使用 gorm.SkipDataPermission(ctx) 以绕过 shop_id 数据权限过滤。请求参数 MUST 包含 identifierICCID、虚拟号、设备号之一。响应体 SHALL 返回 asset_typeasset_ididentifiervirtual_nostatusreal_name_statuscarriergenerationwallet_balance。错误码/消息 MUST 至少包含:INVALID_PARAM/参数错误FORBIDDEN/无权限操作该资产或资源不存在ASSET_NOT_FOUND/资产不存在

Scenario: 个人客户查询已绑定资产

  • WHEN 客户携带有效 Token 调用 GET /api/c/v1/asset/info?identifier=8986xxxx 且资产已绑定到本人
  • THEN 系统返回 200包含资产基础信息与当前 generation

Requirement: B2 可购买套餐列表接口

系统 SHALL 提供 GET /api/c/v1/asset/packages?identifier=xxx,并且 MUST 要求个人客户认证。接口 MUST 在归属校验通过后返回可购买套餐列表。价格规则 MUST 为:代理渠道取 allocation.retail_price,平台渠道取 Package.SuggestedRetailPrice。过滤规则 MUST 同时满足:Package.status=1shelf_status 可售、加油包前置主套餐条件成立、retail_price >= cost_price。结果 MUST 按展示价格升序。响应体 SHALL 包含 packages[],每项至少含 package_idpackage_namepackage_typeretail_pricecost_pricevalidityis_addon。错误码/消息 MUST 至少包含:INVALID_PARAM/参数错误FORBIDDEN/无权限操作该资产或资源不存在PACKAGE_NOT_AVAILABLE/当前无可购买套餐

Scenario: 代理渠道价格与过滤生效

  • WHEN 客户查询可购套餐且其销售链路为代理渠道,部分套餐存在 retail_price < cost_price
  • THEN 系统仅返回可售且满足价格约束的套餐,并按价格升序输出

Requirement: B3 历史套餐列表接口

系统 SHALL 提供 GET /api/c/v1/asset/package-history?identifier=xxx&page=1&page_size=20,并且 MUST 要求个人客户认证。接口 MUST 基于标识符解析资产并进行归属校验。查询条件 MUST 自动追加 generation = 资产当前generation。请求参数 SHALL 支持 pagepage_size(默认 20最大 100。响应体 SHALL 返回 list[]totalpagepage_size,列表项复用 dto.AssetPackageResponse 结构。错误码/消息 MUST 至少包含:INVALID_PARAM/参数错误FORBIDDEN/无权限操作该资产或资源不存在

Scenario: 转手后历史隔离

  • WHEN 资产已发生转手且存在历史套餐记录
  • THEN 系统只返回当前 generation 的记录,不返回旧 generation 数据

Requirement: B4 手动刷新接口

系统 SHALL 提供 POST /api/c/v1/asset/refresh,并且 MUST 要求个人客户认证。请求体 MUST 包含 identifier。当资产为卡时 MUST 调用 Gateway 刷新卡信息;当资产为设备时 MUST 先检查 Redis 冷却窗口,再对设备下卡执行批量刷新。响应体 SHALL 返回 refresh_typecard/device)、acceptedcooldown_seconds(设备场景)。错误码/消息 MUST 至少包含:INVALID_PARAM/参数错误FORBIDDEN/无权限操作该资产或资源不存在TOO_MANY_REQUESTS/刷新过于频繁,请稍后重试GATEWAY_ERROR/网关调用失败

Scenario: 设备刷新冷却拦截

  • WHEN 客户在冷却时间内重复调用设备刷新
  • THEN 系统返回频率限制错误并告知剩余冷却时间