feat: Implement public access for tenant content

- Add TenantOptionalAuth middleware to allow access to public content without requiring authentication.
- Introduce ListPublicPublished and PublicDetail methods in the content service to retrieve publicly accessible content.
- Create tenant_public HTTP routes for listing and showing public content, including preview and main asset retrieval.
- Enhance content tests to cover scenarios for public content access and permissions.
- Update specifications to reflect the new public content access features and rules.
This commit is contained in:
2025-12-22 16:29:44 +08:00
parent 266de2f75e
commit 39454458f1
17 changed files with 1010 additions and 17 deletions

View File

@@ -0,0 +1,178 @@
# Spec01 可执行 Backlog按接口/表/状态机/验收用例)
本文从 `backend/specs/spec01-gap-analysis.md` 的“差异点/未实现项”拆解为可落地的 backlog。每条尽量可独立开发、可验收、可回滚。
## 0. 约定(用于所有条目)
- **P0/P1/P2**优先级从高到低P0=阻塞核心目标P1=重要增强P2=可延后。
- **验收方式**默认用“service 测试 + http 层冒烟”覆盖;涉及对象存储/转码的条目允许先用 mock/本地 minio 方案。
- **租户隔离硬约束**:任何带 `id` 的资源访问都必须校验 `tenant_id` 边界。
## Epic A公开内容/游客访问(当前未落地)
当前 `/t/:tenantCode/v1/*` 默认强制“登录 + 必须是租户成员”,不满足 spec01 的“游客可浏览公开内容(若允许)”。
### A1P0, Middleware/API增加“公开读接口路由组”
- **新增路由组**(建议二选一):
1) `GET /t/:tenantCode/v1/public/contents`(公开列表)
2) `GET /t/:tenantCode/v1/public/contents/:contentID`(公开详情)
3) `GET /t/:tenantCode/v1/public/contents/:contentID/preview`(公开试看资源)
- **中间件策略**
-`TenantResolve`
- 可选 `TenantOptionalAuth`:有 token 则解析写 ctx无 token 允许继续(用于展示“已购/作者/已登录”差异)。
- **响应语义**
- 仅返回 `visibility=public``status=published` 的内容;
- `HasAccess` 在公开接口里定义为:`free || owner || purchased`(若无登录则恒为 false除非 free=真)。
- **验收用例**
- 未登录可拉取公开内容列表/详情/preview
- 未登录访问 `visibility=tenant_only/private` 返回权限错误或 404按统一策略定
- 已登录但非成员:公开内容可读;非公开内容不可读;购买/余额接口不可用。
### A2P1, State/Rule明确 `visibility` 与主资源访问关系
- **规则固化**(写入接口文档 + tests
- `visibility` 只控制“内容详情是否可见”主资源main role仍需 `free/owner/purchased`
- `public + free` 是否允许游客看正片:需要业务明确(建议允许,减少“公开但看不了”的困惑)。
- **验收用例**
- `public + price=0`:游客能拿到 main assets
- `public + price>0`:游客不能拿到 main assets但能拿到 preview assets。
## Epic BMediaAsset 上传/处理全链路(当前缺失)
已有 `media_assets``content_assets` 表,但缺少“上传→处理→对外下发”的闭环接口与状态机。
### B1P0, API上传初始化申请上传凭证/直传参数
- **新增接口**tenant_admin
- `POST /t/:tenantCode/v1/admin/media_assets/upload_init`
- **请求字段**
- `type`video/audio/image
- `content_type`mime可选
- `file_size`(可选,用于限额)
- `sha256`(可选,用于去重/审计)
- **返回字段**(按存储 provider 定):
- `asset_id`
- `upload_url` / `form_fields`S3 POST policy/ `headers`
- `expires_at`
- **DB**
- 创建 `media_assets(status=uploaded, provider/bucket/object_key/meta)`
- `object_key` 由后端生成,避免客户端指定路径。
- **验收用例**
- tenant_admin 调用成功返回可用上传信息;
- member/非 admin 调用被拒绝;
- asset 必须绑定正确 `tenant_id/user_id`
### B2P0, API/State上传完成回调触发处理并进入 processing
- **新增接口**tenant_admin 或 system
- `POST /t/:tenantCode/v1/admin/media_assets/:assetID/upload_complete`
- **行为**
- 校验 `asset.status=uploaded`
- 写入必要 metaduration/width/height 可后置);
- 状态迁移:`uploaded -> processing`
- 触发异步处理(先允许 stub写入任务表或发消息
- **验收用例**
- 重复调用幂等(第二次返回同一结果,不重复触发任务);
- 非法状态迁移返回明确错误码status conflict
### B3P1, API查询资源详情与列表
- **新增接口**tenant_admin
- `GET /t/:tenantCode/v1/admin/media_assets`
- `GET /t/:tenantCode/v1/admin/media_assets/:assetID`
- **查询字段**
- `status/type/created_at` 过滤;分页。
- **验收用例**
- 只能查本租户资源;
- deleted_at 过滤策略一致(默认不返回已删除)。
### B4P1, State Machine固化 media_assets 状态机与允许迁移
- **状态集合**`uploaded/processing/ready/failed/deleted`
- **允许迁移**
- uploaded → processing
- processing → ready | failed
- ready/failed → deleted软删
- **验收用例**
- 任意越权迁移失败;
- ready 才允许绑定到 `content_assets`(见 Epic C
## Epic C资源下发与防直链当前为“返回对象信息”未形成安全下发
目前内容资源接口返回 `models.MediaAsset`,可能包含 `bucket/object_key` 等内部定位信息spec01 希望通过“短时效播放凭证/签名 URL/token”下发并且 preview 与 main 资源彻底区分。
### C1P0, API/DTO资源下发改为“签名 URL/Token”响应
- **调整接口返回 DTO**
- `GET /t/:tenantCode/v1/contents/:contentID/preview`
- `GET /t/:tenantCode/v1/contents/:contentID/assets`
- **响应字段建议**
- `asset_id`
- `type`
- `play_url`(短时效)
- `expires_at`
- `meta`(可展示字段的白名单,如 duration/width/height
- **实现要点**
- 后端对 providerminio/s3/oss生成签名 URL
- 绝不返回可长期复用的直链或裸 object_key除非配置允许且仅内网
- **验收用例**
- 返回的 URL 具备过期时间;
- 无权限时不返回任何可用播放地址;
- 日志/审计中记录 tenant_id/content_id/user_id/role/asset_id。
### C2P1, Rule/Validation校验 preview 必须独立产物
- **约束**(二选一落库方式):
1) `media_assets.meta.variant=preview/main`(或 `is_preview`
2) 新增列 `media_assets.variant`(枚举)
- **绑定校验**
- `content_assets.role=preview` 只能绑定 `variant=preview`
- `role=main` 只能绑定 `variant=main`
- **验收用例**
- 用 main 资源绑定 preview 被拒绝;
- preview 秒数只对 preview 下发生效。
## Epic D异步退款/风控预留(当前 `refunding` 未使用)
### D1P2, State Machine引入 `refunding` 并定义状态迁移
- **订单状态机补齐**
- paid → refunding → refunded | failed
- **接口语义**
- `POST refund` 返回 `refunding`
- 单独的 job/worker 完成 `credit_refund + revoke access + status->refunded`
- **验收用例**
- 重复退款请求幂等;
- refunding 期间不得重复扣款/重复回收权益;
- 失败可重试(明确重试幂等键策略)。
## Epic E审计字段结构化当前充值操作者更多在 snapshot/remark
### E1P1, DB/APItenant_ledgers 增加操作者字段与业务引用字段
- **DB 变更**(建议):
- `tenant_ledgers.operator_user_id bigint NULL`
- `tenant_ledgers.biz_ref_type varchar(32) NULL`order/refund/topup/etc
- `tenant_ledgers.biz_ref_id bigint NULL`
-`(tenant_id, biz_ref_type, biz_ref_id, type)` 做唯一约束(或与 idempotency_key 二选一作为主幂等源)。
- **验收用例**
- 充值/退款/购买相关 ledger 必须写入 operator_user_idadmin/buyer/system
- 后台可按 operator_user_id 检索敏感操作流水。
### E2P1, DB/Ordertopup 结构化操作者字段(可选)
- **DB 变更**(二选一):
1) 在 `orders` 增加 `operator_user_id`(对 topup 更直观)
2) 保持在 snapshot但保证 ledger/operator 字段可追溯
- **验收用例**
- 导出订单时能明确区分“充值发起人”和“充值受益人”。
## 1. 建议交付顺序(最小闭环)
1) A1 → A2先把公开读能力与语义定死
2) B1 → B2 → B4上传/处理状态机闭环;任务系统可先 stub
3) C1 → C2把资源下发安全化再强制 preview 独立产物)
4) E1审计增强避免后续追溯成本
5) D1如确需异步退款/风控,再引入)

