feat(notification): 通知标题和正文支持多语言
- 通知静态配置支持 title/body i18n - 前端通知列表和详情页展示本地化内容 - 新增数据库迁移脚本 - 更新通知协议文档
This commit is contained in:
@@ -27,8 +27,12 @@ def test_load_static_notification_file_parses_valid_yaml(tmp_path: Path) -> None
|
||||
version: 1
|
||||
type: system
|
||||
status: published
|
||||
title: Welcome
|
||||
body: Welcome to the app.
|
||||
title:
|
||||
zh: 欢迎
|
||||
en: Welcome
|
||||
body:
|
||||
zh: 欢迎使用
|
||||
en: Welcome to the app.
|
||||
payload:
|
||||
action: open_route
|
||||
route: /points
|
||||
@@ -43,6 +47,8 @@ def test_load_static_notification_file_parses_valid_yaml(tmp_path: Path) -> None
|
||||
loaded = load_static_notification_file(file_path)
|
||||
|
||||
assert loaded.notification.source_key == "welcome_bonus"
|
||||
assert loaded.notification.title == {"zh": "欢迎", "en": "Welcome"}
|
||||
assert loaded.notification.body == {"zh": "欢迎使用", "en": "Welcome to the app."}
|
||||
assert loaded.notification.payload.action == "open_route"
|
||||
assert loaded.targets.mode == NotificationTargetMode.USER_IDS
|
||||
assert len(loaded.targets.user_ids or []) == 1
|
||||
@@ -58,8 +64,10 @@ def test_load_static_notification_file_parses_new_users(tmp_path: Path) -> None:
|
||||
version: 1
|
||||
type: system
|
||||
status: published
|
||||
title: Welcome
|
||||
body: You got points.
|
||||
title:
|
||||
zh: 欢迎
|
||||
body:
|
||||
zh: 你好
|
||||
payload:
|
||||
action: open_route
|
||||
route: /points
|
||||
@@ -85,8 +93,10 @@ def test_load_static_notification_file_parses_exist_users(tmp_path: Path) -> Non
|
||||
version: 1
|
||||
type: system
|
||||
status: published
|
||||
title: Come back
|
||||
body: We miss you.
|
||||
title:
|
||||
zh: 回来吧
|
||||
body:
|
||||
zh: 想你
|
||||
payload:
|
||||
action: none
|
||||
targets:
|
||||
@@ -110,8 +120,10 @@ def test_load_static_notification_file_parses_all_users(tmp_path: Path) -> None:
|
||||
version: 1
|
||||
type: system
|
||||
status: published
|
||||
title: Announcement
|
||||
body: Maintenance at midnight.
|
||||
title:
|
||||
zh: 公告
|
||||
body:
|
||||
zh: 午夜维护
|
||||
payload:
|
||||
action: none
|
||||
targets:
|
||||
@@ -134,8 +146,10 @@ def test_load_static_notification_file_rejects_invalid_targets(tmp_path: Path) -
|
||||
version: 1
|
||||
type: system
|
||||
status: published
|
||||
title: Invalid
|
||||
body: Invalid targets.
|
||||
title:
|
||||
zh: 无效
|
||||
body:
|
||||
zh: 无效
|
||||
payload:
|
||||
action: none
|
||||
targets:
|
||||
@@ -149,6 +163,88 @@ def test_load_static_notification_file_rejects_invalid_targets(tmp_path: Path) -
|
||||
load_static_notification_file(file_path)
|
||||
|
||||
|
||||
def test_load_static_notification_file_rejects_i18n_without_zh(
|
||||
tmp_path: Path,
|
||||
) -> None:
|
||||
file_path = tmp_path / "missing_zh.yaml"
|
||||
_write_yaml(
|
||||
file_path,
|
||||
"""
|
||||
notification:
|
||||
source_key: missing_zh
|
||||
version: 1
|
||||
type: system
|
||||
status: published
|
||||
title:
|
||||
en: Welcome
|
||||
body:
|
||||
zh: 正文
|
||||
payload:
|
||||
action: none
|
||||
targets:
|
||||
mode: all_users
|
||||
""",
|
||||
)
|
||||
|
||||
with pytest.raises(ValueError, match="Invalid static notification data"):
|
||||
load_static_notification_file(file_path)
|
||||
|
||||
|
||||
def test_load_static_notification_file_rejects_empty_i18n_text(
|
||||
tmp_path: Path,
|
||||
) -> None:
|
||||
file_path = tmp_path / "empty_i18n.yaml"
|
||||
_write_yaml(
|
||||
file_path,
|
||||
"""
|
||||
notification:
|
||||
source_key: empty_i18n
|
||||
version: 1
|
||||
type: system
|
||||
status: published
|
||||
title:
|
||||
zh: ""
|
||||
body:
|
||||
zh: 正文
|
||||
payload:
|
||||
action: none
|
||||
targets:
|
||||
mode: all_users
|
||||
""",
|
||||
)
|
||||
|
||||
with pytest.raises(ValueError, match="Invalid static notification data"):
|
||||
load_static_notification_file(file_path)
|
||||
|
||||
|
||||
def test_load_static_notification_file_rejects_unknown_i18n_locale(
|
||||
tmp_path: Path,
|
||||
) -> None:
|
||||
file_path = tmp_path / "unknown_locale.yaml"
|
||||
_write_yaml(
|
||||
file_path,
|
||||
"""
|
||||
notification:
|
||||
source_key: unknown_locale
|
||||
version: 1
|
||||
type: system
|
||||
status: published
|
||||
title:
|
||||
zh: 标题
|
||||
ja: タイトル
|
||||
body:
|
||||
zh: 正文
|
||||
payload:
|
||||
action: none
|
||||
targets:
|
||||
mode: all_users
|
||||
""",
|
||||
)
|
||||
|
||||
with pytest.raises(ValueError, match="Invalid static notification data"):
|
||||
load_static_notification_file(file_path)
|
||||
|
||||
|
||||
def test_load_static_notification_file_rejects_unknown_mode(tmp_path: Path) -> None:
|
||||
file_path = tmp_path / "bad_mode.yaml"
|
||||
_write_yaml(
|
||||
@@ -159,8 +255,10 @@ def test_load_static_notification_file_rejects_unknown_mode(tmp_path: Path) -> N
|
||||
version: 1
|
||||
type: system
|
||||
status: published
|
||||
title: Bad
|
||||
body: Bad mode.
|
||||
title:
|
||||
zh: 坏
|
||||
body:
|
||||
zh: 坏
|
||||
payload:
|
||||
action: none
|
||||
targets:
|
||||
@@ -184,8 +282,10 @@ def test_load_static_notification_file_rejects_new_users_with_user_ids(
|
||||
version: 1
|
||||
type: system
|
||||
status: published
|
||||
title: Bad
|
||||
body: Bad.
|
||||
title:
|
||||
zh: 坏
|
||||
body:
|
||||
zh: 坏
|
||||
payload:
|
||||
action: none
|
||||
targets:
|
||||
@@ -211,8 +311,10 @@ def test_load_static_notification_file_rejects_user_ids_without_list(
|
||||
version: 1
|
||||
type: system
|
||||
status: published
|
||||
title: Bad
|
||||
body: Bad.
|
||||
title:
|
||||
zh: 坏
|
||||
body:
|
||||
zh: 坏
|
||||
payload:
|
||||
action: none
|
||||
targets:
|
||||
@@ -235,8 +337,10 @@ def test_load_static_notification_documents_rejects_duplicate_source_key(
|
||||
version: 1
|
||||
type: system
|
||||
status: published
|
||||
title: First
|
||||
body: First body.
|
||||
title:
|
||||
zh: 第一
|
||||
body:
|
||||
zh: 第一
|
||||
payload:
|
||||
action: none
|
||||
targets:
|
||||
@@ -251,8 +355,10 @@ def test_load_static_notification_documents_rejects_duplicate_source_key(
|
||||
version: 1
|
||||
type: system
|
||||
status: published
|
||||
title: Second
|
||||
body: Second body.
|
||||
title:
|
||||
zh: 第二
|
||||
body:
|
||||
zh: 第二
|
||||
payload:
|
||||
action: none
|
||||
targets:
|
||||
@@ -275,8 +381,10 @@ def test_content_hash_changes_when_notification_changes(tmp_path: Path) -> None:
|
||||
version: 1
|
||||
type: system
|
||||
status: published
|
||||
title: Title A
|
||||
body: Body A.
|
||||
title:
|
||||
zh: 标题A
|
||||
body:
|
||||
zh: 正文A
|
||||
payload:
|
||||
action: none
|
||||
targets:
|
||||
@@ -291,8 +399,10 @@ def test_content_hash_changes_when_notification_changes(tmp_path: Path) -> None:
|
||||
version: 1
|
||||
type: system
|
||||
status: published
|
||||
title: Title B
|
||||
body: Body A.
|
||||
title:
|
||||
zh: 标题B
|
||||
body:
|
||||
zh: 正文A
|
||||
payload:
|
||||
action: none
|
||||
targets:
|
||||
@@ -319,8 +429,10 @@ def test_load_static_notification_file_supports_deleted_flag(tmp_path: Path) ->
|
||||
type: system
|
||||
status: revoked
|
||||
deleted: true
|
||||
title: Deleted
|
||||
body: Deleted body.
|
||||
title:
|
||||
zh: 已删
|
||||
body:
|
||||
zh: 已删
|
||||
payload:
|
||||
action: none
|
||||
targets:
|
||||
|
||||
Reference in New Issue
Block a user