--- req_id: REQ-USR-004 date: 2026-04-30 round: 1 reviewer: superpower-code-reviewer --- # Review: REQ-USR-004 — round 1 ## 结论 approve ## Must-fix (无) ## Nice-to-have - backend/src/main/java/com/xly/erp/module/mod/service/impl/ModuleServiceImpl.java:33 + UserServiceImpl.java:72 — 移除 fallback 后 `private final StubSecurityProperties stub` 字段已无引用。按 CLAUDE.md surgical-changes 原则建议删除字段 + 构造器参数(测试 setUp 也需同步)。本期保留亦可,留做后续 housekeeping。 - backend/src/test/java/com/xly/erp/module/usr/security/LoginAttemptStoreTest.java:44-59 — `lockExpiresAfterDuration` 只验证另一个 fresh user 在 T+20min 未锁,未驱动同 userName 跨过 lockExpireAt 验证 records.remove 自动清除路径。spec line 117 `loginAfterLockExpired_resetsCounter` 要求;建议加测试用 Clock 推进到 T+15min 以上的同 key 路径。 - ModuleControllerIT / UserControllerIT 的 PUT 无 JWT 闭环回归断言只覆盖单列(sModuleNameZh / sUserName)unchanged;建议同时断 sCreatedBy / tCreateDate 也保持以增强回归覆盖。 - backend/src/main/java/com/xly/erp/module/usr/dto/LoginDTO.java:17 — `version` 字段标 `@NotBlank`,但 spec § 输入 写"version 仅记录, 不参与校验"。建议二选一:去 @NotBlank 或更新 spec 标注必填。 - backend/src/test/java/com/xly/erp/module/usr/controller/AuthControllerIT.java `loginWithEmptyBody_returns40001` 发空对象同时触发 3 个 @NotBlank;建议细粒度补一个仅缺 sUserName 的负例,验证 GlobalExceptionHandler 的字段定位文案。 - AuthControllerIT.loginAfter5WrongPasswords (line 127) 的 `loginAttemptStore.clearFailures("锁定用户")` 在测试体末尾才执行;assertion 失败时跳过会污染跨测试状态。建议挪到 @AfterEach 或用唯一 userName。 ## 反例 / 测试覆盖缺口 Phase A: 5 controller IT + 9 service 单测 + 5 store 单测 + 2 mapper IT 全部到位。 Phase B: 4 ModuleControllerIT + 3 UserControllerIT stub 闭环全部更新。 - SecurityConfig 收紧到仅 `POST /api/usr/auth/login` permitAll,其余 authenticated;JwtAuthenticationEntryPoint 把未认证转 `code=20001`。 - service 层 `stub.getStubUserNo()` 引用和 `// REQ-MOD-001 stub: see USR-004 follow-up` 锚点全部从代码中清除(grep 0 结果)。 - JWT 双 token:accessToken 8h / refreshToken 30d,二者 subject 相同但内容不同。 - 锁定路径:密码正确但锁定中仍返回 42301(spec § 业务规则 #4 ✓)。 - 信息泄漏防护:用户不存在 / 密码错均返回 40101(不区分)。 - LoginVO 排除 sPasswordHash,AuthControllerIT 显式断言 `data.user.has("sPasswordHash")` 为 false。 - 149/149 全绿(mvn test via scripts/test.sh)。 非阻塞缺口:见 nice-to-have 6 项,建议 module-report 阶段一次性批量补齐。