.agents/design/api/s3-object-key-authorization.md
FastGPT 的私有对象存储 key 是 bucket 内的全局路径字符串,例如:
chat/<appId>/<uid>/<chatId>/<filename>dataset/<datasetId>/<filename>temp/<teamId>/<filename>helperBot/<type>/<userId>/<chatId>/<filename>这些路径段携带资源归属信息,但 S3 和下载代理本身只验证签名,不理解业务权限。因此任何来自请求体、query、工具参数或外部输入的 object key,在签发 URL、读取、预览、解析前,都必须先绑定到当前已鉴权的业务资源。
危险模式是“校验 A 资源,使用 B key”:
key、fileId、sourceId 传给 S3 签名或读取函数。不能把以下条件当作权限:
新增入口必须优先复用各 S3 source 的 key helper:
packages/service/common/s3/sources/chat/key.ts: parseChatFileS3Key / isAuthorizedChatFileS3Keypackages/service/common/s3/sources/dataset/key.ts: parseDatasetFileS3Key / isAuthorizedDatasetFileS3Keypackages/service/common/s3/sources/helperbot/key.ts: parseHelperBotFileS3Key / isAuthorizedHelperBotFileS3Keypackages/service/common/s3/sources/temp/key.ts: isAuthorizedTempFileS3Key这些 helper 只负责 key 结构解析和 key 与已鉴权上下文的绑定判断。业务权限仍由对应 auth 函数负责:
authChatCrud,再校验 chat key 的 appId + uid。authDatasetFileKey 从 dataset/<datasetId>/... 解析 datasetId,并复用 dataset 权限体系。teamId,再校验 temp/<teamId>/...。authCert 得到 userId,再校验 helperBot key 的 userId。当 API 或工具入口接收外部传入的 S3 key 时,必须满足以下规则:
parseApiInput 校验请求入参。teamId、appId、datasetId、uid 或 userId。isAuthorized*FileS3Key helper 绑定 key 与可信上下文。createExternalUrl、createGet*URL、jwtSignS3DownloadToken、downloadObject、getDatasetFileRawText、isObjectExists 等存储层能力。S3BaseBucket.createExternalUrl 是裸存储签名方法,只保证 token 有效,不做业务权限判断。调用方不能把它当成鉴权接口。
readDatasetSourceRawText 在 fileLocal 分支额外校验 sourceId 必须属于传入的 datasetId,用于防止未来新增入口绕过 API 层鉴权。
authDatasetFileKey 会先按 key 内的 datasetId 复用 dataset 权限体系,再检查对象是否存在。对象存在性不能出现在权限校验之前。
已排查当前主要 S3 签名/读取点:
teamId/datasetId/collectionId 权限边界。/api/system/file/* 代理只校验 token,它不是业务鉴权入口;安全性依赖 token 签发前的业务授权绑定。新增类似入口时至少补充以下测试: