diff --git a/apps/web/messages/en.json b/apps/web/messages/en.json
index f350c10c..c2edfc7a 100644
--- a/apps/web/messages/en.json
+++ b/apps/web/messages/en.json
@@ -1930,6 +1930,49 @@
"field": "runtime gate pointer; active gates=0"
}
}
+ },
+ "hostOwnerDecisionRecordWriteupReview": {
+ "title": "Host Owner Decision Record Write-Up Review Checklist",
+ "subtitle": "Write-up packets still require read-only review. This only shows whether formal decision record write-up fields are readable and does not mark write-up completed, create or accept decision records, or open runtime gates.",
+ "checkLabel": "Write-up review",
+ "guardLabel": "No upgrade",
+ "items": {
+ "decisionSummaryReadable": {
+ "title": "Decision summary readable",
+ "body": "Confirm the write-up only organizes the human owner decision, risk acceptance boundary, and no-execution statement without adding approval semantics.",
+ "guard": "write-up review only; completed=0"
+ },
+ "scopeExpiryComplete": {
+ "title": "Scope and expiry complete",
+ "body": "Confirm scope, exclusions, observation intent, and expiry are readable so the formal record scope is not ambiguous.",
+ "guard": "record created=false"
+ },
+ "scanModeLimitsExplicit": {
+ "title": "Scan mode limits explicit",
+ "body": "Confirm observe-only, future active scan, and credentialed scan limits are explicit while not becoming scan authorization.",
+ "guard": "scan authorized=false"
+ },
+ "credentialBoundaryMetadataOnly": {
+ "title": "Credential boundary metadata only",
+ "body": "Confirm credential handling still keeps only metadata, owner, and retention boundary without requesting or storing sensitive material.",
+ "guard": "secret collection=false"
+ },
+ "maintenanceRollbackLinked": {
+ "title": "Maintenance and rollback linked",
+ "body": "Confirm maintenance window candidates, constraints, rollback owner, recovery path, and human contact remain traceable.",
+ "guard": "host update=false"
+ },
+ "validationEvidenceLinked": {
+ "title": "Validation evidence linked",
+ "body": "Confirm post-check metrics, baseline, evidence pointer, and human acceptance condition are linked to the write-up.",
+ "guard": "accepted=0"
+ },
+ "runtimeGateStillSeparate": {
+ "title": "Runtime gate still separate",
+ "body": "Confirm the runtime gate pointer still points to a separate follow-up gate and write-up review does not open gates.",
+ "guard": "active runtime gates=0; action buttons=false"
+ }
+ }
}
},
"tickets": {
diff --git a/apps/web/messages/zh-TW.json b/apps/web/messages/zh-TW.json
index 14cd198c..bca51eea 100644
--- a/apps/web/messages/zh-TW.json
+++ b/apps/web/messages/zh-TW.json
@@ -1931,6 +1931,49 @@
"field": "runtime gate pointer;active gates=0"
}
}
+ },
+ "hostOwnerDecisionRecordWriteupReview": {
+ "title": "主機 Owner Decision Record Write-Up Review Checklist",
+ "subtitle": "Write-up packets 後仍需要只讀核對。這裡只顯示正式 decision record 撰寫欄位是否可讀,不會標記 write-up completed、不會建立或接受 decision record、不會開 runtime gate。",
+ "checkLabel": "Write-up review",
+ "guardLabel": "不可升級",
+ "items": {
+ "decisionSummaryReadable": {
+ "title": "Decision summary readable",
+ "body": "確認 write-up 只整理 owner 人工判斷、風險接受邊界與不執行聲明,不新增批准語義。",
+ "guard": "write-up review only;completed=0"
+ },
+ "scopeExpiryComplete": {
+ "title": "Scope and expiry complete",
+ "body": "確認 scope、排除範圍、觀察目的與到期時間可讀,避免正式紀錄範圍不清。",
+ "guard": "record created=false"
+ },
+ "scanModeLimitsExplicit": {
+ "title": "Scan mode limits explicit",
+ "body": "確認 observe-only、future active scan 與 credentialed scan 的限制條件明確,且不代表掃描授權。",
+ "guard": "scan authorized=false"
+ },
+ "credentialBoundaryMetadataOnly": {
+ "title": "Credential boundary metadata only",
+ "body": "確認 credential handling 仍只保留 metadata、owner 與保存邊界,不要求或保存敏感素材。",
+ "guard": "secret collection=false"
+ },
+ "maintenanceRollbackLinked": {
+ "title": "Maintenance and rollback linked",
+ "body": "確認維護窗口候選、限制條件、rollback owner、復原路徑與人工聯絡點都可追。",
+ "guard": "host update=false"
+ },
+ "validationEvidenceLinked": {
+ "title": "Validation evidence linked",
+ "body": "確認 post-check metrics、baseline、evidence pointer 與人工驗收條件已連到 write-up。",
+ "guard": "accepted=0"
+ },
+ "runtimeGateStillSeparate": {
+ "title": "Runtime gate still separate",
+ "body": "確認 runtime gate pointer 仍指向獨立 follow-up gate,write-up review 不會開 gate。",
+ "guard": "active runtime gates=0;action buttons=false"
+ }
+ }
}
},
"tickets": {
diff --git a/apps/web/src/app/[locale]/iwooos/page.tsx b/apps/web/src/app/[locale]/iwooos/page.tsx
index af367544..f1ff7914 100644
--- a/apps/web/src/app/[locale]/iwooos/page.tsx
+++ b/apps/web/src/app/[locale]/iwooos/page.tsx
@@ -186,6 +186,13 @@ type HostOwnerDecisionRecordWriteupPacket = {
tone: 'steady' | 'warn' | 'locked'
}
+type HostOwnerDecisionRecordWriteupReviewItem = {
+ key: string
+ check: string
+ icon: typeof ShieldCheck
+ tone: 'steady' | 'warn' | 'locked'
+}
+
const postureMetrics: PostureMetric[] = [
{ key: 'overall', value: '58%', tone: 'warn' },
{ key: 'framework', value: '80-85%', tone: 'steady' },
@@ -437,6 +444,16 @@ const hostOwnerDecisionRecordWriteupPackets: HostOwnerDecisionRecordWriteupPacke
{ key: 'runtimeGatePointerWriteup', packet: 'WU7', icon: ShieldCheck, tone: 'locked' },
]
+const hostOwnerDecisionRecordWriteupReviewItems: HostOwnerDecisionRecordWriteupReviewItem[] = [
+ { key: 'decisionSummaryReadable', check: 'WR1', icon: ClipboardCheck, tone: 'warn' },
+ { key: 'scopeExpiryComplete', check: 'WR2', icon: Radar, tone: 'warn' },
+ { key: 'scanModeLimitsExplicit', check: 'WR3', icon: Activity, tone: 'locked' },
+ { key: 'credentialBoundaryMetadataOnly', check: 'WR4', icon: Lock, tone: 'locked' },
+ { key: 'maintenanceRollbackLinked', check: 'WR5', icon: Clock3, tone: 'warn' },
+ { key: 'validationEvidenceLinked', check: 'WR6', icon: CheckCircle2, tone: 'warn' },
+ { key: 'runtimeGateStillSeparate', check: 'WR7', icon: ShieldCheck, tone: 'locked' },
+]
+
const evidenceItems = [
'iwooos-posture-projection.snapshot.json',
'security-rollout-policy.snapshot.json',
@@ -1135,6 +1152,34 @@ function HostOwnerDecisionRecordWriteupCard({ item }: { item: HostOwnerDecisionR
)
}
+function HostOwnerDecisionRecordWriteupReviewCard({ item }: { item: HostOwnerDecisionRecordWriteupReviewItem }) {
+ const t = useTranslations('iwooos.hostOwnerDecisionRecordWriteupReview')
+ const Icon = item.icon
+ return (
+
+
+
+
+ {t('checkLabel')}
+
+
{item.check}
+
+
+ {t(`items.${item.key}.title` as never)}
+
+
+ {t(`items.${item.key}.body` as never)}
+
+
+
{t('guardLabel')}
+
+ {t(`items.${item.key}.guard` as never)}
+
+
+
+ )
+}
+
export default function IwoooSPage({ params }: { params: { locale: string } }) {
const t = useTranslations('iwooos')
@@ -1527,6 +1572,26 @@ export default function IwoooSPage({ params }: { params: { locale: string } }) {
+
+
+
{t('hostOwnerDecisionRecordWriteupReview.title')}
+
+ {t('hostOwnerDecisionRecordWriteupReview.subtitle')}
+
+
+
+ {hostOwnerDecisionRecordWriteupReviewItems.map(item => (
+
+ ))}
+
+
+
None:
"s2_27_iwooos_host_owner_decision_record_draft_review_checklist",
"s2_28_iwooos_host_owner_decision_record_draft_review_outcome_lanes",
"s2_29_iwooos_host_owner_decision_record_writeup_packets",
+ "s2_30_iwooos_host_owner_decision_record_writeup_review_checklist",
]
assert_equal(
"progress_delta_ledger.delta_ids",
@@ -488,6 +489,15 @@ def validate(root: Path) -> None:
"host_decision_record_validation_evidence_writeup_packet",
"host_decision_record_runtime_gate_pointer_writeup_packet",
]
+ expected_iwooos_host_owner_decision_record_writeup_review_checklist_item_ids = [
+ "host_decision_record_summary_writeup_review_check",
+ "host_decision_record_scope_writeup_review_check",
+ "host_decision_record_scan_mode_limits_writeup_review_check",
+ "host_decision_record_credential_boundary_writeup_review_check",
+ "host_decision_record_maintenance_rollback_writeup_review_check",
+ "host_decision_record_validation_evidence_writeup_review_check",
+ "host_decision_record_runtime_gate_writeup_review_check",
+ ]
assert_equal(
"iwooos_projection.summary.frontend_surface_coverage_group_count",
iwooos_projection["summary"]["frontend_surface_coverage_group_count"],
@@ -588,6 +598,11 @@ def validate(root: Path) -> None:
iwooos_projection["summary"]["host_owner_decision_record_writeup_packet_count"],
len(expected_iwooos_host_owner_decision_record_writeup_packet_ids),
)
+ assert_equal(
+ "iwooos_projection.summary.host_owner_decision_record_writeup_review_checklist_item_count",
+ iwooos_projection["summary"]["host_owner_decision_record_writeup_review_checklist_item_count"],
+ len(expected_iwooos_host_owner_decision_record_writeup_review_checklist_item_ids),
+ )
iwooos_progress = iwooos_projection["progress"]
assert_equal("iwooos_projection.progress.overall_percent", iwooos_progress["overall_percent"], progress["overall_percent"])
assert_equal(
@@ -1847,6 +1862,91 @@ def validate(root: Path) -> None:
f"iwooos_projection.host_owner_decision_record_writeup_packets.{item['packet_id']}.not_authorization",
item["not_authorization"],
)
+ iwooos_host_owner_decision_record_writeup_review_checklist_items = iwooos_projection[
+ "host_owner_decision_record_writeup_review_checklist_items"
+ ]
+ assert_equal(
+ "iwooos_projection.host_owner_decision_record_writeup_review_checklist_items.ids",
+ [item["check_id"] for item in iwooos_host_owner_decision_record_writeup_review_checklist_items],
+ expected_iwooos_host_owner_decision_record_writeup_review_checklist_item_ids,
+ )
+ assert_equal(
+ "iwooos_projection.host_owner_decision_record_writeup_review_checklist_items.display_order",
+ [item["display_order"] for item in iwooos_host_owner_decision_record_writeup_review_checklist_items],
+ list(range(1, len(expected_iwooos_host_owner_decision_record_writeup_review_checklist_item_ids) + 1)),
+ )
+ expected_iwooos_host_owner_decision_record_writeup_review_conditions = [
+ "decision_summary_risk_acceptance_and_no_execution_statement_readable",
+ "scope_exclusion_observation_intent_and_expiry_complete",
+ "scan_mode_limits_explicit_and_not_authorization",
+ "credential_boundary_metadata_only_and_no_secret_collection",
+ "maintenance_window_constraints_rollback_and_human_contact_linked",
+ "validation_metrics_baseline_evidence_and_acceptance_condition_linked",
+ "runtime_gate_pointer_separate_and_closed",
+ ]
+ assert_equal(
+ "iwooos_projection.host_owner_decision_record_writeup_review_checklist_items.review_conditions",
+ [item["review_condition"] for item in iwooos_host_owner_decision_record_writeup_review_checklist_items],
+ expected_iwooos_host_owner_decision_record_writeup_review_conditions,
+ )
+ for item in iwooos_host_owner_decision_record_writeup_review_checklist_items:
+ assert_equal(
+ f"iwooos_projection.host_owner_decision_record_writeup_review_checklist_items.{item['check_id']}.display_mode",
+ item["display_mode"],
+ "owner_decision_record_writeup_review_checklist_only",
+ )
+ assert_equal(
+ f"iwooos_projection.host_owner_decision_record_writeup_review_checklist_items.{item['check_id']}.decision_record_writeup_review_passed_count",
+ item["decision_record_writeup_review_passed_count"],
+ 0,
+ )
+ assert_equal(
+ f"iwooos_projection.host_owner_decision_record_writeup_review_checklist_items.{item['check_id']}.decision_record_writeup_completed_count",
+ item["decision_record_writeup_completed_count"],
+ 0,
+ )
+ assert_false(
+ f"iwooos_projection.host_owner_decision_record_writeup_review_checklist_items.{item['check_id']}.decision_record_created",
+ item["decision_record_created"],
+ )
+ assert_equal(
+ f"iwooos_projection.host_owner_decision_record_writeup_review_checklist_items.{item['check_id']}.owner_decision_received_count",
+ item["owner_decision_received_count"],
+ 0,
+ )
+ assert_equal(
+ f"iwooos_projection.host_owner_decision_record_writeup_review_checklist_items.{item['check_id']}.owner_decision_accepted_count",
+ item["owner_decision_accepted_count"],
+ 0,
+ )
+ assert_false(
+ f"iwooos_projection.host_owner_decision_record_writeup_review_checklist_items.{item['check_id']}.owner_approval_record_created",
+ item["owner_approval_record_created"],
+ )
+ assert_false(
+ f"iwooos_projection.host_owner_decision_record_writeup_review_checklist_items.{item['check_id']}.runtime_gate_opened",
+ item["runtime_gate_opened"],
+ )
+ assert_false(
+ f"iwooos_projection.host_owner_decision_record_writeup_review_checklist_items.{item['check_id']}.raw_payload_allowed",
+ item["raw_payload_allowed"],
+ )
+ assert_false(
+ f"iwooos_projection.host_owner_decision_record_writeup_review_checklist_items.{item['check_id']}.secret_value_collection_allowed",
+ item["secret_value_collection_allowed"],
+ )
+ assert_false(
+ f"iwooos_projection.host_owner_decision_record_writeup_review_checklist_items.{item['check_id']}.runtime_execution_authorized",
+ item["runtime_execution_authorized"],
+ )
+ assert_false(
+ f"iwooos_projection.host_owner_decision_record_writeup_review_checklist_items.{item['check_id']}.action_buttons_allowed",
+ item["action_buttons_allowed"],
+ )
+ assert_true(
+ f"iwooos_projection.host_owner_decision_record_writeup_review_checklist_items.{item['check_id']}.not_authorization",
+ item["not_authorization"],
+ )
assert_equal(
"iwooos_projection.non_blocking_lane_ids",
iwooos_projection["non_blocking_lane_ids"],
@@ -1885,6 +1985,7 @@ def validate(root: Path) -> None:
"display_host_owner_decision_record_draft_review_checklist",
"display_host_owner_decision_record_draft_review_outcome_lanes",
"display_host_owner_decision_record_writeup_packets",
+ "display_host_owner_decision_record_writeup_review_checklist",
"display_evidence_refs",
"display_forbidden_actions",
]:
@@ -1950,6 +2051,11 @@ def validate(root: Path) -> None:
"mark_host_owner_decision_record_writeup_completed",
"mark_host_owner_decision_record_accepted_from_writeup",
"open_runtime_gate_from_owner_decision_record_writeup",
+ "treat_host_owner_decision_record_writeup_review_as_approval",
+ "mark_host_owner_decision_record_writeup_review_passed",
+ "mark_host_owner_decision_record_writeup_review_completed",
+ "create_host_owner_decision_record_from_writeup_review",
+ "open_runtime_gate_from_owner_decision_record_writeup_review",
"apply_runtime_blocking_control",
"switch_github_primary",
"production_deploy",