364e688382
主要改动: backend 业务: - feat(error-codes): 统一错误码表 E1011/E1012 拆码 - E1011 AUTH_PASSWORD_WRONG: 本地密码错误 - E1012 AUTH_FIRST_LOGIN_PASSWORD_REQUIRED: 首次登录请先设置密码 - E1015 AUTH_OLD_PASSWORD_REQUIRED: 改密需要旧密码 - E1016 AUTH_OLD_PASSWORD_WRONG: 旧密码错误 - fix(agents): P0 降级放行时,如坐席已注册但未设密码,正确 raise 1012 (修复前会撞 1011 本地密码错误,与场景不符) - feat(approval): 审批模块 (T审批/A审批) - feat(config): approval_template_resource / approval_template_device 配置 - feat(main): /ready, /metrics, /version 端点(K8s 友好) backend 测试: - test(agents): 新增 test_agents.py — 3 个 Fix-4 降级登录测试 - 错误密码拒绝 - 缺密码拒绝 - 正确密码通过 pytest tests/test_agents.py → 3/3 通过 - test(conftest): 模块级 mock + slowapi 限流重置 + UTF-8 patch 解决 Windows pytest GBK 读 .env 失败 + 降级路径无法测试 仓库治理: - chore(gitignore): 排除 .workbuddy/memory/(workbuddy 本地记忆) - chore(docs): 重命名两份 IT 文档(前缀加智能区分版本) 部署与文档: - docs: RELEASE_NOTES_v0.5.0-beta.md / dashboard.html / 需求-发版预览页面 - docs: 部署、架构、PRD、安全、评审报告等同步 v0.5.0-beta - deploy-server: 打包脚本、nginx、docker-compose 版本号 bump 前端 (frontend-h5 / frontend-agent / frontend-admin / frontend-portal): - index.html / package.json 版本号与构建号 bump 自动验收(RELEASE_NOTES L100-104): - [x] pytest tests/test_agents.py -v → 3 passed - [x] grep Bs7ucT backend frontend-h5 frontend-agent → 无输出 - [x] grep AppException(101[123]) backend → 仅 1 处(登录场景 1012) - [ ] npm run build (frontend-h5 / frontend-agent) → 合并后跑 后续: 合并 feature/t-1-t4-merge → main,tag v0.5.0-beta
181 lines
6.5 KiB
Python
181 lines
6.5 KiB
Python
# =============================================================================
|
|
# 企微智能IT支持服务台 — 坐席降级登录测试
|
|
# =============================================================================
|
|
# 覆盖 P0 修复 Fix-4: 企微 API 不可达时,已注册坐席必须验证本地密码
|
|
# 创建日期: 2026-06-15 (Claude Code 补最小测试,因 WB 提交时未含此测试)
|
|
# =============================================================================
|
|
|
|
import pytest
|
|
import pytest_asyncio
|
|
from unittest.mock import AsyncMock, patch
|
|
|
|
from app.models.agent import Agent
|
|
from app.utils.error_codes import ErrorCode
|
|
from tests.conftest import create_test_agent
|
|
|
|
|
|
class TestAgentDegradedLogin:
|
|
"""P0 修复 Fix-4: 降级登录密码验证"""
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_degraded_login_wrong_password_rejected(
|
|
self, client, db_session, mock_redis, mock_wecom_instance
|
|
):
|
|
"""场景: 企微 API 不可达,坐席有 password_hash,登录用错密码 → 拒绝
|
|
|
|
验证:
|
|
- 状态码非 200(或响应 code 非 0)
|
|
- 错误码属于 AUTH_PASSWORD_WRONG 类(1011 当前,2006 改完后)
|
|
"""
|
|
# 1. 预置坐席:有 password_hash
|
|
import bcrypt
|
|
|
|
correct_pw = "CorrectP@ss123"
|
|
pw_hash = bcrypt.hashpw(correct_pw.encode("utf-8"), bcrypt.gensalt()).decode(
|
|
"utf-8"
|
|
)
|
|
|
|
agent = create_test_agent(
|
|
user_id="degraded_agent_001",
|
|
name="降级坐席",
|
|
)
|
|
agent.password_hash = pw_hash
|
|
db_session.add(agent)
|
|
await db_session.flush()
|
|
|
|
# 2. 改写 conftest 模块级 mock 行为,让企微 API 抛异常(降级场景触发)
|
|
original_side_effect = mock_wecom_instance.get_user_info.side_effect
|
|
|
|
async def fail_get_user_info(*args, **kwargs):
|
|
raise Exception("企微 API 不可达 - 验证降级路径")
|
|
|
|
mock_wecom_instance.get_user_info.side_effect = fail_get_user_info
|
|
|
|
try:
|
|
# 3. 用错误密码登录
|
|
response = await client.post(
|
|
"/agents/login",
|
|
json={
|
|
"user_id": "degraded_agent_001",
|
|
"name": "降级坐席",
|
|
"password": "WrongPassword",
|
|
},
|
|
)
|
|
finally:
|
|
# 恢复默认 side_effect,避免污染后续测试
|
|
mock_wecom_instance.get_user_info.side_effect = original_side_effect
|
|
|
|
# 4. 断言:被拒绝
|
|
assert response.status_code in (200, 401, 403), (
|
|
f"预期被拒绝,实际 status={response.status_code}, body={response.text}"
|
|
)
|
|
body = response.json()
|
|
# 业务 code 应该非 0
|
|
assert body.get("code") != 0, f"预期失败 code,实际成功: {body}"
|
|
|
|
# 错误码: WB 修复后是 AUTH_PASSWORD_WRONG=2006,旧码 1011 也接受
|
|
error_code = body.get("code")
|
|
assert error_code in (
|
|
ErrorCode.AUTH_PASSWORD_WRONG.value, # 2006
|
|
1011, # 旧数字码,WB 接入 ErrorCode 前的过渡
|
|
), f"错误码不匹配: {error_code}, body={body}"
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_degraded_login_no_password_blocked(
|
|
self, client, db_session, mock_redis, mock_wecom_instance
|
|
):
|
|
"""场景: 企微 API 不可达,坐席有 password_hash,登录不传密码 → 拒绝"""
|
|
# 1. 预置坐席
|
|
import bcrypt
|
|
|
|
pw_hash = bcrypt.hashpw(b"AnyP@ss", bcrypt.gensalt()).decode("utf-8")
|
|
agent = create_test_agent(
|
|
user_id="degraded_agent_002",
|
|
name="降级坐席2",
|
|
)
|
|
agent.password_hash = pw_hash
|
|
db_session.add(agent)
|
|
await db_session.flush()
|
|
|
|
# 2. 改写 conftest 模块级 mock,让企微 API 抛异常
|
|
original_side_effect = mock_wecom_instance.get_user_info.side_effect
|
|
|
|
async def fail_get_user_info(*args, **kwargs):
|
|
raise Exception("企微 API 不可达 - 验证降级路径")
|
|
|
|
mock_wecom_instance.get_user_info.side_effect = fail_get_user_info
|
|
|
|
try:
|
|
# 3. 不传 password 登录
|
|
response = await client.post(
|
|
"/agents/login",
|
|
json={
|
|
"user_id": "degraded_agent_002",
|
|
"name": "降级坐席2",
|
|
},
|
|
)
|
|
finally:
|
|
mock_wecom_instance.get_user_info.side_effect = original_side_effect
|
|
|
|
# 4. 断言:被拒绝
|
|
body = response.json()
|
|
assert body.get("code") != 0, f"预期被拒绝: {body}"
|
|
error_code = body.get("code")
|
|
# 2006 (AUTH_PASSWORD_WRONG) 或 1011 (旧码)
|
|
assert error_code in (
|
|
ErrorCode.AUTH_PASSWORD_WRONG.value,
|
|
1011,
|
|
), f"错误码不匹配: {error_code}, body={body}"
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_degraded_login_correct_password_succeeds(
|
|
self, client, db_session, mock_redis, mock_wecom_instance
|
|
):
|
|
"""场景: 企微 API 不可达,坐席有 password_hash,登录用对密码 → 成功
|
|
|
|
验证降级路径正常工作时,正确密码可以登录
|
|
"""
|
|
import bcrypt
|
|
|
|
correct_pw = "CorrectP@ss456"
|
|
pw_hash = bcrypt.hashpw(correct_pw.encode("utf-8"), bcrypt.gensalt()).decode(
|
|
"utf-8"
|
|
)
|
|
|
|
agent = create_test_agent(
|
|
user_id="degraded_agent_003",
|
|
name="降级坐席3",
|
|
)
|
|
agent.password_hash = pw_hash
|
|
db_session.add(agent)
|
|
await db_session.flush()
|
|
|
|
# 改写 conftest 模块级 mock,让企微 API 抛异常
|
|
original_side_effect = mock_wecom_instance.get_user_info.side_effect
|
|
|
|
async def fail_get_user_info(*args, **kwargs):
|
|
raise Exception("企微 API 不可达 - 验证降级路径")
|
|
|
|
mock_wecom_instance.get_user_info.side_effect = fail_get_user_info
|
|
|
|
try:
|
|
response = await client.post(
|
|
"/agents/login",
|
|
json={
|
|
"user_id": "degraded_agent_003",
|
|
"name": "降级坐席3",
|
|
"password": correct_pw,
|
|
},
|
|
)
|
|
finally:
|
|
mock_wecom_instance.get_user_info.side_effect = original_side_effect
|
|
|
|
# 降级 + 正确密码应能登录
|
|
body = response.json()
|
|
assert body.get("code") == 0, (
|
|
f"预期降级登录成功,实际失败: {body}"
|
|
)
|
|
assert "token" in body.get("data", {}), (
|
|
f"响应缺 token: {body}"
|
|
)
|