使用 Lua 脚本原子设置计数与过期,修复 TTL 缺失\n支持 fail-open/fail-close 并对优惠码验证启用 fail-close\n新增单元与集成测试覆盖关键分支\n\n测试:go test ./...
1.7 KiB
1.7 KiB
Context
限流中间件当前采用 INCR 后 EXPIRE 的两步操作,且未处理 EXPIRE 失败,导致计数 key 可能没有过期时间。该情况一旦发生,计数会持续累加,触发长期限流并造成 Redis key 膨胀。
Goals / Non-Goals
- Goals:
- 原子化 Redis 计数与过期设置
- 修复 TTL 缺失的历史 key
- 支持按接口配置 Redis 故障策略(fail-open/fail-close)
- 为需要强制保护的接口启用 fail-close
- Non-Goals:
- 改变现有固定窗口限流算法
- 调整限流 key 格式或前缀
- 引入新的外部依赖
Decisions
- 使用 Lua 脚本在 Redis 内部原子执行
INCR、TTL与PEXPIRE - 过期时间统一采用毫秒精度窗口(
window.Milliseconds()向下取整)以保持精度一致 - 当毫秒窗口小于 1 时,按 1ms 设置过期,避免 0 导致立即过期
- 当
count == 1或TTL == -1时设置过期,避免刷新已有 TTL - 新增
RateLimitOptions并提供LimitWithOptions,由调用方显式配置故障策略 Limit默认使用 fail-open 以保持兼容- 当 fail-close 生效时,Redis 执行失败直接返回 429
Alternatives considered
- 使用
MULTI/EXEC事务封装INCR+EXPIRE:原子性可保证,但无法在同一事务内便捷修复TTL == -1,且仍需额外判断逻辑 - 使用
SET+EX/NX组合:无法保留计数累加语义
Risks / Trade-offs
- Lua 脚本会带来轻微 CPU 开销,但可接受
- TTL 修复会在首次访问时设定过期,可能缩短历史脏 key 的“无限期”状态,这是期望的修复效果
Migration Plan
- 上线后脚本在请求路径上自动修复 TTL 缺失的 key
- 如需回滚,恢复原有两步命令即可
Open Questions
- 无