View File

@@ -0,0 +1,115 @@
# Spec01 vs 当前实现:功能对比与后续需求规则
本文基于 `backend/specs/spec01.md`,对照当前后端实现(数据表 / service / HTTP 路由),用于:
- 快速确认“已实现/部分实现/未实现”的范围边界;
- 固化后续需求补充时需要遵循的规则与约束,避免在多租户与资金链路上走偏。
## 1. 已实现(与 spec01 对齐)
### 1.1 多租户隔离与租户成员
- **租户上下文解析**:所有租户 API 按 `tenantCode` 解析租户并写入 ctxmiddleware
- **必须为租户成员**`/t/:tenantCode/v1/*` 默认强制登录 + 必须属于租户middleware不属于租户会直接拒绝。
- **角色模型**`tenant_users.role`member/tenant_admin存在租户管理接口有 role 校验。
- **加入租户**支持邀请码加入与申请加入tenantjoin 模块)。
### 1.2 余额体系(可用 + 冻结)与账本流水
- **账户维度**`tenant_users(tenant_id,user_id)`;字段包含 `balance``balance_frozen`
- **账本流水**`tenant_ledgers` 记录每次余额变更,含:
- `type`credit_topup / freeze / unfreeze / debit_purchase / credit_refund 等);
- `balance_before/after``frozen_before/after` 快照;
- `idempotency_key` 唯一约束tenant+user 维度)用于幂等落账。
- **一致性**:账本落地实现包含行锁与“余额/冻结余额不得为负”的不变量校验。
### 1.3 内容、定价与权益
- **内容模型**`contents`status/visibility/preview_seconds 等)。
- **内容定价**`content_prices`price_amount + discount_* 时间窗)。
- **订单快照**:购买时将价格/折扣/内容信息写入 `orders.snapshot`,避免改价影响历史订单。
- **权益模型**`content_access(tenant_id,user_id,content_id)`;购买授予 `active`,退款置为 `revoked`
- **试看**:区分 preview/main 资源角色;`/preview` 不要求购买,`/assets` 要求已购/免费/作者。
### 1.4 订单、购买、充值与退款
- **订单与明细**`orders` + `order_items`;支持 type=content_purchase/topup 与状态流转。
- **购买(余额支付)**:支持冻结→扣款(消耗冻结)→授予权益;并发靠行锁+冻结方案防止透支。
- **购买幂等**`idempotency_key` 支持“至多一次”购买语义;失败会写回滚标记并稳定返回“失败+已回滚”。
- **充值**:租户管理员可为租户成员单笔充值 + 批量充值;写 topup 订单 + credit_topup 账本。
- **退款**租户管理员可对已支付订单退款默认时间窗paid_at + 24h可 force 绕过;退款入账 + 回收权益。
- **后台订单查询**支持管理员按条件分页查询与导出CSV
## 2. 部分实现 / 需要明确的差异点
### 2.1 “游客/公开内容”未落地
spec01 允许“游客/未加入租户用户浏览公开内容(若允许)”。当前实现中,`/t/:tenantCode/v1/*` 默认要求登录且必须是租户成员。
若要支持“公开内容给非成员/未登录用户访问”,需要单独的路由与中间件策略(至少绕过 `TenantRequireMember`,并重新定义 `visibility=public` 的含义)。
### 2.2 订单状态 `refunding` 未使用
spec01 给出 `refunding` 中间态建议;当前退款实现通常直接落到 `refunded`(事务内完成退款账本+权益回收+订单更新)。
若未来需要异步退款(例如接第三方支付、风控审核),应补齐 `refunding` 状态的状态机与重试/幂等规则。
### 2.3 操作者审计字段不完全结构化
spec01 建议在订单侧保留 `operator_user_id` 等结构化字段。当前实现:
- 退款操作者落在 `orders.refund_operator_user_id`
- 充值操作者主要在 `orders.snapshot`/ledger remark 中体现(结构化程度较弱)。
若后续需要强审计/报表,应明确哪些操作必须“结构化字段 + 快照”双写。
## 3. 未实现spec01 提到但系统暂缺)
### 3.1 MediaAsset 上传/处理全链路
已存在 `media_assets` 表及内容关联 `content_assets`,但目前缺少:
- 上传/回调/转码/处理状态流转接口;
- 存储签名 URL/防盗链/短时 token 下发机制;
- “preview 资源必须是独立转码产物”的生产链路约束与校验。
## 4. 后续需求“规则”(建议强制遵循)
### 4.1 多租户规则(硬约束)
- 所有新增业务表必须具备 `tenant_id`,或能从主实体可推导且在查询/写入时强校验租户边界。
- 每个 HTTP API 必须明确:是否需要登录、是否需要租户成员、是否需要 tenant_admin、是否允许跨租户访问默认禁止
### 4.2 资金/余额规则(硬约束)
- 任何会改变余额/冻结余额的行为,都必须:
- 走账本tenant_ledgers并记录 before/after
- 定义唯一幂等键策略(稳定、可重放、可查证);
- 明确事务边界与失败补偿(尤其是“冻结成功但后续失败”的回滚路径)。
- 余额不允许为负;冻结余额不允许为负;这是系统级不变量,需求不得破坏。
### 4.3 订单规则(硬约束)
- 订单必须有快照(至少包含:内容标题、定价、折扣、成交价、时间、请求 idempotency_key
- 任何“可重试”的下单/退款/充值动作必须给出幂等语义:重复请求返回同一结果,不重复扣款/入账。
- 需求必须明确:失败时是否保留订单、订单处于何种终态、以及客户端应如何重试。
### 4.4 权益与资源访问规则(硬约束)
- “是否可看正片”只取决于:免费/作者/权益content_access=active客户端表现不构成安全措施。
- 试看资源必须与正片资源隔离(不同 asset role + 不同存储对象),需求不得允许复用正片资源做试看。
- 退款后权益必须立即失效revoked并且该规则优先级高于缓存/前端展示。
### 4.5 状态机与审计规则(建议)
- 对所有引入状态的实体content/order/media_asset/tenant_user需求必须附带
- 允许的状态集合;
- 允许的状态迁移;
- 幂等行为(重复迁移是否允许、返回什么)。
- 对敏感操作(充值/退款/调账/封禁),需求必须明确:
- 操作者字段operator_user_id是否结构化落库
- 原因字段是否必填;
- 审计可检索性(按租户/用户/时间/单据维度)。
## 5. 参考实现位置(便于后续对齐)
- 数据库迁移:
- `backend/database/migrations/20251216011456_tenant_users.sql`
- `backend/database/migrations/20251217223000_media_contents.sql`
- `backend/database/migrations/20251218120000_orders_ledgers.sql`
- 中间件(租户上下文/成员校验):
- `backend/app/middlewares/tenant.go`
- `backend/app/http/tenant/routes.manual.go`
- `backend/app/http/tenant_join/routes.manual.go`
- 业务服务(核心资金与订单链路):
- `backend/app/services/ledger.go`
- `backend/app/services/order.go`
- `backend/app/services/content.go`
- HTTP 路由(对外能力清单):
- `backend/app/http/tenant/*.go`
- `backend/app/http/tenant_join/*.go`
- `backend/app/http/super/*.go`(可选:平台侧)