新增钱包、换卡、标签系统的数据模型和规范

本次提交完成 add-wallet-transfer-tag-models 提案的实施和归档:

## 新增功能模块
- 钱包系统:用户/代理钱包管理,支持充值、扣款、退款、乐观锁防并发
- 换卡记录:物联卡更换历史追溯,包含套餐快照(JSONB)
- 标签系统:设备/IoT卡/号卡的统一标签管理
- 运营商渠道:四大运营商(CMCC/CUCC/CTCC/CBN)的渠道管理

## 数据库变更
- 新增 6 张表:tb_wallet, tb_wallet_transaction, tb_recharge_record, tb_card_replacement_record, tb_tag, tb_resource_tag
- 修改 2 张表:tb_carrier(新增渠道字段), tb_order(新增混合支付字段)
- 迁移版本:v6 → v7(执行时间 282.5ms)

## 代码变更
- 新增 8 个 Go 模型(符合统一规范:gorm.Model + BaseModel)
- 新增 40+ 个常量定义(含完整中文注释)
- 新增 7 个 Redis Key 生成函数
- 修复模型规范:移除重复字段,统一使用 gorm.Model 嵌入

## 文档变更
- 新增 3 个业务文档:数据模型设计、字段说明、迁移验证报告
- 更新 AGENTS.md:新增 Model 模型规范和常量注释规范
- 新增 4 个 OpenSpec 规范:wallet, carrier, card-replacement, tag
- 更新 1 个 OpenSpec 规范:iot-order(支持混合支付)

## 验证通过
-  LSP 诊断:所有模型和常量文件无错误
-  OpenSpec 验证:openspec validate --strict 通过
-  迁移执行:表结构创建成功,索引正确
-  提案归档:2026-01-13-add-wallet-transfer-tag-models

变更文件统计:29 个文件,新增 3682 行
This commit is contained in:
2026-01-13 15:47:32 +08:00
parent 2150fb6ab9
commit 6e2dc325d7
29 changed files with 3682 additions and 87 deletions

View File

