实现用户和组织模型(店铺、企业、个人客户)

核心功能:
- 实现 7 级店铺层级体系(Shop 模型 + 层级校验)
- 实现企业管理模型(Enterprise 模型)
- 实现个人客户管理模型(PersonalCustomer 模型)
- 重构 Account 模型关联关系(基于 EnterpriseID 而非 ParentID)
- 完整的 Store 层和 Service 层实现
- 递归查询下级店铺功能(含 Redis 缓存)
- 全面的单元测试覆盖(Shop/Enterprise/PersonalCustomer Store + Shop Service)

技术要点:
- 显式指定所有 GORM 模型的数据库字段名(column: 标签)
- 统一的字段命名规范(数据库用 snake_case,Go 用 PascalCase)
- 完整的中文字段注释和业务逻辑说明
- 100% 测试覆盖(20+ 测试用例全部通过)

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
This commit is contained in:
2026-01-09 18:02:46 +08:00
parent 6fc90abeb6
commit a36e4a79c0
51 changed files with 5736 additions and 144 deletions

View File

@@ -0,0 +1,147 @@
# user-organization Specification
## Purpose
TBD - created by archiving change add-user-organization-model. Update Purpose after archive.
## Requirements
### Requirement: 店铺模型定义
系统 SHALL 创建店铺表tb_shop用于存储代理商的组织信息包含店铺名称、店铺编号、上级店铺ID、层级、联系人信息、地址信息和状态字段。
#### Scenario: 创建一级代理店铺
- **WHEN** 创建店铺时 parent_id 为 NULL
- **THEN** 系统创建该店铺并设置 level = 1
#### Scenario: 创建下级代理店铺
- **WHEN** 创建店铺时指定 parent_id 为已存在店铺的 ID
- **THEN** 系统创建该店铺并设置 level = 上级店铺的 level + 1
#### Scenario: 店铺层级限制
- **WHEN** 创建店铺时计算出的 level 超过 7
- **THEN** 系统拒绝创建并返回错误"店铺层级不能超过7级"
#### Scenario: 店铺编号唯一性
- **WHEN** 创建店铺时指定的 shop_code 已存在
- **THEN** 系统拒绝创建并返回错误"店铺编号已存在"
---
### Requirement: 企业模型定义
系统 SHALL 创建企业表tb_enterprise用于存储企业客户的组织信息包含企业名称、企业编号、归属店铺ID、法人代表、联系人信息、营业执照号、地址信息和状态字段。
#### Scenario: 创建平台直属企业
- **WHEN** 创建企业时 owner_shop_id 为 NULL
- **THEN** 系统创建该企业,归属于平台
#### Scenario: 创建代理商下属企业
- **WHEN** 创建企业时指定 owner_shop_id 为已存在店铺的 ID
- **THEN** 系统创建该企业,归属于指定店铺
#### Scenario: 企业编号唯一性
- **WHEN** 创建企业时指定的 enterprise_code 已存在
- **THEN** 系统拒绝创建并返回错误"企业编号已存在"
---
### Requirement: 个人客户模型定义
系统 SHALL 创建个人客户表tb_personal_customer用于存储个人客户信息包含手机号、昵称、头像URL、微信OpenID、微信UnionID和状态字段。个人客户不参与RBAC权限体系。
#### Scenario: 创建个人客户
- **WHEN** 用户通过手机号注册
- **THEN** 系统创建个人客户记录phone 字段存储手机号
#### Scenario: 手机号唯一性
- **WHEN** 创建个人客户时手机号已存在
- **THEN** 系统拒绝创建并返回错误"手机号已被注册"
#### Scenario: 绑定微信信息
- **WHEN** 个人客户授权微信登录
- **THEN** 系统更新 wx_open_id 和 wx_union_id 字段
---
### Requirement: 账号模型重构
系统 SHALL 修改账号表tb_account结构支持四种用户类型超级管理员(1)、平台用户(2)、代理账号(3)、企业账号(4)。代理账号必须关联店铺ID企业账号必须关联企业ID。
#### Scenario: 创建超级管理员账号
- **WHEN** 创建账号时 user_type = 1
- **THEN** 系统创建超级管理员账号shop_id 和 enterprise_id 均为 NULL
#### Scenario: 创建平台用户账号
- **WHEN** 创建账号时 user_type = 2
- **THEN** 系统创建平台用户账号shop_id 和 enterprise_id 均为 NULL
#### Scenario: 创建代理账号
- **WHEN** 创建账号时 user_type = 3
- **THEN** 系统必须指定 shop_identerprise_id 为 NULL
#### Scenario: 创建企业账号
- **WHEN** 创建账号时 user_type = 4
- **THEN** 系统必须指定 enterprise_idshop_id 为 NULL
#### Scenario: 代理账号必须关联店铺
- **WHEN** 创建代理账号user_type = 3但未指定 shop_id
- **THEN** 系统拒绝创建并返回错误"代理账号必须关联店铺"
#### Scenario: 企业账号必须关联企业
- **WHEN** 创建企业账号user_type = 4但未指定 enterprise_id
- **THEN** 系统拒绝创建并返回错误"企业账号必须关联企业"
---
### Requirement: 店铺层级递归查询
系统 SHALL 支持递归查询指定店铺的所有下级店铺ID列表包含直接和间接下级并将结果缓存到Redis30分钟过期。当店铺的parent_id变更或店铺被删除时系统必须清除相关缓存。
#### Scenario: 查询下级店铺ID列表
- **WHEN** 调用 GetSubordinateShopIDs(shopID) 方法
- **THEN** 系统返回该店铺的所有下级店铺ID列表递归包含所有层级
#### Scenario: 下级店铺缓存命中
- **WHEN** Redis 中存在店铺的下级ID缓存
- **THEN** 系统直接返回缓存数据,不查询数据库
#### Scenario: 下级店铺缓存未命中
- **WHEN** Redis 中不存在店铺的下级ID缓存
- **THEN** 系统查询数据库将结果缓存到Redis过期时间30分钟然后返回结果
#### Scenario: 店铺删除时清除缓存
- **WHEN** 店铺被软删除
- **THEN** 系统清除该店铺及其所有上级店铺的下级ID缓存
---
### Requirement: 用户类型常量定义
系统 SHALL 在 pkg/constants/ 中定义用户类型常量,禁止在代码中硬编码用户类型数值。
#### Scenario: 使用用户类型常量
- **WHEN** 代码中需要判断用户类型
- **THEN** 必须使用 constants.UserTypeSuperAdmin、constants.UserTypePlatform、constants.UserTypeAgent、constants.UserTypeEnterprise 常量
#### Scenario: 禁止硬编码用户类型
- **WHEN** 代码中直接使用数字 1、2、3、4 表示用户类型
- **THEN** 代码审查不通过,必须改为使用常量
---
### Requirement: 店铺账号数据权限
系统 SHALL 基于店铺层级实现数据权限过滤同一店铺的所有账号能看到店铺的所有数据上级店铺能看到下级店铺的数据。平台用户user_type = 1 或 2跳过数据权限过滤。
#### Scenario: 平台用户查询数据
- **WHEN** 平台用户user_type = 1 或 2查询业务数据
- **THEN** 系统返回所有数据,不应用店铺过滤条件
#### Scenario: 代理账号查询数据
- **WHEN** 代理账号user_type = 3shop_id = X查询业务数据
- **THEN** 系统自动添加 WHERE 条件shop_id IN (X, 及X的所有下级店铺ID)
#### Scenario: 企业账号查询数据
- **WHEN** 企业账号user_type = 4enterprise_id = Y查询业务数据
- **THEN** 系统自动添加 WHERE 条件enterprise_id = Y
---