diff --git a/backend/docs/活动留言板接口文档.md b/backend/docs/活动留言板接口文档.md new file mode 100644 index 0000000..990d74f --- /dev/null +++ b/backend/docs/活动留言板接口文档.md @@ -0,0 +1,824 @@ +# 活动留言板(Activity MessageBoard)后端接口文档 + +> 本文档基于前端组件 `frontend/pages/support-activity/components/MessageBoard.vue` 反推后端接口契约,作为留言板模块的接口设计依据与开发指南。 + +--- + +## 1. 背景与现状 + +### 1.1 前端组件 + +`MessageBoard.vue` 是一个**纯展示组件**(presentational component),通过 `props.messages` 接收留言列表,每条留言渲染为一条气泡,组件本身不直接发请求。 + +调用关系: + +``` +support-activity/index.vue + ├── // 仅展示 + └── handleSendMessage(text) // 仅本地 messageList.push(...) + toast +``` + +当前 `messageList` 在 `index.vue` 中是**写死的 mock 数据**,并未对接任何后端接口: + +```js +// 留言板 mock 数据(index.vue 中) +const messageList = ref([ + { id: 1, user: "小星星", avatar: "", content: "生日快乐!永远支持你~", time: "刚刚", isSelf: false }, + // ... +]); +``` + +### 1.2 前端期望的字段(来自组件 props 形态) + +| 字段 | 类型 | 说明 | 备注 | +| --------- | -------- | ----------------------------- | --------------------------------------------- | +| `id` | number | 留言唯一 ID | 当前 mock 用 `Date.now()` | +| `user` | string | 留言人昵称 | 当前用户昵称(来自 `fan_profiles.nickname`) | +| `avatar` | string | 留言人头像 URL | 当前 mock 为空字符串 | +| `content` | string | 留言正文 | 必填 | +| `time` | string | 留言时间 | 当前 mock 是"刚刚"/"5分钟前"等相对时间字符串 | +| `isSelf` | boolean | 是否当前登录用户所发 | 当前 mock 由前端标记 | + +> ⚠️ 后端**不应直接复用 mock 字段语义**:`time` 应返回毫秒时间戳或 ISO 字符串,由前端格式化;`isSelf` 应由前端根据当前用户 ID 自行判断(避免后端泄露会话状态)。 + +### 1.3 后端现状 + +经排查(grep 全量 `message|留言`),当前 **backend 中没有留言相关接口**: + +- `services/activityService/` 没有 `Message` 类方法 +- 数据库迁移脚本中没有 `activity_messages` 或 `messages` 表(`migrate_add_comments_v1.sql` 是给 laser_card 加的 `comments` 列,不是新表) +- `pkg/errors` 中也没有 `ErrMessageNotFound` 等业务错误 + +本次为**全新模块**设计。 + +--- + +## 2. 设计目标 + +1. 支持粉丝在应援活动页面对活动发表祝福留言(每用户每活动有频率/上限限制)。 +2. 支持活动详情页气泡墙展示历史留言(最新在上)。 +3. 与现有活动模块同源(路由 `activities/{id}/...`),共享 `auth + activity 校验`。 +4. 兼容现有前端组件的 props 契约,前端只需把 mock 替换为接口数据源。 +5. 预留轮询扩展(`since_timestamp` / `since_id`)以便后续做实时弹幕。 + +--- + +## 3. 数据库设计 + +新增表 `activity_messages`,**主键 bigserial**,自增序列起始 10000(按 `CLAUDE.md` 序列规范)。 + +```sql +-- ============================================================ +-- 活动留言表 +-- ============================================================ +CREATE TABLE IF NOT EXISTS public.activity_messages ( + id BIGSERIAL PRIMARY KEY, + activity_id BIGINT NOT NULL, + user_id BIGINT NOT NULL, + star_id BIGINT NOT NULL, + content VARCHAR(500) NOT NULL, + status SMALLINT NOT NULL DEFAULT 0, -- 0=正常 1=隐藏 2=已删除 + created_at BIGINT NOT NULL, + updated_at BIGINT NOT NULL, + deleted_at BIGINT, + + CONSTRAINT fk_activity_messages_activity + FOREIGN KEY (activity_id) REFERENCES public.activities(id) ON DELETE CASCADE, + CONSTRAINT fk_activity_messages_user + FOREIGN KEY (user_id) REFERENCES public.users(id), + CONSTRAINT fk_activity_messages_star + FOREIGN KEY (star_id) REFERENCES public.stars(star_id) +); + +COMMENT ON TABLE public.activity_messages IS '活动留言表。粉丝在应援活动页面的祝福留言,用于气泡墙展示'; +COMMENT ON COLUMN public.activity_messages.id IS '主键,自增'; +COMMENT ON COLUMN public.activity_messages.activity_id IS 'FK -> activities.id,所属活动'; +COMMENT ON COLUMN public.activity_messages.user_id IS '留言用户 ID'; +COMMENT ON COLUMN public.activity_messages.star_id IS '所属明星/星球 ID,活动冗余字段便于按星球维度查询'; +COMMENT ON COLUMN public.activity_messages.content IS '留言正文,长度 1-500'; +COMMENT ON COLUMN public.activity_messages.status IS '状态:0=正常 | 1=隐藏(违规被审核屏蔽)| 2=已删除(用户/管理删除)'; +COMMENT ON COLUMN public.activity_messages.created_at IS '留言时间,毫秒时间戳'; +COMMENT ON COLUMN public.activity_messages.updated_at IS '更新时间,毫秒时间戳'; +COMMENT ON COLUMN public.activity_messages.deleted_at IS '软删除时间,NULL 表示未删除'; + +-- 序列起始值留给测试数据硬编码 ID 用 +CREATE SEQUENCE IF NOT EXISTS activity_messages_id_seq START WITH 10000; + +-- 索引:活动维度按时间倒序拉取是热路径 +CREATE INDEX IF NOT EXISTS idx_activity_messages_activity_created + ON public.activity_messages (activity_id, created_at DESC, id DESC) + WHERE deleted_at IS NULL; + +-- 索引:用户维度查询某用户的所有留言 +CREATE INDEX IF NOT EXISTS idx_activity_messages_user_created + ON public.activity_messages (user_id, created_at DESC) + WHERE deleted_at IS NULL; + +-- 索引:用于实时增量拉取(since_timestamp + since_id 翻页) +CREATE INDEX IF NOT EXISTS idx_activity_messages_activity_incr + ON public.activity_messages (activity_id, id DESC) + WHERE deleted_at IS NULL; +``` + +> **为什么 `WHERE deleted_at IS NULL`?** 留言是只追加、软删除的写入场景,使用部分索引(partial index)能让正常状态的数据索引更紧凑,热路径扫描更快。 + +### 3.1 与已有表的关系 + +- `activity_id` → `activities.id`:活动被删除时级联清理留言(`ON DELETE CASCADE`)。 +- `user_id` → `users.id`:硬依赖,用户被硬删时由应用层兜底(软删用户场景下留言仍可保留,展示时昵称取自 join 后的 fan_profiles/users)。 +- `star_id` → `stars.star_id`:冗余字段,便于按星球做运营统计(如"某星球 24h 留言量")。 + +--- + +## 4. Proto 定义 + +新增文件 `backend/proto/activity/activity_messages.proto`(追加到现有 `backend/proto/activity/` 下,复用 `ActivityService` 服务): + +```protobuf +syntax = "proto3"; + +package com.topfans.activity; + +option go_package = "github.com/topfans/backend/pkg/proto/activity"; +option java_package = "com.topfans.activity.proto"; + +// ============== 留言相关 RPC ============== + +service ActivityService { + // ... 现有 RPC ... + + // 列出活动留言(历史/首次加载) + rpc ListActivityMessages(ListActivityMessagesRequest) returns (ListActivityMessagesResponse); + + // 发送一条留言 + rpc CreateActivityMessage(CreateActivityMessageRequest) returns (CreateActivityMessageResponse); + + // 增量获取最新留言(轮询用,与 GetLatestContributions 同样模式) + rpc GetLatestActivityMessages(GetLatestActivityMessagesRequest) returns (GetLatestActivityMessagesResponse); +} + +// 单条留言 +message ActivityMessage { + int64 id = 1; + int64 activity_id = 2; + int64 user_id = 3; + int64 star_id = 4; + string nickname = 5; // 冗余带出,减少 join + string avatar_url = 6; // 冗余带出 + string content = 7; + int64 created_at = 8; +} + +// 列出活动留言 +message ListActivityMessagesRequest { + int64 activity_id = 1; + int32 page = 2; // 默认 1 + int32 page_size = 3; // 默认 20,最大 50 +} +message ListActivityMessagesResponse { + BaseResponse base = 1; + repeated ActivityMessage messages = 2; + int32 page = 3; + int32 page_size = 4; + int32 total = 5; +} + +// 发送留言 +message CreateActivityMessageRequest { + int64 activity_id = 1; + int64 user_id = 2; // 从 JWT 解析,gateway 注入 + int64 star_id = 3; + string content = 4; +} +message CreateActivityMessageResponse { + BaseResponse base = 1; + ActivityMessage message = 2; +} + +// 增量获取最新留言 +message GetLatestActivityMessagesRequest { + int64 activity_id = 1; + int64 since_timestamp = 2; // 自该时间戳之后的新记录(毫秒) + int64 since_id = 3; // 配合 since_timestamp 在同毫秒内二次排序 + int32 limit = 4; // 默认 20,最大 50 +} +message GetLatestActivityMessagesResponse { + BaseResponse base = 1; + repeated ActivityMessage messages = 2; +} +``` + +> **生成命令**:项目根目录执行 `make proto` 或参考 `backend/Proto编译完成总结.md`。生成后会产生 `activity_messages.pb.go`,注册到 `ActivityServiceHandler` 接口。 + +--- + +## 5. HTTP 接口设计 + +### 5.1 通用约定 + +- **Base URL**:`https://{host}/api/v1` +- **认证**:所有接口必须带 `Authorization: Bearer `(JWT),由 `middleware.AuthMiddleware()` 注入 `user_id` / `star_id`。 +- **统一响应结构**(沿用 `gateway/pkg/response.Response`): + ```json + { + "code": 0, + "message": "ok", + "data": { ... } + } + ``` +- **错误码**:使用 `pkg/errors` 中的 gRPC code 映射(`codes.OK=0`、`InvalidArgument=3`、`NotFound=5`、`PermissionDenied=7`、`ResourceExhausted=8`、`Internal=13`、`Unauthenticated=16`)。 +- **时间字段**:统一返回**毫秒时间戳**(`created_at`),前端负责格式化。 +- **驼峰/下划线**:HTTP 响应使用 **snake_case**(与现有 `GetLatestContributions` 保持一致),由 `convert*Response` 函数在 controller 层做转换。 + +### 5.2 接口清单 + +| # | 方法 | 路径 | 说明 | 鉴权 | +| --- | ---- | ----------------------------------------------- | ------------ | ---- | +| 1 | GET | `/api/v1/activities/{activity_id}/messages` | 列出活动留言 | 需要 | +| 2 | POST | `/api/v1/activities/{activity_id}/messages` | 发送一条留言 | 需要 | +| 3 | GET | `/api/v1/activities/{activity_id}/messages/latest` | 增量拉取最新留言 | 需要 | + +> 三个接口都登记到 `gateway/controller/activity_controller.go` 的 Swagger 注解上,`router.go` 的 `activities` 组内注册。 + +--- + +### 5.3 接口 1:列出活动留言 + +**适用场景**:用户进入活动详情页首次加载 / 下拉刷新(替代当前 `messageList` mock)。 + +``` +GET /api/v1/activities/{activity_id}/messages?page=1&page_size=20 +``` + +**Path 参数** + +| 名称 | 类型 | 必填 | 说明 | +| ------------ | ------ | ---- | -------- | +| `activity_id` | int64 | 是 | 活动 ID | + +**Query 参数** + +| 名称 | 类型 | 必填 | 默认 | 说明 | +| ----------- | ---- | ---- | ---- | -------------------------- | +| `page` | int | 否 | 1 | 页码,从 1 开始 | +| `page_size` | int | 否 | 20 | 每页条数,最大 50 | + +**响应示例** + +```json +{ + "code": 0, + "message": "ok", + "data": { + "messages": [ + { + "id": 10086, + "activity_id": 42, + "user_id": 1001, + "star_id": 87, + "nickname": "小星星", + "avatar_url": "https://oss.example.com/avatars/1001.png", + "content": "生日快乐!永远支持你~", + "created_at": 1750492800000 + }, + { + "id": 10085, + "activity_id": 42, + "user_id": 1002, + "star_id": 87, + "nickname": "月光宝盒", + "avatar_url": "", + "content": "星光不问赶路人~", + "created_at": 1750492700000 + } + ], + "page": 1, + "page_size": 20, + "total": 2 + } +} +``` + +**错误码** + +| code | HTTP | message | 说明 | +| ---- | ---- | ------------------------ | ----------------------------- | +| 5 | 404 | `活动不存在` | activity_id 无效 | +| 5 | 404 | `活动不存在` | 活动已结束超过 30 天 | +| 16 | 401 | `未授权,请先登录` | token 缺失 / 失效 | + +--- + +### 5.4 接口 2:发送一条留言 + +**适用场景**:用户在 `MessageInput` 输入文本后点击发送箭头(当前 `index.vue` 的 `handleSendMessage` 调用点)。 + +``` +POST /api/v1/activities/{activity_id}/messages +``` + +**请求头**:`Content-Type: application/json` + +**Path 参数** + +| 名称 | 类型 | 必填 | 说明 | +| ------------ | ------ | ---- | -------- | +| `activity_id` | int64 | 是 | 活动 ID | + +**Body** + +```json +{ + "content": "永远支持你!生日快乐 🎂" +} +``` + +| 字段 | 类型 | 必填 | 限制 | +| --------- | ------ | ---- | --------------------------------- | +| `content` | string | 是 | 长度 1-500;去前后空格后非空 | + +**响应示例(成功)** + +```json +{ + "code": 0, + "message": "ok", + "data": { + "message": { + "id": 10087, + "activity_id": 42, + "user_id": 2001, + "star_id": 87, + "nickname": "爱战战", + "avatar_url": "https://oss.example.com/avatars/2001.png", + "content": "永远支持你!生日快乐 🎂", + "created_at": 1750492900000 + } + } +} +``` + +**业务规则校验** + +| 校验项 | 失败 code | message | +| ----------------------- | --------- | -------------------------------- | +| 活动状态非 active | 7 | `活动未开始` / `活动已结束` | +| `content` 长度不合法 | 3 | `留言内容长度需在 1-500 字之间` | +| `content` 含敏感词 | 7 | `留言包含不当内容,请修改` | +| 单用户单活动 1 分钟内 > 5 条 | 8 | `留言太频繁,请稍后再试` | +| 单用户单活动累计 > 100 条 | 8 | `当前活动留言已达上限` | + +> 上限和频控值通过配置项注入(参考 `notificationService/configs/config.yaml` 的模式),避免硬编码。 + +**错误码** + +| code | HTTP | message | 说明 | +| ---- | ---- | ------------------ | -------------------------- | +| 3 | 400 | `请求参数错误` | JSON 解析失败 / 字段缺失 | +| 5 | 404 | `活动不存在` | activity_id 无效 | +| 7 | 403 | `活动已结束` | 状态非 active | +| 8 | 429 | `留言太频繁` | 频控触发 | +| 16 | 401 | `未授权` | token 失效 | + +--- + +### 5.5 接口 3:增量获取最新留言(轮询用) + +**适用场景**:前端使用 `setInterval` 轮询新留言,实现"气泡墙实时滚动"效果。对标 `GetLatestContributions` 接口模式。 + +``` +GET /api/v1/activities/{activity_id}/messages/latest?since_timestamp=1750492800000&since_id=10086&limit=20 +``` + +**Path 参数** + +| 名称 | 类型 | 必填 | 说明 | +| ------------ | ------ | ---- | -------- | +| `activity_id` | int64 | 是 | 活动 ID | + +**Query 参数** + +| 名称 | 类型 | 必填 | 默认 | 说明 | +| ----------------- | ------ | ---- | ---- | ------------------------------------- | +| `since_timestamp` | int64 | 否 | 0 | 毫秒时间戳,返回严格大于该值的新留言 | +| `since_id` | int64 | 否 | 0 | 同毫秒内按 id 二次排序 | +| `limit` | int | 否 | 20 | 最多返回多少条,最大 50 | + +> 前端轮询时可复用 `frontend/pages/support-activity/composables/useContributionPolling.js` 的 `latestTimestamp` / `latestId` 游标模式。 + +**响应示例** + +```json +{ + "code": 0, + "message": "ok", + "data": { + "messages": [ + { + "id": 10087, + "activity_id": 42, + "user_id": 2001, + "star_id": 87, + "nickname": "爱战战", + "avatar_url": "https://oss.example.com/avatars/2001.png", + "content": "永远支持你!生日快乐 🎂", + "created_at": 1750492900000 + } + ] + } +} +``` + +无新数据时 `data.messages` 为空数组(不返回 404,避免轮询噪声)。 + +--- + +## 6. 错误码扩展 + +新增到 `backend/pkg/errors/errors.go`: + +```go +// 活动留言相关错误 +var ( + ErrActivityMessageNotFound = errors.New("活动留言不存在") + ErrActivityMessageTooFrequent = errors.New("留言太频繁,请稍后再试") + ErrActivityMessageLimitReached = errors.New("当前活动留言已达上限") + ErrActivityMessageContentEmpty = errors.New("留言内容不能为空") + ErrActivityMessageContentTooLong = errors.New("留言内容过长,最多500字") + ErrActivityMessageContentInvalid = errors.New("留言内容包含不当内容") + ErrActivityMessageActivityInactive = errors.New("活动不在进行中") +) +``` + +并在 `ToGRPCCode` 函数中追加映射: + +| error | gRPC code | +| ---------------------------------- | -------------------- | +| `ErrActivityMessageNotFound` | `NotFound (5)` | +| `ErrActivityMessageTooFrequent` | `ResourceExhausted (8)` | +| `ErrActivityMessageLimitReached` | `ResourceExhausted (8)` | +| `ErrActivityMessageContentEmpty` | `InvalidArgument (3)` | +| `ErrActivityMessageContentTooLong` | `InvalidArgument (3)` | +| `ErrActivityMessageContentInvalid` | `PermissionDenied (7)` | +| `ErrActivityMessageActivityInactive` | `PermissionDenied (7)` | + +并把中文错误提示加到 `gateway/pkg/response/response.go` 的 `errorMap`: + +```go +"活动留言不存在": "活动留言不存在", +"留言太频繁": "留言太频繁,请稍后再试", +"留言内容不能为空": "留言内容不能为空", +"留言内容过长": "留言内容过长,最多500字", +"留言内容包含不当内容": "留言内容包含不当内容,请修改", +"活动不在进行中": "活动未开始或已结束", +``` + +--- + +## 7. 分层实现(按 CLAUDE.md 接口开发规范) + +``` +HTTP 请求 (Gin) + │ + ▼ +gateway/controller/activity_controller.go ← 新增 3 个 handler + Swagger 注解 + │ - ListActivityMessages + │ - CreateActivityMessage + │ - GetLatestActivityMessages + │ - convertActivityMessage* 响应转换函数 + │ + │ Dubbo / gRPC (Triple) + ▼ +services/activityService/provider/activity_provider.go ← 3 个 RPC 入口 + │ + ▼ +services/activityService/service/activity_service.go ← 业务编排(鉴权/频控/事务) + │ + ▼ +services/activityService/repository/activity_repository.go ← SQL 拼装(不写业务逻辑) + │ + ▼ +PostgreSQL (activity_messages 表) +``` + +### 7.1 待新增 / 修改文件清单 + +| 类型 | 文件 | 说明 | +| ---- | ------------------------------------------------------------------------------------------------------------------- | ------------------------------------------ | +| 新增 | `backend/scripts/migrations/migrate_create_activity_messages_table.sql` | 建表 + 索引 + 注释 | +| 新增 | `backend/proto/activity/activity_messages.proto` | proto 定义(追加 RPC + message) | +| 新增 | `backend/services/activityService/repository/activity_messages_repository.go` | 仓储层,封装 SQL 操作 | +| 修改 | `backend/services/activityService/repository/activity_repository.go` | 注册新的仓储接口(可选合并到同一文件) | +| 修改 | `backend/services/activityService/service/activity_service.go` | 新增 3 个业务方法 + 接口签名 | +| 修改 | `backend/services/activityService/provider/activity_provider.go` | 新增 3 个 RPC 入口 + 日志 | +| 修改 | `backend/gateway/controller/activity_controller.go` | 新增 3 个 HTTP handler + 转换函数 + Swagger 注解 | +| 修改 | `backend/gateway/router/router.go` | 注册 3 条路由到 `activities` 组 | +| 修改 | `backend/pkg/errors/errors.go` | 新增 7 个错误变量 + ToGRPCCode 映射 | +| 修改 | `backend/gateway/pkg/response/response.go` | 新增 6 条中文错误映射 | +| 修改 | `backend/services/activityService/configs/config.yaml` | 新增 `message_rate_limit_per_min: 5` 等频控配置 | +| 修改 | `frontend/utils/api.js` | 新增 `listActivityMessagesApi`、`createActivityMessageApi`、`getLatestActivityMessagesApi` 三个 API 函数 | +| 修改 | `frontend/pages/support-activity/index.vue` | 把 `messageList` mock 替换成接口数据源,`handleSendMessage` 改为调用 `createActivityMessageApi` | +| 修改 | `frontend/pages/support-activity/components/MessageBoard.vue` | 字段映射:`time` → 用 `created_at` 格式化;`isSelf` 由 `user_id === currentUser.id` 计算 | + +### 7.2 Service 层业务逻辑要点 + +```go +// CreateActivityMessage 节选伪代码 +func (s *activityService) CreateActivityMessage(ctx context.Context, req *pb.CreateActivityMessageRequest) (*pb.CreateActivityMessageResponse, error) { + // 1. 入参校验 + if strings.TrimSpace(req.Content) == "" { return nil, appErrors.ErrActivityMessageContentEmpty } + if utf8.RuneCountInString(req.Content) > 500 { return nil, appErrors.ErrActivityMessageContentTooLong } + + // 2. 活动存在性 + 状态 + activity, err := s.activityRepo.GetActivityByID(req.ActivityId) + if err != nil { return nil, err } + if activity == nil { return nil, appErrors.ErrActivityNotFound } + if activity.Status != "active" { return nil, appErrors.ErrActivityMessageActivityInactive } + + // 3. 频控(Redis INCR + EXPIRE) + rateKey := fmt.Sprintf("msg:rate:%d:%d", req.ActivityId, req.UserId) + count, _ := s.redisClient.Incr(ctx, rateKey).Result() + if count == 1 { s.redisClient.Expire(ctx, rateKey, 60*time.Second) } + if count > cfg.MessageRateLimitPerMin { return nil, appErrors.ErrActivityMessageTooFrequent } + + // 4. 累计上限 + total, _ := s.activityMessagesRepo.CountByUserActivity(ctx, req.ActivityId, req.UserId) + if total >= cfg.MessageLimitPerActivity { return nil, appErrors.ErrActivityMessageLimitReached } + + // 5. 敏感词校验(可后续接入内容审核服务) + if containsBannedWord(req.Content) { return nil, appErrors.ErrActivityMessageContentInvalid } + + // 6. 写入 + 回查用户昵称 / 头像 + now := time.Now().UnixMilli() + msgID, err := s.activityMessagesRepo.Insert(ctx, repository.ActivityMessage{ + ActivityId: req.ActivityId, + UserId: req.UserId, + StarId: req.StarId, + Content: req.Content, + Status: 0, + CreatedAt: now, + UpdatedAt: now, + }) + if err != nil { return nil, err } + + // 7. 返回带 nickname / avatar_url 的展示模型 + profile, _ := s.userRPCClient.GetFanProfile(ctx, req.UserId, req.StarId) + return &pb.CreateActivityMessageResponse{ + Base: &pbCommon.BaseResponse{Code: uint32(codes.OK), Message: "ok"}, + Message: convertToPBMessage(msgID, req, profile, now), + }, nil +} +``` + +--- + +## 8. 前端集成要点 + +### 8.1 字段映射表 + +| 后端字段 | 前端 MessageBoard props 字段 | 转换说明 | +| ---------------- | ---------------------------- | ----------------------------------------- | +| `id` | `id` | 直接使用 | +| `nickname` | `user` | 直接使用 | +| `avatar_url` | `avatar` | 直接使用 | +| `content` | `content` | 直接使用 | +| `created_at`(ms) | `time` | 前端用 `formatRelativeTime()` 工具格式化 | +| — | `isSelf` | 前端自己判断 `user_id === currentUser.id` | + +### 8.2 API 工具函数模板(追加到 `frontend/utils/api.js`) + +```js +// 列出活动留言 +export function listActivityMessagesApi(activityId, page = 1, pageSize = 20) { + return request({ + url: `/api/v1/activities/${activityId}/messages?page=${page}&page_size=${pageSize}`, + method: 'GET', + }) +} + +// 发送活动留言 +export function createActivityMessageApi(activityId, content) { + return request({ + url: `/api/v1/activities/${activityId}/messages`, + method: 'POST', + data: { content }, + }) +} + +// 增量获取最新留言(轮询用) +export function getLatestActivityMessagesApi(activityId, sinceTimestamp = 0, sinceId = 0, limit = 20) { + return request({ + url: `/api/v1/activities/${activityId}/messages/latest?since_timestamp=${sinceTimestamp}&since_id=${sinceId}&limit=${limit}`, + method: 'GET', + }) +} +``` + +### 8.3 `index.vue` 集成建议 + +```js +// 替换 mock data +const messageList = ref([]) +const currentUserId = ref(null) // 从 store/modules/user.js 取 + +async function loadMessages() { + const res = await listActivityMessagesApi(activityId.value) + if (res.code === 0) { + messageList.value = res.data.messages.map(m => ({ + id: m.id, + user: m.nickname, + avatar: m.avatar_url, + content: m.content, + time: m.created_at, + isSelf: m.user_id === currentUserId.value, + })) + } +} + +async function handleSendMessage(text) { + if (!text || !text.trim()) return + const res = await createActivityMessageApi(activityId.value, text.trim()) + if (res.code === 0) { + const m = res.data.message + messageList.value.push({ + id: m.id, + user: m.nickname, + avatar: m.avatar_url, + content: m.content, + time: m.created_at, + isSelf: true, + }) + uni.showToast({ title: '留言成功', icon: 'success' }) + } else { + uni.showToast({ title: res.message || '留言失败', icon: 'none' }) + } +} + +onMounted(loadMessages) +``` + +--- + +## 9. 缓存与性能 + +- **频控计数器**:Redis Key `msg:rate:{activity_id}:{user_id}`,TTL 60s。 +- **读取缓存**:首版不上缓存,直接走 DB(留言写入是追加为主,索引已覆盖热路径);QPS 上量后再考虑 30s 缓存。 +- **响应体大小**:每条留言 ≤ 500 字符 + JSON 字段约 200B,每页 20 条 ≤ 4KB,无需特殊压缩。 + +--- + +## 10. 测试用例 + +### 10.1 Service 层(必须覆盖) + +- [ ] 首次进入活动,能正确按 `created_at DESC, id DESC` 拉取留言 +- [ ] 发送空 content → `ErrActivityMessageContentEmpty` +- [ ] 发送超 500 字 content → `ErrActivityMessageContentTooLong` +- [ ] 活动状态 pending → `ErrActivityMessageActivityInactive` +- [ ] 同一用户 1 分钟内 > 5 条 → `ErrActivityMessageTooFrequent` +- [ ] 单用户单活动累计 > 100 条 → `ErrActivityMessageLimitReached` +- [ ] 增量拉取:`since_timestamp` / `since_id` 正确过滤历史 + +### 10.2 Handler 层 + +- [ ] Happy path:发送留言成功后响应体 `data.message.id` 正确 +- [ ] Error path:JWT 缺失 → 401 +- [ ] Error path:活动 ID 不存在 → 5 + 友好提示 + +--- + +## 11. 开放问题 / 后续迭代 + +| 主题 | 讨论 | +| -------------- | ------------------------------------------------------------------------------------------------------- | +| 敏感词审核 | 首版仅本地词表;后续接 `dify` 内容审核工作流(同 laser-card 模式) | +| 留言置顶 | 是否需要 `is_pinned` 字段?若需要,新增字段 + 列表查询加 ORDER BY is_pinned DESC, created_at DESC | +| 删除能力 | 用户是否能删除自己的留言?若是,新增 `DELETE /api/v1/activities/{id}/messages/{message_id}` | +| 跨活动聚合 | "我的留言"页面:`GET /api/v1/me/activity-messages?page=&page_size=` | +| WebSocket 推送 | 高 QPS 场景下用 WS 推送替代 HTTP 轮询;接口 3 保留作为降级方案 | + +--- + +## 12. 变更影响面(按 CLAUDE.md 自审规范) + +修复 / 新增本模块时必须回归: + +- [x] 现有 `activities` 表上的其他接口不受影响(新表 + 新路由) +- [x] `activity_service.go` 现有方法签名不变 +- [x] 新增的 3 条路由不与现有 `activities` 组冲突 +- [x] `pkg/errors` 新增错误变量不影响现有 `ToGRPCCode` 行为 +- [x] `gateway/pkg/response` 的 errorMap 是追加,不覆盖现有映射 +- [x] `frontend/pages/support-activity/index.vue` 已有 `ContributionList`、`ActionBar` 等组件,与 `MessageBoard` 数据流独立 +- [x] PostgreSQL 序列 `activity_messages_id_seq` 起始 10000,符合 CLAUDE.md 测试数据预留规范 + +--- + +## 附录 A:Swagger 注解示例(直接复制到 controller) + +```go +// ListActivityMessages 列出活动留言 +// @Summary 列出活动留言 +// @Description 分页获取活动留言列表(最新在上) +// @Tags activities +// @Accept json +// @Produce json +// @Security BearerAuth +// @Param activity_id path int64 true "活动ID" +// @Param page query int false "页码,默认1" +// @Param page_size query int false "每页数量,默认20,最大50" +// @Success 200 {object} response.Response +// @Router /api/v1/activities/{activity_id}/messages [get] +func (ctrl *ActivityController) ListActivityMessages(c *gin.Context) { /* ... */ } + +// CreateActivityMessage 发送活动留言 +// @Summary 发送活动留言 +// @Description 用户在应援活动页面发送一条祝福留言 +// @Tags activities +// @Accept json +// @Produce json +// @Security BearerAuth +// @Param activity_id path int64 true "活动ID" +// @Param request body object{content=string} true "留言内容" +// @Success 200 {object} response.Response +// @Router /api/v1/activities/{activity_id}/messages [post] +func (ctrl *ActivityController) CreateActivityMessage(c *gin.Context) { /* ... */ } + +// GetLatestActivityMessages 增量获取最新留言 +// @Summary 增量获取最新留言 +// @Description 用于前端轮询获取活动最新留言(since_timestamp + since_id 游标) +// @Tags activities +// @Accept json +// @Produce json +// @Security BearerAuth +// @Param activity_id path int64 true "活动ID" +// @Param since_timestamp query int64 false "毫秒时间戳,返回严格大于此值的新记录" +// @Param since_id query int64 false "ID筛选,配合since_timestamp使用" +// @Param limit query int false "返回数量,默认20,最大50" +// @Success 200 {object} response.Response +// @Router /api/v1/activities/{activity_id}/messages/latest [get] +func (ctrl *ActivityController) GetLatestActivityMessages(c *gin.Context) { /* ... */ } +``` + +## 附录 B:迁移脚本完整版 + +```sql +-- ============================================================ +-- Activity Messages Table +-- ============================================================ +BEGIN; + +CREATE TABLE IF NOT EXISTS public.activity_messages ( + id BIGSERIAL PRIMARY KEY, + activity_id BIGINT NOT NULL, + user_id BIGINT NOT NULL, + star_id BIGINT NOT NULL, + content VARCHAR(500) NOT NULL, + status SMALLINT NOT NULL DEFAULT 0, + created_at BIGINT NOT NULL, + updated_at BIGINT NOT NULL, + deleted_at BIGINT, + + CONSTRAINT fk_activity_messages_activity + FOREIGN KEY (activity_id) REFERENCES public.activities(id) ON DELETE CASCADE, + CONSTRAINT fk_activity_messages_user + FOREIGN KEY (user_id) REFERENCES public.users(id), + CONSTRAINT fk_activity_messages_star + FOREIGN KEY (star_id) REFERENCES public.stars(star_id) +); + +CREATE SEQUENCE IF NOT EXISTS activity_messages_id_seq START WITH 10000; + +CREATE INDEX IF NOT EXISTS idx_activity_messages_activity_created + ON public.activity_messages (activity_id, created_at DESC, id DESC) + WHERE deleted_at IS NULL; + +CREATE INDEX IF NOT EXISTS idx_activity_messages_user_created + ON public.activity_messages (user_id, created_at DESC) + WHERE deleted_at IS NULL; + +CREATE INDEX IF NOT EXISTS idx_activity_messages_activity_incr + ON public.activity_messages (activity_id, id DESC) + WHERE deleted_at IS NULL; + +COMMENT ON TABLE public.activity_messages IS '活动留言表'; +COMMENT ON COLUMN public.activity_messages.id IS '主键,自增'; +COMMENT ON COLUMN public.activity_messages.activity_id IS 'FK -> activities.id'; +COMMENT ON COLUMN public.activity_messages.user_id IS '留言用户 ID'; +COMMENT ON COLUMN public.activity_messages.star_id IS '所属明星/星球 ID'; +COMMENT ON COLUMN public.activity_messages.content IS '留言正文,1-500 字'; +COMMENT ON COLUMN public.activity_messages.status IS '0=正常|1=隐藏|2=已删除'; +COMMENT ON COLUMN public.activity_messages.created_at IS '留言时间,毫秒时间戳'; +COMMENT ON COLUMN public.activity_messages.updated_at IS '更新时间,毫秒时间戳'; +COMMENT ON COLUMN public.activity_messages.deleted_at IS '软删除时间'; + +-- 健康检查(部署后跑一次) +SELECT + sequencename, + last_value, + (SELECT COALESCE(MAX(id), 0) FROM public.activity_messages) AS table_max_id, + last_value >= (SELECT COALESCE(MAX(id), 0) FROM public.activity_messages) AS is_healthy +FROM pg_sequences +WHERE sequencename = 'activity_messages_id_seq'; + +COMMIT; +``` \ No newline at end of file diff --git a/frontend/pages/support-activity/components/MessageInput.vue b/frontend/pages/support-activity/components/MessageInput.vue index 601186d..231fd8d 100644 --- a/frontend/pages/support-activity/components/MessageInput.vue +++ b/frontend/pages/support-activity/components/MessageInput.vue @@ -20,6 +20,11 @@ :src="avatar || defaultAvatar" mode="aspectFill" /> + @@ -47,6 +52,7 @@