@@ -0,0 +1,187 @@
# card-replacement Specification
## Purpose
TBD - created by archiving change add-wallet-transfer-tag-models. Update Purpose after archive.
## Requirements
### Requirement: 换卡记录实体定义
系统 SHALL 定义换卡记录(CardReplacementRecord)实体,记录老卡到新卡的完整转移过程,包括套餐权益、代理关系、所有者信息等。
**核心概念**
- **换卡场景**:老卡损坏、丢失或故障,需要更换新卡
- **权益转移**:老卡的套餐(含剩余流量)、代理关系、所有者信息等全部转移到新卡
- **套餐继续生效**:转移后套餐不作废,剩余流量继续可用
**实体字段**
- `id`:换卡记录 ID主键BIGINT
- `replacement_no`换卡单号VARCHAR(50),唯一)
- `old_card_id`:老卡 IDBIGINT关联 tb_iot_card.id
- `old_iccid`:老卡 ICCIDVARCHAR(50),冗余存储,防止老卡被删除后无法追踪)
- `new_card_id`:新卡 IDBIGINT关联 tb_iot_card.id
- `new_iccid`:新卡 ICCIDVARCHAR(50),冗余存储)
- `old_owner_type`老卡所有者类型VARCHAR(20)
- `old_owner_id`:老卡所有者 IDBIGINT
- `old_agent_id`:老卡代理 IDBIGINT可空
- `new_owner_type`新卡所有者类型VARCHAR(20)
- `new_owner_id`:新卡所有者 IDBIGINT
- `new_agent_id`:新卡代理 IDBIGINT可空
- `package_snapshot`套餐快照JSONB记录转移时的套餐详情
- `replacement_reason`换卡原因VARCHAR(20),枚举值:"damaged"-损坏 | "lost"-丢失 | "malfunction"-故障 | "upgrade"-升级 | "other"-其他)
- `remark`备注TEXT
- `status`换卡状态INT1-待审批 2-已通过 3-已拒绝 4-已完成)
- `approved_by`:审批人 IDBIGINT可空
- `approved_at`审批时间TIMESTAMP可空
- `completed_at`完成时间TIMESTAMP可空
- `creator`:创建人 IDBIGINT
- `updater`:更新人 IDBIGINT
- `created_at`创建时间TIMESTAMP自动填充
- `updated_at`更新时间TIMESTAMP自动填充
- `deleted_at`删除时间TIMESTAMP可空软删除
**套餐快照 JSON 格式示例**
```json
{
"package_id": 3001,
"package_name": "月套餐 10GB",
"package_code": "PKG-M-001",
"data_limit_mb": 10240,
"data_usage_mb": 5120,
"real_data_usage_mb": 4000,
"virtual_data_usage_mb": 1120,
"data_remaining_mb": 5120,
"activated_at": "2026-01-01T00:00:00Z",
"expires_at": "2026-02-01T00:00:00Z",
"remaining_days": 15,
"order_id": 10001
}
```
#### Scenario: 创建换卡记录
- **WHEN** 用户ID 为 2001的老卡ICCID 为 "8986001"损坏需要换新卡ICCID 为 "8986002"
- **THEN** 系统创建换卡记录,`old_card_id` 为老卡 ID`new_card_id` 为新卡 ID`replacement_reason` 为 "damaged"`status` 为 1待审批
#### Scenario: 审批通过换卡
- **WHEN** 运营人员ID 为 999审批通过换卡记录ID 为 5001
- **THEN** 系统将换卡记录状态从 1待审批变更为 2已通过记录 `approved_by` 为 999`approved_at` 为当前时间
#### Scenario: 完成换卡
- **WHEN** 换卡记录ID 为 5001状态为 2已通过系统执行换卡操作
- **THEN** 系统将:
1. 记录老卡和新卡的快照信息(所有者、代理、套餐)
2. 将老卡的套餐权益转移到新卡(套餐使用记录的 `iot_card_id` 更新为新卡 ID
3. 将新卡的 `owner_type``owner_id` 更新为老卡的值
4. 将新卡的代理关系更新为老卡的值(如有)
5. 将换卡记录状态变更为 4已完成记录 `completed_at` 为当前时间
#### Scenario: 拒绝换卡
- **WHEN** 运营人员ID 为 999拒绝换卡记录ID 为 5001原因为"新卡不符合要求"
- **THEN** 系统将换卡记录状态从 1待审批变更为 3已拒绝记录 `approved_by` 为 999`approved_at` 为当前时间,`remark` 为拒绝原因
---
### Requirement: 套餐权益转移
系统 SHALL 在换卡完成后,将老卡的套餐权益(包括剩余流量、过期时间等)转移到新卡,套餐继续生效。
**转移内容**
- 套餐使用记录(`tb_package_usage`
- 剩余流量(`data_limit_mb - data_usage_mb`
- 套餐过期时间(`expires_at`
- 关联的订单信息
**转移规则**
- 老卡的套餐使用记录的 `iot_card_id` 更新为新卡 ID
- 剩余流量完整保留
- 套餐过期时间不变
- 如果老卡有多个套餐(正式套餐 + 加油包),全部转移
#### Scenario: 套餐转移
- **WHEN** 老卡有月套餐(剩余 5120 MB 流量,还有 15 天过期)
- **THEN** 系统将套餐使用记录的 `iot_card_id` 从老卡 ID 更新为新卡 ID流量和过期时间保持不变
#### Scenario: 多套餐转移
- **WHEN** 老卡有正式套餐和 2 个加油包
- **THEN** 系统将所有套餐使用记录的 `iot_card_id` 更新为新卡 ID所有套餐继续生效
---
### Requirement: 代理关系转移
系统 SHALL 在换卡完成后,将老卡的代理关系转移到新卡。
**转移内容**
- 新卡的 `owner_type` 更新为老卡的 `owner_type`
- 新卡的 `owner_id` 更新为老卡的 `owner_id`
- 如果老卡通过代理销售,新卡继承相同的代理关系
#### Scenario: 代理关系转移
- **WHEN** 老卡的 `owner_type` 为 "agent"`owner_id` 为 123
- **THEN** 系统将新卡的 `owner_type` 更新为 "agent"`owner_id` 更新为 123
---
### Requirement: 换卡记录查询
系统 SHALL 支持按老卡 ID、新卡 ID、用户 ID、换卡单号等条件查询换卡记录。
**查询条件**
- 换卡单号(精确匹配)
- 老卡 ID精确匹配
- 新卡 ID精确匹配
- 老卡 ICCID精确匹配或模糊匹配
- 新卡 ICCID精确匹配或模糊匹配
- 换卡状态(单选或多选)
- 换卡原因(单选或多选)
- 创建时间范围
- 完成时间范围
**分页**
- 默认每页 20 条,最大每页 100 条
- 返回总记录数和总页数
#### Scenario: 按老卡 ICCID 查询换卡记录
- **WHEN** 查询老卡 ICCID 为 "8986001" 的换卡记录
- **THEN** 系统返回所有 `old_iccid` 为 "8986001" 的换卡记录列表
#### Scenario: 按状态查询换卡记录
- **WHEN** 查询状态为 1待审批的换卡记录
- **THEN** 系统返回所有 `status` 为 1 的换卡记录列表,按创建时间倒序排列
---
### Requirement: 换卡数据校验
系统 SHALL 对换卡数据进行校验,确保数据完整性和一致性。
**校验规则**
- `old_card_id`:必填,≥ 1必须是有效的 IoT 卡 ID
- `new_card_id`:必填,≥ 1必须是有效的 IoT 卡 ID不能与 `old_card_id` 相同
- `old_iccid`:必填,长度 19-20 字符
- `new_iccid`:必填,长度 19-20 字符,不能与 `old_iccid` 相同
- `replacement_reason`:必填,枚举值 "damaged" | "lost" | "malfunction" | "upgrade" | "other"
- `status`:必填,枚举值 1-4
#### Scenario: 换卡时老卡和新卡相同
- **WHEN** 创建换卡记录,`old_card_id``new_card_id` 都为 1001
- **THEN** 系统拒绝创建,返回错误信息"新卡不能与老卡相同"
#### Scenario: 换卡时新卡 ICCID 无效
- **WHEN** 创建换卡记录,`new_iccid` 长度为 15小于 19
- **THEN** 系统拒绝创建,返回错误信息"ICCID 长度必须为 19-20 字符"
#### Scenario: 换卡时老卡不存在
- **WHEN** 创建换卡记录,`old_card_id` 为 99999不存在的 IoT 卡)
- **THEN** 系统拒绝创建,返回错误信息"老卡不存在"

View File

@@ -0,0 +1,80 @@
# carrier Specification
## Purpose
TBD - created by archiving change add-wallet-transfer-tag-models. Update Purpose after archive.
## Requirements
### Requirement: 运营商实体定义
系统 SHALL 定义运营商(Carrier)实体,管理四大固定运营商(中国移动、中国联通、中国电信、广电)的渠道信息
**四大运营商固定枚举**
- **CMCC**:中国移动
- **CUCC**:中国联通
- **CTCC**:中国电信
- **CBN**:广电
**实体字段**
- `id`:运营商 ID主键BIGINT
- `carrier_type`运营商类型VARCHAR(20),枚举值:"CMCC" | "CUCC" | "CTCC" | "CBN"**【新增】**
- `carrier_name`运营商名称VARCHAR(100),如"中国移动"
- `carrier_code`运营商编码VARCHAR(50),保留字段,建议填充与 carrier_type 相同)
- `channel_name`渠道名称VARCHAR(100),可自定义,如"北京渠道1"**【新增】**
- `channel_code`渠道编码VARCHAR(50),可自定义,如"BJ001"**【新增】**
- `status`状态INT1-启用 2-禁用)
- `creator`:创建人 IDBIGINT
- `updater`:更新人 IDBIGINT
- `created_at`创建时间TIMESTAMP自动填充
- `updated_at`更新时间TIMESTAMP自动填充
- `deleted_at`删除时间TIMESTAMP可空软删除
**唯一约束**`(carrier_type, channel_code)``deleted_at IS NULL` 条件下唯一
#### Scenario: 创建中国移动的渠道
- **WHEN** 平台创建中国移动的北京渠道,`carrier_type` 为 "CMCC"`carrier_name` 为 "中国移动"`channel_name` 为 "北京渠道1"`channel_code` 为 "BJ001"
- **THEN** 系统创建运营商记录,`carrier_type` 为 "CMCC"`channel_name` 为 "北京渠道1"`channel_code` 为 "BJ001"
#### Scenario: 同一运营商创建多个渠道
- **WHEN** 平台为中国移动创建两个渠道北京渠道BJ001和上海渠道SH001
- **THEN** 系统创建两条运营商记录,`carrier_type` 都为 "CMCC",但 `channel_code` 不同
#### Scenario: 渠道编码重复
- **WHEN** 平台创建中国移动的渠道,`carrier_type` 为 "CMCC"`channel_code` 为已存在的 "BJ001"
- **THEN** 系统拒绝创建,返回错误信息"该运营商的渠道编码已存在"
#### Scenario: 不同运营商可以使用相同渠道编码
- **WHEN** 平台为中国移动创建渠道carrier_type=CMCC, channel_code=BJ001然后为中国联通创建渠道carrier_type=CUCC, channel_code=BJ001
- **THEN** 系统允许创建,因为 `carrier_type` 不同
#### Scenario: 运营商类型枚举限制
- **WHEN** 平台创建运营商,`carrier_type` 为 "OTHER"(不在枚举中)
- **THEN** 系统拒绝创建,返回错误信息"运营商类型必须是 CMCC/CUCC/CTCC/CBN 之一"
---
### Requirement: 运营商数据校验
系统 SHALL 对运营商数据进行校验,确保数据完整性和一致性。
**校验规则**
- `carrier_type`:必填,枚举值 "CMCC" | "CUCC" | "CTCC" | "CBN"
- `carrier_name`:必填,长度 1-100 字符
- `carrier_code`:必填,长度 1-50 字符
- `channel_name`:可选,长度 1-100 字符
- `channel_code`:可选,长度 1-50 字符
- `status`:必填,枚举值 1-2
#### Scenario: 创建运营商时 carrier_type 无效
- **WHEN** 创建运营商,`carrier_type` 为 "INVALID"
- **THEN** 系统拒绝创建,返回错误信息"运营商类型无效"
#### Scenario: 创建运营商时 carrier_name 为空
- **WHEN** 创建运营商,`carrier_name` 为空
- **THEN** 系统拒绝创建,返回错误信息"运营商名称不能为空"

View File

@@ -11,56 +11,50 @@ This capability supports:
- Commission triggering on order completion
- Device-level order commission (counted once regardless of bound card count)
- Multi-dimensional order querying and filtering
## Requirements
### Requirement: 订单实体定义
系统 SHALL 定义订单(Order)实体,统一管理两种订单类型:套餐订单、号卡订单。
系统 SHALL 定义订单(Order)实体统一管理两种订单类型套餐订单、号卡订单,并支持混合支付方式(钱包 + 在线支付)
**核心概念**:
- **套餐订单**: 用户为 IoT 卡或设备购买套餐的订单,包括单卡套餐订单和设备级套餐订单
- **号卡订单**: 运营商回传的号卡订单,用户直接在上游平台下单,系统只接收订单状态更新
**修改说明**
- 增加 `wallet_payment_amount` 字段:钱包支付金额
- 增加 `online_payment_amount` 字段:在线支付金额
- 支持用户在购买套餐时选择支付方式(全部钱包支付、全部在线支付、混合支付)
**实体字段**:
- `id`: 订单 ID(主键,BIGINT)
- `order_no`: 订单编号(VARCHAR(50),唯一)
- `order_type`: 订单类型(INT,1-套餐订单 2-号卡订单)
- `iot_card_id`: IoT 卡 ID(BIGINT,可空,单卡套餐订单时有值)
- `device_id`: 设备 ID(BIGINT,可空,设备级套餐订单时有值)
- `number_card_id`: 号卡 ID(BIGINT,可空,号卡订单时有值)
- `package_id`: 套餐 ID(BIGINT,可空,仅当 order_type 为 1 时有值)
- `user_id`: 用户 ID(BIGINT,购买用户)
- `agent_id`: 代理 ID(BIGINT,可空,通过代理购买时有值)
- `amount`: 订单金额(DECIMAL(10,2),元)
- `payment_method`: 支付方式(VARCHAR(20),"wallet"-钱包 | "online"-在线支付 | "carrier"-运营商直付)
- `status`: 订单状态(INT,1-待支付 2-已支付 3-已完成 4-已取消 5-已退款)
- `carrier_order_id`: 运营商订单 ID(VARCHAR(255),可空,仅号卡订单有值)
- `carrier_order_data`: 运营商订单原始数据(JSONB,可空)
- `paid_at`: 支付时间(TIMESTAMP,可空)
- `completed_at`: 完成时间(TIMESTAMP,可空)
- `created_at`: 创建时间(TIMESTAMP,自动填充)
- `updated_at`: 更新时间(TIMESTAMP,自动填充)
**实体字段**(只列出新增字段):
- `wallet_payment_amount`钱包支付金额BIGINT单位默认 0**【新增】**
- `online_payment_amount`在线支付金额BIGINT单位默认 0**【新增】**
**订单类型说明**:
- **单卡套餐订单**: `order_type` 为 1,`iot_card_id` 有值,`device_id` 为 NULL
- **设备级套餐订单**: `order_type` 为 1,`device_id` 有值,`iot_card_id` 为 NULL
- **号卡订单**: `order_type` 为 2,`number_card_id` 有值,`iot_card_id``device_id` 为 NULL
**支付规则**
- `wallet_payment_amount` + `online_payment_amount` = `amount`(订单总金额)
- `payment_method` 为 "wallet" 时,`wallet_payment_amount` = `amount``online_payment_amount` = 0
- `payment_method` 为 "online" 时,`online_payment_amount` = `amount``wallet_payment_amount` = 0
- 混合支付时,`payment_method` 为 "mixed",两个字段都 > 0
#### Scenario: 创建单卡套餐购买订单
#### Scenario: 全额钱包支付
- **WHEN** 用户(ID 为 2001)为 IoT 卡(ID 为 1001)购买套餐(ID 为 3001),金额为 30.00
- **THEN** 系统创建订单记录,`order_type` 为 1,`iot_card_id` 为 1001,`device_id` 为 NULL,`package_id` 为 3001,`user_id` 为 2001,`amount` 为 30.00,状态为 1(待支付)
- **WHEN** 用户购买套餐,订单金额为 30 00 分30 元),选择钱包支付,钱包余额为 10000
- **THEN** 系统创建订单`amount` 为 3000`payment_method` 为 "wallet"`wallet_payment_amount` 为 3000`online_payment_amount` 为 0
#### Scenario: 创建设备级套餐购买订单
#### Scenario: 全额在线支付
- **WHEN** 用户(ID 为 2001)为设备(ID 为 5001,绑定 3 张 IoT 卡)购买套餐(ID 为 3002),金额为 399.00 元
- **THEN** 系统创建订单记录,`order_type` 为 1,`device_id` 为 5001,`iot_card_id` 为 NULL,`package_id` 为 3002,`user_id` 为 2001,`amount` 为 399.00,状态为 1(待支付)
- **WHEN** 用户购买套餐,订单金额为 3000 分30 元),选择在线支付
- **THEN** 系统创建订单`amount` 为 3000`payment_method` 为 "online"`wallet_payment_amount` 为 0`online_payment_amount` 为 3000
#### Scenario: 创建号卡订单(运营商回传)
#### Scenario: 混合支付
- **WHEN** Gateway 回传运营商订单,虚拟商品编码对应号卡 ID 6001,代理 ID 为 123,订单金额为 30.00
- **THEN** 系统创建订单记录,`order_type` 为 2,`number_card_id` 为 6001,`iot_card_id` 为 NULL,`device_id` 为 NULL,`agent_id` 为 123,`amount` 为 30.00,`payment_method` 为 "carrier",状态为 2(已支付)
- **WHEN** 用户购买套餐,订单金额5000 分50 元),钱包余额为 3000 分,用户选择钱包支付 3000 分 + 在线支付 2000 分
- **THEN** 系统创建订单`amount` 为 5000`payment_method` 为 "mixed"`wallet_payment_amount` 为 3000`online_payment_amount` 为 2000
#### Scenario: 钱包余额不足,部分钱包支付
- **WHEN** 用户购买套餐,订单金额为 5000 分50 元),钱包余额为 2000 分,用户选择钱包支付 2000 分 + 在线支付 3000 分
- **THEN** 系统先冻结钱包余额 2000 分,创建订单,`wallet_payment_amount` 为 2000`online_payment_amount` 为 3000等待用户完成在线支付
#### Scenario: 钱包余额不足,无法全额钱包支付
- **WHEN** 用户购买套餐,订单金额为 5000 分50 元),钱包余额为 3000 分,用户选择钱包支付
- **THEN** 系统拒绝创建订单,返回错误信息"钱包余额不足",建议用户选择混合支付或在线支付
---
@@ -207,41 +201,75 @@ This capability supports:
### Requirement: 订单数据校验
系统 SHALL 对订单数据进行校验,确保数据完整性和一致性。
系统 SHALL 对订单数据进行校验确保数据完整性和一致性,特别是支付金额的一致性
**校验规则**:
- 订单编号(order_no):必填,长度 1-50 字符,唯一
- 订单类型(order_type):必填,枚举值 1(套餐订单) | 2(号卡订单)
- IoT 卡 ID(iot_card_id):套餐订单时 iot_card_id 和 device_id 二选一
- 设备 ID(device_id):套餐订单时 iot_card_id 和 device_id 二选一
- 号卡 ID(number_card_id):号卡订单时必填
- 套餐 ID(package_id):套餐订单时必填
- 用户 ID(user_id):必填,≥ 1
- 订单金额(amount):必填,≥ 0,最多 2 位小数
- 支付方式(payment_method):必填,枚举值 "wallet" | "online" | "carrier"
- 状态(status):必填,枚举值 1-5
**新增校验规则**
- `wallet_payment_amount`:必填,≥ 0最多精确到分
- `online_payment_amount`:必填,≥ 0最多精确到分
- `wallet_payment_amount` + `online_payment_amount` = `amount`(订单总金额)
- `payment_method` 为 "wallet" 时,`wallet_payment_amount` 必须 = `amount`
- `payment_method` 为 "online" 时,`online_payment_amount` 必须 = `amount`
- `payment_method` 为 "mixed" 时,两个字段都必须 > 0
#### Scenario: 创建订单时金额为负数
#### Scenario: 支付金额不一致
- **WHEN** 创建订单,金额为 -10.00
- **THEN** 系统拒绝创建,返回错误信息"订单金额必须 ≥ 0"
- **WHEN** 创建订单`amount` 为 5000`wallet_payment_amount` 为 2000`online_payment_amount` 为 2000
- **THEN** 系统拒绝创建返回错误信息"支付金额总和与订单金额不一致"
#### Scenario: 创建订单时订单编号重复
#### Scenario: 钱包支付时在线支付金额不为 0
- **WHEN** 创建订单,订单编号为已存在的 "ORD-2025-001"
- **THEN** 系统拒绝创建,返回错误信息"订单编号已存在"
- **WHEN** 创建订单`payment_method` 为 "wallet"`wallet_payment_amount` 为 3000`online_payment_amount` 为 0正确但用户错误地设置 `online_payment_amount` 为 100
- **THEN** 系统拒绝创建返回错误信息"钱包支付时在线支付金额必须为 0"
#### Scenario: 创建套餐订单时未关联 IoT 卡或设备
#### Scenario: 混合支付时钱包支付金额为 0
- **WHEN** 创建套餐订单,`iot_card_id``device_id` NULL
- **THEN** 系统拒绝创建,返回错误信息"套餐订单必须关联 IoT 卡或设备"
- **WHEN** 创建订单`payment_method` 为 "mixed"`wallet_payment_amount` 为 0`online_payment_amount`5000
- **THEN** 系统拒绝创建返回错误信息"混合支付时钱包支付金额和在线支付金额都必须大于 0"
#### Scenario: 创建套餐订单时同时关联 IoT 卡和设备
### Requirement: 订单支付处理
- **WHEN** 创建套餐订单,`iot_card_id` 为 1001,`device_id` 为 5001
- **THEN** 系统拒绝创建,返回错误信息"套餐订单不能同时关联 IoT 卡和设备"
系统 SHALL 根据支付方式正确处理订单支付,包括钱包扣款、在线支付、混合支付等。
#### Scenario: 创建号卡订单时未关联号卡
**钱包支付流程**
1. 检查钱包可用余额是否充足
2. 冻结钱包余额(`frozen_balance` 增加)
3. 创建订单,状态为"待支付"
4. 订单完成后,扣减钱包余额(`balance` 减少,`frozen_balance` 减少),创建钱包明细记录
5. 订单取消时,解冻钱包余额(`frozen_balance` 减少)
**在线支付流程**
1. 创建订单,状态为"待支付"
2. 调用第三方支付接口
3. 用户完成支付后,订单状态变更为"已支付"
4. 订单完成后,订单状态变更为"已完成"
**混合支付流程**
1. 检查钱包可用余额是否充足(钱包支付部分)
2. 冻结钱包余额
3. 创建订单,状态为"待支付"
4. 调用第三方支付接口(在线支付部分)
5. 用户完成在线支付后,扣减钱包余额,订单状态变更为"已支付"
6. 订单完成后,订单状态变更为"已完成"
#### Scenario: 钱包支付订单完成
- **WHEN** 用户使用钱包支付购买套餐,订单金额为 3000 分
- **THEN** 系统:
1. 创建订单,状态为"待支付",冻结钱包余额 3000 分
2. 订单处理完成后,扣减钱包余额 3000 分,解冻 3000 分,创建钱包明细记录(类型为"扣款"),订单状态变更为"已完成"
#### Scenario: 混合支付订单完成
- **WHEN** 用户使用混合支付购买套餐,钱包支付 2000 分 + 在线支付 3000 分
- **THEN** 系统:
1. 创建订单,状态为"待支付",冻结钱包余额 2000 分
2. 用户完成在线支付 3000 分后,扣减钱包余额 2000 分,解冻 2000 分,创建钱包明细记录,订单状态变更为"已支付"
3. 订单处理完成后,订单状态变更为"已完成"
#### Scenario: 订单取消,解冻钱包余额
- **WHEN** 用户使用钱包支付创建订单,订单金额为 3000 分,然后取消订单
- **THEN** 系统解冻钱包余额 3000 分(`frozen_balance` 减少 3000订单状态变更为"已取消"
---
- **WHEN** 创建号卡订单,`number_card_id` 为 NULL
- **THEN** 系统拒绝创建,返回错误信息"号卡订单必须关联号卡"

222
openspec/specs/tag/spec.md Normal file
View File

@@ -0,0 +1,222 @@
# tag Specification
## Purpose
TBD - created by archiving change add-wallet-transfer-tag-models. Update Purpose after archive.
## Requirements
### Requirement: 标签实体定义
系统 SHALL 定义标签(Tag)实体用于为资源设备、IoT卡、号卡提供自定义标签分类功能。
**核心概念**
- 企业用户可以为自己的设备/卡片创建和管理标签
- 标签可以跨资源类型使用(一个标签可以同时用于设备和卡片)
- 支持按标签查询和筛选资源
**实体字段**
- `id`:标签 ID主键BIGINT
- `name`标签名称VARCHAR(100),唯一)
- `color`标签颜色VARCHAR(20),可选,用于前端显示,如 "#FF5733"
- `usage_count`使用次数INT默认 0记录有多少资源使用了该标签
- `creator`:创建人 IDBIGINT
- `updater`:更新人 IDBIGINT
- `created_at`创建时间TIMESTAMP自动填充
- `updated_at`更新时间TIMESTAMP自动填充
- `deleted_at`删除时间TIMESTAMP可空软删除
**唯一约束**`name``deleted_at IS NULL` 条件下唯一
#### Scenario: 创建标签
- **WHEN** 用户创建标签,名称为"生产设备",颜色为"#FF5733"
- **THEN** 系统创建标签记录,`name` 为 "生产设备"`color` 为 "#FF5733"`usage_count` 为 0
#### Scenario: 标签名称重复
- **WHEN** 用户创建标签,名称为已存在的"生产设备"
- **THEN** 系统拒绝创建,返回错误信息"标签名称已存在"
#### Scenario: 更新标签
- **WHEN** 用户更新标签ID 为 101将颜色从"#FF5733"改为"#33FF57"
- **THEN** 系统更新标签记录,`color` 为 "#33FF57"`updated_at` 为当前时间
---
### Requirement: 资源-标签关联
系统 SHALL 定义资源-标签关联(ResourceTag)实体建立资源与标签的多对多关系统一管理设备、IoT卡、号卡的标签。
**实体字段**
- `id`:关联记录 ID主键BIGINT
- `resource_type`资源类型VARCHAR(20),枚举值:"device"-设备 | "iot_card"-IoT卡 | "number_card"-号卡)
- `resource_id`:资源 IDBIGINT
- `tag_id`:标签 IDBIGINT关联 tb_tag.id
- `creator`:创建人 IDBIGINT
- `updater`:更新人 IDBIGINT
- `created_at`创建时间TIMESTAMP自动填充
- `updated_at`更新时间TIMESTAMP自动填充
- `deleted_at`删除时间TIMESTAMP可空软删除
**唯一约束**`(resource_type, resource_id, tag_id)``deleted_at IS NULL` 条件下唯一
#### Scenario: 为设备添加标签
- **WHEN** 用户为设备ID 为 1001添加标签"生产设备"ID 为 101
- **THEN** 系统创建关联记录,`resource_type` 为 "device"`resource_id` 为 1001`tag_id` 为 101标签的 `usage_count` 增加 1
#### Scenario: 为 IoT 卡添加标签
- **WHEN** 用户为 IoT 卡ID 为 2001添加标签"GPS"ID 为 102
- **THEN** 系统创建关联记录,`resource_type` 为 "iot_card"`resource_id` 为 2001`tag_id` 为 102标签的 `usage_count` 增加 1
#### Scenario: 重复添加标签
- **WHEN** 用户为设备ID 为 1001添加已存在的标签"生产设备"ID 为 101
- **THEN** 系统拒绝操作,返回错误信息"该资源已添加此标签"
#### Scenario: 移除资源标签
- **WHEN** 用户移除设备ID 为 1001的标签"生产设备"ID 为 101
- **THEN** 系统删除关联记录(软删除),标签的 `usage_count` 减少 1
---
### Requirement: 按标签查询资源
系统 SHALL 支持按标签查询资源,用户可以选择一个或多个标签,查询包含这些标签的资源。
**查询模式**
- **AND 模式**:查询同时包含所有指定标签的资源(交集)
- **OR 模式**:查询包含任一指定标签的资源(并集)
**查询条件**
- 资源类型(必选,单选)
- 标签 ID 列表(必选,可多选)
- 查询模式(可选,默认 OR
**分页**
- 默认每页 20 条,最大每页 100 条
- 返回总记录数和总页数
#### Scenario: OR 模式查询设备
- **WHEN** 用户查询包含标签"生产设备"ID 为 101或"测试设备"ID 为 102的设备
- **THEN** 系统返回所有包含标签 101 或标签 102 的设备列表
#### Scenario: AND 模式查询设备
- **WHEN** 用户查询同时包含标签"生产设备"ID 为 101和"GPS"ID 为 103的设备
- **THEN** 系统返回同时包含标签 101 和标签 103 的设备列表
#### Scenario: 按标签查询 IoT 卡
- **WHEN** 用户查询包含标签"GPS"ID 为 102的 IoT 卡
- **THEN** 系统返回所有包含标签 102 的 IoT 卡列表
---
### Requirement: 获取资源的标签列表
系统 SHALL 支持查询指定资源的所有标签。
**查询条件**
- 资源类型(必选)
- 资源 ID必选
**返回内容**
- 标签列表ID、名称、颜色
- 按创建时间倒序排列
#### Scenario: 查询设备的标签
- **WHEN** 用户查询设备ID 为 1001的所有标签
- **THEN** 系统返回设备 1001 的标签列表,包含标签 ID、名称、颜色
#### Scenario: 查询没有标签的设备
- **WHEN** 用户查询设备ID 为 1002的所有标签但该设备没有任何标签
- **THEN** 系统返回空列表
---
### Requirement: 热门标签查询
系统 SHALL 支持查询热门标签,按使用次数倒序排列。
**查询条件**
- 限制数量(可选,默认 20
**返回内容**
- 标签列表ID、名称、颜色、使用次数
- 按使用次数倒序排列
#### Scenario: 查询热门标签
- **WHEN** 用户查询热门标签,限制 10 条
- **THEN** 系统返回使用次数最多的 10 个标签,按使用次数倒序排列
---
### Requirement: 标签批量操作
系统 SHALL 支持为资源批量添加或移除标签。
**批量添加**
- 为一个资源添加多个标签
- 为多个资源添加同一个标签
**批量移除**
- 为一个资源移除多个标签
- 为多个资源移除同一个标签
#### Scenario: 为设备批量添加标签
- **WHEN** 用户为设备ID 为 1001批量添加标签["生产设备", "GPS", "4G"]
- **THEN** 系统为设备 1001 创建 3 条关联记录,所有标签的 `usage_count` 各增加 1
#### Scenario: 批量为设备添加标签
- **WHEN** 用户为设备列表 [1001, 1002, 1003] 批量添加标签"生产设备"ID 为 101
- **THEN** 系统为 3 个设备各创建一条关联记录,标签"生产设备"的 `usage_count` 增加 3
#### Scenario: 为设备批量移除标签
- **WHEN** 用户为设备ID 为 1001批量移除标签["生产设备", "GPS"]
- **THEN** 系统删除设备 1001 的 2 条关联记录(软删除),所有标签的 `usage_count` 各减少 1
---
### Requirement: 标签数据校验
系统 SHALL 对标签数据进行校验,确保数据完整性和一致性。
**标签校验规则**
- `name`:必填,长度 1-100 字符,唯一
- `color`:可选,长度 1-20 字符,建议使用十六进制颜色值(如 "#FF5733"
- `usage_count`:必填,≥ 0
**资源-标签关联校验规则**
- `resource_type`:必填,枚举值 "device" | "iot_card" | "number_card"
- `resource_id`:必填,≥ 1
- `tag_id`:必填,≥ 1必须是有效的标签 ID
#### Scenario: 创建标签时名称为空
- **WHEN** 用户创建标签,名称为空
- **THEN** 系统拒绝创建,返回错误信息"标签名称不能为空"
#### Scenario: 创建标签时名称过长
- **WHEN** 用户创建标签,名称长度为 101 字符
- **THEN** 系统拒绝创建,返回错误信息"标签名称长度不能超过 100 字符"
#### Scenario: 添加标签时资源类型无效
- **WHEN** 用户为资源添加标签,`resource_type` 为 "invalid"
- **THEN** 系统拒绝操作,返回错误信息"资源类型无效"
#### Scenario: 添加标签时标签不存在
- **WHEN** 用户为设备添加标签,`tag_id` 为 99999不存在的标签
- **THEN** 系统拒绝操作,返回错误信息"标签不存在"

View File

@@ -0,0 +1,203 @@
# wallet Specification
## Purpose
TBD - created by archiving change add-wallet-transfer-tag-models. Update Purpose after archive.
## Requirements
### Requirement: 钱包实体定义
系统 SHALL 定义钱包(Wallet)实体,统一管理用户钱包和代理钱包,支持余额管理、充值、扣款等操作。
**核心概念**
- **用户钱包**:普通用户和企业用户的钱包,用于购买套餐
- **代理钱包**:代理商的钱包,支持预充值,可用成本价购买套餐
**实体字段**
- `id`:钱包 ID主键BIGINT
- `user_id`:用户 IDBIGINT关联 tb_account.id
- `wallet_type`钱包类型VARCHAR(20),枚举值:"user"-用户钱包 | "agent"-代理钱包)
- `balance`余额BIGINT单位默认 0
- `frozen_balance`冻结余额BIGINT单位默认 0用于订单待支付、提现申请中等场景
- `currency`币种VARCHAR(10),默认 "CNY"
- `status`钱包状态INT1-正常 2-冻结 3-关闭)
- `version`版本号INT默认 0乐观锁字段用于防止并发扣款
- `creator`:创建人 IDBIGINT
- `updater`:更新人 IDBIGINT
- `created_at`创建时间TIMESTAMP自动填充
- `updated_at`更新时间TIMESTAMP自动填充
- `deleted_at`删除时间TIMESTAMP可空软删除
**唯一约束**`(user_id, wallet_type, currency)``deleted_at IS NULL` 条件下唯一
**可用余额计算**:可用余额 = balance - frozen_balance
#### Scenario: 创建用户钱包
- **WHEN** 用户ID 为 2001首次充值
- **THEN** 系统创建钱包记录,`user_id` 为 2001`wallet_type` 为 "user"`balance` 为 0`status` 为 1正常
#### Scenario: 创建代理钱包
- **WHEN** 代理商ID 为 123首次充值
- **THEN** 系统创建钱包记录,`user_id` 为 123`wallet_type` 为 "agent"`balance` 为 0`status` 为 1正常
#### Scenario: 计算可用余额
- **WHEN** 用户钱包余额为 10000 分100 元),冻结余额为 3000 分30 元)
- **THEN** 系统计算可用余额为 7000 分70 元)
---
### Requirement: 钱包明细记录
系统 SHALL 记录所有钱包余额变动,包括充值、扣款、退款、分佣、提现等操作,确保完整的审计追踪。
**实体字段**
- `id`:明细 ID主键BIGINT
- `wallet_id`:钱包 IDBIGINT关联 tb_wallet.id
- `user_id`:用户 IDBIGINT关联 tb_account.id
- `transaction_type`交易类型VARCHAR(20),枚举值:"recharge"-充值 | "deduct"-扣款 | "refund"-退款 | "commission"-分佣 | "withdrawal"-提现)
- `amount`变动金额BIGINT单位正数为增加负数为减少
- `balance_before`变动前余额BIGINT单位
- `balance_after`变动后余额BIGINT单位
- `status`交易状态INT1-成功 2-失败 3-处理中)
- `reference_type`关联业务类型VARCHAR(50),如 "order" | "commission" | "withdrawal" | "topup"
- `reference_id`:关联业务 IDBIGINT
- `remark`备注TEXT
- `metadata`扩展信息JSONB如手续费、支付方式等
- `creator`:创建人 IDBIGINT
- `created_at`创建时间TIMESTAMP自动填充
- `updated_at`更新时间TIMESTAMP自动填充
- `deleted_at`删除时间TIMESTAMP可空软删除
#### Scenario: 充值创建明细记录
- **WHEN** 用户ID 为 2001充值 10000 分100 元)
- **THEN** 系统创建钱包明细记录,`transaction_type` 为 "recharge"`amount` 为 10000`balance_before` 为 0`balance_after` 为 10000`status` 为 1成功
#### Scenario: 购买套餐扣款创建明细记录
- **WHEN** 用户ID 为 2001使用钱包支付购买套餐金额 3000 分30 元)
- **THEN** 系统创建钱包明细记录,`transaction_type` 为 "deduct"`amount` 为 -3000`balance_before` 为 10000`balance_after` 为 7000`reference_type` 为 "order"`reference_id` 为订单 ID
#### Scenario: 分佣发放创建明细记录
- **WHEN** 代理ID 为 123的分佣 5000 分50 元)审批通过并发放
- **THEN** 系统创建钱包明细记录,`transaction_type` 为 "commission"`amount` 为 5000`balance_before` 为 20000`balance_after` 为 25000`reference_type` 为 "commission"`reference_id` 为分佣记录 ID
---
### Requirement: 充值记录管理
系统 SHALL 记录所有充值操作,包括充值订单号、金额、支付方式、支付状态等信息。
**实体字段**
- `id`:充值记录 ID主键BIGINT
- `user_id`:用户 IDBIGINT关联 tb_account.id
- `wallet_id`:钱包 IDBIGINT关联 tb_wallet.id
- `recharge_no`充值订单号VARCHAR(50),唯一)
- `amount`充值金额BIGINT单位
- `payment_method`支付方式VARCHAR(20),枚举值:"alipay"-支付宝 | "wechat"-微信 | "bank"-银行转账 | "offline"-线下)
- `payment_channel`支付渠道VARCHAR(50)
- `payment_transaction_id`第三方支付交易号VARCHAR(100)
- `status`充值状态INT1-待支付 2-已支付 3-已完成 4-已关闭 5-已退款)
- `paid_at`支付时间TIMESTAMP可空
- `completed_at`完成时间TIMESTAMP可空
- `creator`:创建人 IDBIGINT
- `updater`:更新人 IDBIGINT
- `created_at`创建时间TIMESTAMP自动填充
- `updated_at`更新时间TIMESTAMP自动填充
- `deleted_at`删除时间TIMESTAMP可空软删除
#### Scenario: 创建充值订单
- **WHEN** 用户ID 为 2001发起充值 10000 分100 元),选择支付宝支付
- **THEN** 系统创建充值记录,生成唯一的 `recharge_no``amount` 为 10000`payment_method` 为 "alipay"`status` 为 1待支付
#### Scenario: 充值支付完成
- **WHEN** 用户完成支付宝支付
- **THEN** 系统将充值记录状态从 1待支付变更为 2已支付记录 `paid_at` 时间和 `payment_transaction_id`
#### Scenario: 充值到账
- **WHEN** 充值记录状态为 2已支付系统处理充值到账
- **THEN** 系统将钱包余额增加 10000 分,创建钱包明细记录,将充值记录状态变更为 3已完成记录 `completed_at` 时间
---
### Requirement: 钱包余额操作
系统 SHALL 支持钱包余额的充值、扣款、退款、冻结、解冻等操作,使用乐观锁防止并发问题。
**操作类型**
- **充值**:增加钱包余额
- **扣款**:减少钱包余额(如购买套餐)
- **退款**:增加钱包余额(如订单退款)
- **冻结**:将部分余额转为冻结状态(如订单待支付)
- **解冻**:将冻结余额转回可用余额(如订单取消)
**并发控制**
- 使用 `version` 字段实现乐观锁
- 每次更新余额时,检查 `version` 是否匹配
- 如果 `version` 不匹配,说明有并发更新,操作失败并重试
#### Scenario: 钱包充值
- **WHEN** 用户钱包当前余额为 10000 分,充值 5000 分
- **THEN** 系统将钱包余额更新为 15000 分,`version` 从 1 变更为 2创建钱包明细记录
#### Scenario: 钱包扣款
- **WHEN** 用户钱包当前余额为 15000 分,购买套餐扣款 3000 分
- **THEN** 系统检查可用余额15000 - 0 = 15000≥ 3000将钱包余额更新为 12000 分,`version` 从 2 变更为 3创建钱包明细记录
#### Scenario: 余额不足扣款失败
- **WHEN** 用户钱包当前余额为 2000 分,购买套餐需要扣款 3000 分
- **THEN** 系统检查可用余额2000 - 0 = 2000< 3000拒绝扣款返回错误信息"余额不足"
#### Scenario: 并发扣款乐观锁生效
- **WHEN** 用户钱包当前余额为 10000 分version 为 1两个并发请求同时扣款 3000 分和 5000 分
- **THEN** 第一个请求成功,余额变为 7000 分version 变为 2第二个请求因 version 不匹配失败需重新读取最新余额7000 分)后重试
#### Scenario: 冻结余额
- **WHEN** 用户创建订单 10001订单金额 3000 分,选择钱包支付
- **THEN** 系统将钱包的 `frozen_balance` 增加 3000 分,可用余额减少 3000 分
#### Scenario: 解冻余额
- **WHEN** 用户取消订单 10001订单金额 3000 分
- **THEN** 系统将钱包的 `frozen_balance` 减少 3000 分,可用余额增加 3000 分
---
### Requirement: 钱包数据校验
系统 SHALL 对钱包数据进行校验,确保数据完整性和一致性。
**校验规则**
- `user_id`:必填,≥ 1
- `wallet_type`:必填,枚举值 "user" | "agent"
- `balance`:必填,≥ 0
- `frozen_balance`:必填,≥ 0≤ balance
- `currency`:必填,长度 1-10 字符
- `status`:必填,枚举值 1-3
- `version`:必填,≥ 0
#### Scenario: 创建钱包时 user_id 无效
- **WHEN** 创建钱包,`user_id` 为 0
- **THEN** 系统拒绝创建,返回错误信息"用户 ID 无效"
#### Scenario: 创建钱包时 wallet_type 无效
- **WHEN** 创建钱包,`wallet_type` 为 "invalid"
- **THEN** 系统拒绝创建,返回错误信息"钱包类型无效"
#### Scenario: 冻结余额超过总余额
- **WHEN** 钱包余额为 10000 分,尝试冻结 15000 分
- **THEN** 系统拒绝操作,返回错误信息"冻结余额不能超过总余额"