341 lines
13 KiB
TypeScript
341 lines
13 KiB
TypeScript
import { readFileSync } from "node:fs";
|
|
|
|
type JsonRecord = Record<string, unknown>;
|
|
type DocTypeCode = "DCSN" | "GOAL" | "PLAN" | "RPRT" | "ACTN" | "ISSU" | "RETR" | "RQST" | "RESP" | "MINS";
|
|
|
|
interface DocMetadata {
|
|
docNo: string;
|
|
docType: DocTypeCode | "";
|
|
priority: string;
|
|
year: string;
|
|
sequence: string;
|
|
}
|
|
|
|
const docTypeLabels: Record<DocTypeCode, string> = {
|
|
DCSN: "决策/决议",
|
|
GOAL: "目标",
|
|
PLAN: "计划",
|
|
RPRT: "报告",
|
|
ACTN: "行动",
|
|
ISSU: "问题",
|
|
RETR: "复盘",
|
|
RQST: "请示",
|
|
RESP: "批复/答复",
|
|
MINS: "会议纪要",
|
|
};
|
|
|
|
const docTypeOrder: DocTypeCode[] = ["DCSN", "GOAL", "PLAN", "RPRT", "ACTN", "ISSU", "RETR", "RQST", "RESP", "MINS"];
|
|
const docTypeSet = new Set<string>(docTypeOrder);
|
|
const docNoPattern = /\bDC[-−–—]([A-Z]{2,5})[-−–—]([A-Z][0-9])[-−–—](\d{4})[-−–—](\d{1,6})\b/iu;
|
|
|
|
function assertCondition(condition: unknown, message: string, detail: JsonRecord = {}): void {
|
|
if (!condition) throw new Error(`${message}: ${JSON.stringify(detail)}`);
|
|
}
|
|
|
|
function source(path: string): string {
|
|
return readFileSync(path, "utf8");
|
|
}
|
|
|
|
function includesAll(text: string, snippets: string[]): boolean {
|
|
return snippets.every((snippet) => text.includes(snippet));
|
|
}
|
|
|
|
function assertEqual<T>(actual: T, expected: T, message: string, detail: JsonRecord = {}): void {
|
|
assertCondition(Object.is(actual, expected), message, { ...detail, actual, expected });
|
|
}
|
|
|
|
function stringField(record: JsonRecord, keys: string[]): string {
|
|
for (const key of keys) {
|
|
const value = record[key];
|
|
if (typeof value === "string" && value.trim()) return value.trim();
|
|
if (typeof value === "number" && Number.isFinite(value)) return String(value);
|
|
}
|
|
return "";
|
|
}
|
|
|
|
function recordTags(record: JsonRecord): string[] {
|
|
const tags = record.tags;
|
|
return Array.isArray(tags) ? tags.map((tag) => String(tag)) : [];
|
|
}
|
|
|
|
function tagValues(record: JsonRecord, prefix: string): string[] {
|
|
const normalized = `${prefix.toLowerCase()}:`;
|
|
return recordTags(record)
|
|
.map((tag) => tag.trim())
|
|
.filter((tag) => tag.toLowerCase().startsWith(normalized))
|
|
.map((tag) => tag.slice(normalized.length).trim())
|
|
.filter(Boolean);
|
|
}
|
|
|
|
function normalizeDocType(value: unknown): DocTypeCode | "" {
|
|
const upper = String(value || "").trim().toUpperCase();
|
|
return docTypeSet.has(upper) ? upper as DocTypeCode : "";
|
|
}
|
|
|
|
function normalizeSequence(value: unknown): string {
|
|
const raw = String(value || "").trim();
|
|
if (!/^\d+$/u.test(raw)) return "";
|
|
return String(Number(raw)).padStart(3, "0");
|
|
}
|
|
|
|
function parseDocNo(value: unknown): DocMetadata | null {
|
|
const match = String(value || "").match(docNoPattern);
|
|
if (match === null) return null;
|
|
const docType = normalizeDocType(match[1]);
|
|
const priority = String(match[2] || "").toUpperCase();
|
|
const year = String(match[3] || "");
|
|
const sequence = normalizeSequence(match[4]);
|
|
if (!docType || !/^P[0-3]$/u.test(priority) || !year || !sequence) return null;
|
|
return {
|
|
docNo: `DC-${docType}-${priority}-${year}-${sequence}`,
|
|
docType,
|
|
priority,
|
|
year,
|
|
sequence,
|
|
};
|
|
}
|
|
|
|
function mergeParsed(metadata: DocMetadata, parsed: DocMetadata | null): void {
|
|
if (parsed === null) return;
|
|
if (!metadata.docNo) metadata.docNo = parsed.docNo;
|
|
if (!metadata.docType) metadata.docType = parsed.docType;
|
|
if (!metadata.priority) metadata.priority = parsed.priority;
|
|
if (!metadata.year) metadata.year = parsed.year;
|
|
if (!metadata.sequence) metadata.sequence = parsed.sequence;
|
|
}
|
|
|
|
function completeDocNo(metadata: DocMetadata): string {
|
|
return metadata.docType && metadata.priority && metadata.year && metadata.sequence
|
|
? `DC-${metadata.docType}-${metadata.priority}-${metadata.year}-${metadata.sequence}`
|
|
: "";
|
|
}
|
|
|
|
function bodyFirstWindow(record: JsonRecord): string {
|
|
const body = stringField(record, ["body", "summary", "markdown"]);
|
|
return body.replace(/\r\n?/gu, "\n").split(/\n\s*\n/gu).map((part) => part.trim()).find(Boolean) || "";
|
|
}
|
|
|
|
function extractDocMetadata(record: JsonRecord): DocMetadata {
|
|
const metadata: DocMetadata = { docNo: "", docType: "", priority: "", year: "", sequence: "" };
|
|
mergeParsed(metadata, parseDocNo(stringField(record, ["docNo", "documentNo", "documentNumber", "documentId"])));
|
|
metadata.docType ||= normalizeDocType(stringField(record, ["docType", "documentType", "documentKind"]));
|
|
metadata.priority ||= stringField(record, ["docPriority", "priority", "level"]).toUpperCase();
|
|
metadata.year ||= stringField(record, ["docYear", "year"]);
|
|
metadata.sequence ||= normalizeSequence(stringField(record, ["docSeq", "sequence", "seq", "docSequence", "documentSequence"]));
|
|
metadata.docNo ||= completeDocNo(metadata);
|
|
mergeParsed(metadata, parseDocNo(stringField(record, ["title"])));
|
|
for (const value of tagValues(record, "doc-no")) mergeParsed(metadata, parseDocNo(value));
|
|
metadata.docType ||= normalizeDocType(tagValues(record, "doc-type")[0]);
|
|
metadata.priority ||= String(tagValues(record, "doc-priority")[0] || "").toUpperCase();
|
|
metadata.year ||= tagValues(record, "doc-year")[0] || "";
|
|
metadata.sequence ||= normalizeSequence(tagValues(record, "doc-sequence")[0]);
|
|
metadata.docNo ||= completeDocNo(metadata);
|
|
mergeParsed(metadata, parseDocNo(bodyFirstWindow(record)));
|
|
metadata.docNo ||= completeDocNo(metadata);
|
|
return metadata;
|
|
}
|
|
|
|
function extractDocNo(record: JsonRecord): string {
|
|
return extractDocMetadata(record).docNo;
|
|
}
|
|
|
|
function groupCount(records: JsonRecord[], type: string): number {
|
|
const groups = new Map<DocTypeCode, JsonRecord[]>();
|
|
for (const code of docTypeOrder) groups.set(code, []);
|
|
for (const record of records) {
|
|
const code = extractDocMetadata(record).docType || "DCSN";
|
|
groups.get(code)?.push(record);
|
|
}
|
|
return groups.get(type as DocTypeCode)?.length || 0;
|
|
}
|
|
|
|
export function runDecisionCenterWorkspaceContract(): JsonRecord {
|
|
const frontend = source("src/components/frontend/src/decision-center.tsx");
|
|
const css = source("src/components/frontend/public/style.css");
|
|
|
|
const titleDcsn = {
|
|
id: "title-dcsn",
|
|
title: "DC-DCSN-P0-2026-001 决策记录",
|
|
tags: [],
|
|
};
|
|
const tagGoal = {
|
|
id: "tag-goal",
|
|
title: "目标文书",
|
|
tags: ["doc-no:DC-GOAL-P0-2026-002"],
|
|
};
|
|
const bodyRprt = {
|
|
id: "body-rprt",
|
|
title: "报告正文",
|
|
body: "DC-RPRT-P2-2026-003\n\n报告正文第一段。",
|
|
tags: [],
|
|
};
|
|
const tagRetr = {
|
|
id: "tag-retr",
|
|
title: "复盘文书",
|
|
tags: ["doc-type:RETR"],
|
|
};
|
|
const structuredPlan = {
|
|
id: "structured-plan",
|
|
title: "结构化计划文书",
|
|
docType: "PLAN",
|
|
priority: "P1",
|
|
year: 2026,
|
|
sequence: 4,
|
|
};
|
|
|
|
assertEqual(extractDocNo(titleDcsn), "DC-DCSN-P0-2026-001", "title prefix must extract complete DCSN doc number");
|
|
assertEqual(extractDocMetadata(titleDcsn).docType, "DCSN", "title prefix must extract DCSN doc type");
|
|
assertEqual(extractDocNo(tagGoal), "DC-GOAL-P0-2026-002", "doc-no tag must extract complete GOAL doc number");
|
|
assertEqual(extractDocMetadata(tagGoal).docType, "GOAL", "doc-no tag must extract GOAL doc type");
|
|
assertEqual(extractDocNo(bodyRprt), "DC-RPRT-P2-2026-003", "body first paragraph must fallback extract complete RPRT doc number");
|
|
assertEqual(extractDocMetadata(bodyRprt).priority, "P2", "body fallback must extract RPRT priority");
|
|
assertEqual(extractDocMetadata(tagRetr).docType, "RETR", "doc-type tag must extract RETR doc type");
|
|
assertEqual(docTypeLabels.RETR, "复盘", "RETR label must be 复盘");
|
|
assertEqual(extractDocNo(structuredPlan), "DC-PLAN-P1-2026-004", "structured fields must compose complete PLAN doc number");
|
|
assertEqual(groupCount([titleDcsn, tagGoal, bodyRprt, tagRetr, structuredPlan], "DCSN"), 1, "DCSN records must be grouped by parsed type");
|
|
assertEqual(groupCount([titleDcsn, tagGoal, bodyRprt, tagRetr, structuredPlan], "GOAL"), 1, "GOAL records must be grouped by parsed type");
|
|
assertEqual(groupCount([titleDcsn, tagGoal, bodyRprt, tagRetr, structuredPlan], "RPRT"), 1, "RPRT records must be grouped by parsed type");
|
|
assertEqual(groupCount([titleDcsn, tagGoal, bodyRprt, tagRetr, structuredPlan], "RETR"), 1, "RETR records must be grouped by parsed type");
|
|
assertEqual(groupCount([titleDcsn, tagGoal, bodyRprt, tagRetr, structuredPlan], "PLAN"), 1, "PLAN records must be grouped by structured type");
|
|
|
|
assertCondition(
|
|
includesAll(frontend, [
|
|
"export function extractDocMetadata(record: any): DocMetadata",
|
|
"export function extractDocNo(record: any): string",
|
|
"export function buildDocTypeTree(records: any[]): Array<{ type: DocTypeCode",
|
|
"function buildTagGroups(records: any[]): Array<{ tag: string",
|
|
"DOC_TYPE_CODES = [\"DCSN\", \"GOAL\", \"PLAN\", \"RPRT\", \"ACTN\", \"ISSU\", \"RETR\", \"RQST\", \"RESP\", \"MINS\"]",
|
|
"docTypeLabels: Record<DocTypeCode, string>",
|
|
]),
|
|
"frontend must implement exported doc metadata extraction, doc-type tree, and tag grouping",
|
|
);
|
|
|
|
assertCondition(
|
|
includesAll(frontend, [
|
|
"function DocTreeNode({ record, depth, onSelect, selectedId }: AnyRecord)",
|
|
"function DocTreeGroup({ group, onSelect, selectedId, defaultOpen }: AnyRecord)",
|
|
"function TagTreeGroup({ group, onSelect, selectedId, defaultOpen }: AnyRecord)",
|
|
"function DocTypeTree({ records, onSelect, selectedId, activeGrouping }: AnyRecord)",
|
|
]),
|
|
"frontend must implement DocTreeNode, DocTreeGroup, TagTreeGroup, and DocTypeTree components",
|
|
);
|
|
|
|
assertCondition(
|
|
includesAll(frontend, [
|
|
"function DocDetailSidebar({ record, onSelect, onSave, saving, error, saveMsg, onRaw }: AnyRecord)",
|
|
"function DocEditor({ form, saving, onChange, onSubmit }: AnyRecord)",
|
|
"function DocViewer({ record }: AnyRecord)",
|
|
"function DocDetailHeader({ record, onBack }: AnyRecord)",
|
|
]),
|
|
"frontend must implement DocDetailSidebar, DocEditor, DocViewer, and DocDetailHeader",
|
|
);
|
|
|
|
assertCondition(
|
|
includesAll(frontend, [
|
|
"function DocWorkspace({ records, selectedRecord, onSelect, onSave, saving, saveError, saveMsg, onRaw }: AnyRecord)",
|
|
"h(DocWorkspace, {",
|
|
"activeView === \"workspace\"",
|
|
"decision-tab-workspace",
|
|
]),
|
|
"frontend must implement DocWorkspace component and workspace tab",
|
|
);
|
|
|
|
assertCondition(
|
|
includesAll(frontend, [
|
|
"mode === \"view\"",
|
|
"mode === \"edit\"",
|
|
"setMode(\"view\")",
|
|
"setMode(\"edit\")",
|
|
"doc-sidebar-mode-tabs",
|
|
]),
|
|
"frontend must implement read/edit mode toggle in DocDetailSidebar",
|
|
);
|
|
|
|
assertCondition(
|
|
includesAll(frontend, [
|
|
"selectedRecord?.id",
|
|
"setState((prev: any) => ({ ...prev, selectedDoc: record",
|
|
"state.selectedDoc",
|
|
]),
|
|
"frontend must track selectedDoc in state and sync on selection",
|
|
);
|
|
|
|
assertCondition(
|
|
includesAll(frontend, [
|
|
"await onSave(form)",
|
|
"recordPayloadFromForm(form)",
|
|
"setRecordSaveState({ saving: false, error: \"\", message:",
|
|
"setState((prev: any) => ({ ...prev, selectedDoc: saved",
|
|
]),
|
|
"frontend must call onSave, payload conversion, error handling, and sync selectedDoc after save",
|
|
);
|
|
|
|
assertCondition(
|
|
css.includes(".doc-workspace"),
|
|
"CSS must include .doc-workspace layout",
|
|
);
|
|
|
|
assertCondition(
|
|
css.includes(".doc-sidebar"),
|
|
"CSS must include .doc-sidebar",
|
|
);
|
|
|
|
assertCondition(
|
|
css.includes("grid-template-columns: minmax(200px, 260px) minmax(0, 1fr) minmax(0, 50%)"),
|
|
"CSS must implement three-column layout with right sidebar at 50%",
|
|
);
|
|
|
|
assertCondition(
|
|
includesAll(css, [
|
|
".doc-type-tree",
|
|
".doc-tree-group",
|
|
".doc-tree-node",
|
|
".doc-list-item",
|
|
".doc-full-markdown",
|
|
".doc-editor-form",
|
|
]),
|
|
"CSS must include all workspace component styles",
|
|
);
|
|
|
|
assertCondition(
|
|
includesAll(frontend, [
|
|
"extractDocMetadata(record)",
|
|
"recordTagValues(record, \"doc-no\")",
|
|
"firstBodyWindow(record)",
|
|
]),
|
|
"extractDocNo must use metadata extraction from title, doc-no tag, and body fallback",
|
|
);
|
|
|
|
assertCondition(
|
|
includesAll(frontend, [
|
|
"grouping === \"type\"",
|
|
"grouping === \"tag\"",
|
|
"setGrouping(\"type\")",
|
|
"setGrouping(\"tag\")",
|
|
]),
|
|
"workspace must support type and tag grouping toggle",
|
|
);
|
|
|
|
return {
|
|
ok: true,
|
|
checks: [
|
|
"doc-no-extraction-functions",
|
|
"doc-type-tree-components",
|
|
"doc-detail-sidebar-components",
|
|
"doc-workspace-component",
|
|
"read-edit-mode-toggle",
|
|
"selected-doc-state-sync",
|
|
"save-and-sync",
|
|
"workspace-css-layout",
|
|
"workspace-css-components",
|
|
"three-column-layout-50-percent",
|
|
"doc-metadata-fixtures",
|
|
"type-tag-grouping-toggle",
|
|
],
|
|
};
|
|
}
|
|
|
|
if (import.meta.main) {
|
|
process.stdout.write(`${JSON.stringify(runDecisionCenterWorkspaceContract(), null, 2)}\n`);
|
|
}
|