# 数据库设计 ## 1. 变更概览 | 操作 | 表名 | 说明 | |------|------|------| | 修改 | `users` | 新增 phone、wx_openid、user_source、city 等字段 | | 修改 | `tenants` | 新增 tenant_type 字段 | | 修改 | `registrations` | 新增 participant_type、child_id 字段 | | 修改 | `contests` | 新增 visibility 字段 | | 新增 | `children` | 子女信息表 | | 新增(二期) | `user_identities` | 用户身份表(一号多身份) | | 新增(二期) | `contest_co_organizers` | 活动协办机构表 | --- ## 2. 表结构详细设计 ### 2.1 users 表(改造) 新增字段: | 字段名 | 类型 | 必填 | 默认值 | 说明 | |--------|------|------|--------|------| | `phone` | VARCHAR(20) | 否 | NULL | 手机号,全局唯一,用于手机号登录 | | `wx_openid` | VARCHAR(64) | 否 | NULL | 微信 OpenID,全局唯一 | | `wx_unionid` | VARCHAR(64) | 否 | NULL | 微信 UnionID,跨应用统一标识 | | `user_source` | ENUM | 否 | 'admin_created' | 用户来源:admin_created / self_registered | | `city` | VARCHAR(50) | 否 | NULL | 所在城市 | | `birthday` | DATE | 否 | NULL | 出生日期(自注册用户可填写) | ```sql -- 新增字段 ALTER TABLE users ADD COLUMN phone VARCHAR(20) UNIQUE DEFAULT NULL COMMENT '手机号'; ALTER TABLE users ADD COLUMN wx_openid VARCHAR(64) UNIQUE DEFAULT NULL COMMENT '微信OpenID'; ALTER TABLE users ADD COLUMN wx_unionid VARCHAR(64) DEFAULT NULL COMMENT '微信UnionID'; ALTER TABLE users ADD COLUMN user_source VARCHAR(20) NOT NULL DEFAULT 'admin_created' COMMENT '用户来源:admin_created/self_registered'; ALTER TABLE users ADD COLUMN city VARCHAR(50) DEFAULT NULL COMMENT '所在城市'; ALTER TABLE users ADD COLUMN birthday DATE DEFAULT NULL COMMENT '出生日期'; -- 索引 CREATE INDEX idx_users_phone ON users(phone); CREATE INDEX idx_users_wx_openid ON users(wx_openid); CREATE INDEX idx_users_user_source ON users(user_source); ``` **设计说明**: - `phone` 设为 UNIQUE 但允许 NULL,因为管理员创建的机构用户可能没有手机号 - `wx_openid` 和 `wx_unionid` 为二期微信登录预留 - `user_source` 区分用户是管理员创建的还是自主注册的,方便后台统计和管理 - 自注册用户归属公众租户(code='public', tenantId=8),而非 tenant_id 为 NULL,以兼容现有 JWT 认证链路 --- ### 2.2 children 表(新增) 子女信息表,关联家长账号。 | 字段名 | 类型 | 必填 | 默认值 | 说明 | |--------|------|------|--------|------| | `id` | INT | 是 | AUTO_INCREMENT | 主键 | | `parent_id` | INT | 是 | - | 家长用户 ID,关联 users.id | | `name` | VARCHAR(50) | 是 | - | 子女姓名 | | `gender` | VARCHAR(10) | 否 | NULL | 性别:male / female | | `birthday` | DATE | 否 | NULL | 出生日期 | | `grade` | VARCHAR(20) | 否 | NULL | 年级(如:大班、一年级) | | `city` | VARCHAR(50) | 否 | NULL | 所在城市 | | `school_name` | VARCHAR(100) | 否 | NULL | 学校/幼儿园名称(手动填写) | | `avatar` | VARCHAR(500) | 否 | NULL | 头像 URL | | `is_deleted` | TINYINT(1) | 否 | 0 | 软删除标记 | | `create_time` | DATETIME | 否 | CURRENT_TIMESTAMP | 创建时间 | | `modify_time` | DATETIME | 否 | CURRENT_TIMESTAMP | 修改时间 | ```sql CREATE TABLE children ( id INT AUTO_INCREMENT PRIMARY KEY, parent_id INT NOT NULL COMMENT '家长用户ID', name VARCHAR(50) NOT NULL COMMENT '子女姓名', gender VARCHAR(10) DEFAULT NULL COMMENT '性别:male/female', birthday DATE DEFAULT NULL COMMENT '出生日期', grade VARCHAR(20) DEFAULT NULL COMMENT '年级', city VARCHAR(50) DEFAULT NULL COMMENT '所在城市', school_name VARCHAR(100) DEFAULT NULL COMMENT '学校/幼儿园名称', avatar VARCHAR(500) DEFAULT NULL COMMENT '头像URL', is_deleted TINYINT(1) NOT NULL DEFAULT 0 COMMENT '软删除标记', create_time DATETIME(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3), modify_time DATETIME(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3) ON UPDATE CURRENT_TIMESTAMP(3), INDEX idx_children_parent_id (parent_id), CONSTRAINT fk_children_parent FOREIGN KEY (parent_id) REFERENCES users(id) ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='子女信息表'; ``` **设计说明**: - `parent_id` 关联 users 表,一个家长可以有多个子女 - `school_name` 是手动填写的文本字段,不关联租户表,因为公众用户的学校不一定是平台上的租户 - `grade` 存储文本如"大班""一年级",不做枚举约束,因为不同教育体系的年级名称不同 - 使用软删除(`is_deleted`),避免误删导致历史报名数据关联丢失 --- ### 2.3 tenants 表(改造) 新增字段: | 字段名 | 类型 | 必填 | 默认值 | 说明 | |--------|------|------|--------|------| | `tenant_type` | VARCHAR(20) | 否 | 'other' | 租户类型 | ```sql ALTER TABLE tenants ADD COLUMN tenant_type VARCHAR(20) NOT NULL DEFAULT 'other' COMMENT '租户类型:platform/library/kindergarten/school/institution/other'; CREATE INDEX idx_tenants_type ON tenants(tenant_type); ``` **租户类型枚举值**: | 值 | 说明 | |----|------| | `platform` | 平台租户(智创未来,全局唯一) | | `library` | 图书馆 | | `kindergarten` | 幼儿园 | | `school` | 学校 | | `institution` | 社会机构 | | `other` | 其他 | --- ### 2.4 registrations 表(改造) 新增字段: | 字段名 | 类型 | 必填 | 默认值 | 说明 | |--------|------|------|--------|------| | `participant_type` | VARCHAR(10) | 否 | 'self' | 参与者类型:self(自己)/ child(子女) | | `child_id` | INT | 否 | NULL | 子女 ID,participant_type 为 child 时必填 | ```sql ALTER TABLE registrations ADD COLUMN participant_type VARCHAR(10) NOT NULL DEFAULT 'self' COMMENT '参与者类型:self-自己/child-代子女报名'; ALTER TABLE registrations ADD COLUMN child_id INT DEFAULT NULL COMMENT '子女ID,代子女报名时填写'; CREATE INDEX idx_registrations_participant ON registrations(participant_type); CREATE INDEX idx_registrations_child ON registrations(child_id); ``` **设计说明**: - `user_id`(现有字段)始终记录操作人(谁提交的报名) - `participant_type = 'self'`:用户以自己身份报名,`child_id` 为 NULL - `participant_type = 'child'`:家长代子女报名,`child_id` 指向 children 表 - 管理端查看报名记录时,能清晰区分"张妈妈帮小明报名"和"李同学自己报名" --- ### 2.5 contests 表(改造) 新增字段: | 字段名 | 类型 | 必填 | 默认值 | 说明 | |--------|------|------|--------|------| | `visibility` | VARCHAR(20) | 否 | 'designated' | 活动可见范围 | ```sql ALTER TABLE contests ADD COLUMN visibility VARCHAR(20) NOT NULL DEFAULT 'designated' COMMENT '可见范围:public-公开/designated-指定机构/internal-仅内部'; ``` **可见范围说明**: | 值 | 说明 | 数据过滤逻辑 | |----|------|-------------| | `public` | 全体公开 | 所有注册用户可见,公众端活动大厅展示 | | `designated` | 指定机构 | 仅 contest_tenants 中列出的租户可见(现有逻辑) | | `internal` | 仅内部 | 仅创建该活动的机构内部可见 | --- ### 2.6 user_identities 表(二期新增) 一号多身份的身份关联表。 | 字段名 | 类型 | 必填 | 默认值 | 说明 | |--------|------|------|--------|------| | `id` | INT | 是 | AUTO_INCREMENT | 主键 | | `user_id` | INT | 是 | - | 用户 ID | | `identity_type` | VARCHAR(20) | 是 | - | 身份类型 | | `tenant_id` | INT | 否 | NULL | 关联租户(公众身份为 NULL) | | `role_id` | INT | 否 | NULL | 关联角色 | | `status` | VARCHAR(10) | 否 | 'active' | 状态:active / inactive | | `create_time` | DATETIME | 否 | CURRENT_TIMESTAMP | 创建时间 | ```sql -- 二期实现 CREATE TABLE user_identities ( id INT AUTO_INCREMENT PRIMARY KEY, user_id INT NOT NULL COMMENT '用户ID', identity_type VARCHAR(20) NOT NULL COMMENT '身份类型:public/tenant_admin/judge/staff', tenant_id INT DEFAULT NULL COMMENT '关联租户ID', role_id INT DEFAULT NULL COMMENT '关联角色ID', status VARCHAR(10) NOT NULL DEFAULT 'active' COMMENT '状态:active/inactive', create_time DATETIME(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3), INDEX idx_identity_user (user_id), INDEX idx_identity_tenant (tenant_id), UNIQUE KEY uk_user_tenant_type (user_id, tenant_id, identity_type), CONSTRAINT fk_identity_user FOREIGN KEY (user_id) REFERENCES users(id), CONSTRAINT fk_identity_tenant FOREIGN KEY (tenant_id) REFERENCES tenants(id) ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='用户身份表'; ``` --- ### 2.7 contest_co_organizers 表(二期新增) 活动协办机构关联表。 ```sql -- 二期实现 CREATE TABLE contest_co_organizers ( id INT AUTO_INCREMENT PRIMARY KEY, contest_id INT NOT NULL COMMENT '活动ID', tenant_id INT NOT NULL COMMENT '协办机构租户ID', permission_level VARCHAR(20) NOT NULL DEFAULT 'readonly' COMMENT '权限级别:readonly/limited/full', create_time DATETIME(3) NOT NULL DEFAULT CURRENT_TIMESTAMP(3), UNIQUE KEY uk_contest_tenant (contest_id, tenant_id), CONSTRAINT fk_co_org_contest FOREIGN KEY (contest_id) REFERENCES contests(id), CONSTRAINT fk_co_org_tenant FOREIGN KEY (tenant_id) REFERENCES tenants(id) ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='活动协办机构表'; ``` --- ## 3. ER 关系图 ``` ┌──────────┐ │ tenants │ │──────────│ │ id │ │ name │ │ tenant_ │ ┌──────────│ type │──────────┐ │ └──────────┘ │ │ tenant_id │ organizer_ │ │ tenant_id ┌────▼─────┐ ┌──────▼──────┐ │ users │ │ contests │ │──────────│ │─────────────│ │ id │ │ id │ │ username │ │ contestName │ │ phone │ │ visibility │ │ user_ │ │ contestType │ │ source │ └──────┬──────┘ └──┬───┬───┘ │ │ │ │ contest_id │ │ parent_id ┌─────▼────────┐ │ │ │registrations │ │ ├──────────┐ │──────────────│ │ │ │ │ id │ │ │ ┌──────▼──────┐ │ user_id ◄────┼──── users.id │ │ │ children │ │ participant_ │ │ │ │─────────────│ │ type │ │ │ │ id │ │ child_id ◄───┼──── children.id │ │ │ parent_id │ │ contest_id │ │ │ │ name │ │ status │ │ │ │ birthday │ └──────────────┘ │ │ └─────────────┘ │ │ │ └─── user_id ─────────── registrations.user_id │ └─── tenant_id ──────────── tenants.id (公众用户归属 public 租户) ``` --- ## 4. 数据迁移注意事项 ### 4.1 现有数据兼容 - `users.user_source`:现有用户默认值为 `admin_created`,无需迁移 - `users.phone`:允许 NULL,现有用户无需填写 - `registrations.participant_type`:现有报名记录默认值为 `self`,无需迁移 - `tenants.tenant_type`:现有租户默认值为 `other`,需要手动为已有租户设置正确类型 - `contests.visibility`:现有活动默认值为 `designated`,保持现有行为不变 ### 4.2 Prisma Schema 同步 所有数据库变更需要同步到 Prisma Schema 文件,通过 `prisma migrate` 管理迁移。 ### 4.3 广东省图初始化数据 ```sql -- 创建广东省图租户 INSERT INTO tenants (name, code, tenant_type, description, valid_state) VALUES ('广东省立中山图书馆', 'gdlib', 'library', '广东省图少儿绘本创作活动主办方', 1); -- 创建广东省图管理员账号 -- (通过初始化脚本创建,关联对应的角色和权限) ```