
关于
在添加认证、处理用户输入、管理密钥、创建 API 端点或实现支付/敏感功能时使用。提供全面的安全检查清单和模式。
name: security-review description: 在添加认证、处理用户输入、使用密钥、创建 API 端点或实现支付/敏感功能时使用此技能。提供全面的安全检查清单和模式。 origin: ECC
安全审查技能
此技能确保所有代码遵循安全最佳实践并识别潜在漏洞。
何时激活
- 实现认证或授权
- 处理用户输入或文件上传
- 创建新的 API 端点
- 使用密钥或凭证
- 实现支付功能
- 存储或传输敏感数据
- 集成第三方 API
安全检查清单
1. 密钥管理
失败:绝对不要这样做
const apiKey = "sk-proj-xxxxx" // 硬编码密钥
const dbPassword = "password123" // 写在源代码中
通过:始终这样做
const apiKey = process.env.OPENAI_API_KEY
const dbUrl = process.env.DATABASE_URL
// 验证密钥存在
if (!apiKey) {
throw new Error('OPENAI_API_KEY not configured')
}
验证步骤
- [ ] 没有硬编码的 API 密钥、令牌或密码
- [ ] 所有密钥存储在环境变量中
- [ ]
.env.local已添加到 .gitignore - [ ] git 历史中没有密钥
- [ ] 生产密钥存储在托管平台(Vercel、Railway)
2. 输入验证
始终验证用户输入
import { z } from 'zod'
// 定义验证 schema
const CreateUserSchema = z.object({
email: z.string().email(),
name: z.string().min(1).max(100),
age: z.number().int().min(0).max(150)
})
// 处理前进行验证
export async function createUser(input: unknown) {
try {
const validated = CreateUserSchema.parse(input)
return await db.users.create(validated)
} catch (error) {
if (error instanceof z.ZodError) {
return { success: false, errors: error.errors }
}
throw error
}
}
文件上传验证
function validateFileUpload(file: File) {
// 大小检查(最大 5MB)
const maxSize = 5 * 1024 * 1024
if (file.size > maxSize) {
throw new Error('File too large (max 5MB)')
}
// 类型检查
const allowedTypes = ['image/jpeg', 'image/png', 'image/gif']
if (!allowedTypes.includes(file.type)) {
throw new Error('Invalid file type')
}
// 扩展名检查
const allowedExtensions = ['.jpg', '.jpeg', '.png', '.gif']
const extension = file.name.toLowerCase().match(/\.[^.]+$/)?.[0]
if (!extension || !allowedExtensions.includes(extension)) {
throw new Error('Invalid file extension')
}
return true
}
验证步骤
- [ ] 所有用户输入使用 schema 验证
- [ ] 文件上传受限(大小、类型、扩展名)
- [ ] 用户输入不直接用于查询
- [ ] 使用白名单验证(非黑名单)
- [ ] 错误消息不泄露敏感信息
3. SQL 注入防护
失败:绝对不要拼接 SQL
// 危险 - SQL 注入漏洞
const query = `SELECT * FROM users WHERE email = '${userEmail}'`
await db.query(query)
通过:始终使用参数化查询
// 安全 - 参数化查询
const { data } = await supabase
.from('users')
.select('*')
.eq('email', userEmail)
// 或使用原始 SQL
await db.query(
'SELECT * FROM users WHERE email = $1',
[userEmail]
)
验证步骤
- [ ] 所有数据库查询使用参数化查询
- [ ] SQL 中没有字符串拼接
- [ ] ORM/查询构建器正确使用
- [ ] Supabase 查询正确清理
4. 认证与授权
JWT 令牌处理
// 失败:错误做法:localStorage(易受 XSS 攻击)
localStorage.setItem('token', token)
// 通过:正确做法:httpOnly cookies
res.setHeader('Set-Cookie',
`token=${token}; HttpOnly; Secure; SameSite=Strict; Max-Age=3600`)
授权检查
export async function deleteUser(userId: string, requesterId: string) {
// 始终先验证授权
const requester = await db.users.findUnique({
where: { id: requesterId }
})
if (requester.role !== 'admin') {
return NextResponse.json(
{ error: 'Unauthorized' },
{ status: 403 }
)
}
// 执行删除
await db.users.delete({ where: { id: userId } })
}
行级安全(Supabase)
-- 在所有表上启用 RLS
ALTER TABLE users ENABLE ROW LEVEL SECURITY;
-- 用户只能查看自己的数据
CREATE POLICY "Users view own data"
ON users FOR SELECT
USING (auth.uid() = id);
-- 用户只能更新自己的数据
CREATE POLICY "Users update own data"
ON users FOR UPDATE
USING (auth.uid() = id);
验证步骤
- [ ] 令牌存储在 httpOnly cookies 中(非 localStorage)
- [ ] 敏感操作前进行授权检查
- [ ] Supabase 中启用行级安全
- [ ] 实现基于角色的访问控制
- [ ] 会话管理安全
5. XSS 防护
清理 HTML
兼容工具
Claude CodeCursor
标签
安全
