@ncukondo/reference-manager 0.25.0 → 0.26.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +11 -2
- package/dist/chunks/{SearchableMultiSelect-B7qEWPDT.js → SearchableMultiSelect-D2IzthN4.js} +2 -2
- package/dist/chunks/{SearchableMultiSelect-B7qEWPDT.js.map → SearchableMultiSelect-D2IzthN4.js.map} +1 -1
- package/dist/chunks/{action-menu-DD0RtNVD.js → action-menu-DWdoHTFh.js} +3 -3
- package/dist/chunks/{action-menu-DD0RtNVD.js.map → action-menu-DWdoHTFh.js.map} +1 -1
- package/dist/chunks/checker-CKfdG8Ia.js +170 -0
- package/dist/chunks/checker-CKfdG8Ia.js.map +1 -0
- package/dist/chunks/crossref-client-Gs75LMVf.js +94 -0
- package/dist/chunks/crossref-client-Gs75LMVf.js.map +1 -0
- package/dist/chunks/{fix-interaction-BpfMLRNY.js → fix-interaction-DNXbmlPr.js} +79 -20
- package/dist/chunks/fix-interaction-DNXbmlPr.js.map +1 -0
- package/dist/chunks/{index-QTYx5RaF.js → index-BEQ4YIXx.js} +115 -44
- package/dist/chunks/index-BEQ4YIXx.js.map +1 -0
- package/dist/chunks/{index-PQkbePWV.js → index-k67fQbe4.js} +3 -3
- package/dist/chunks/index-k67fQbe4.js.map +1 -0
- package/dist/chunks/{index-D2HsxXnK.js → index-of6eJn8N.js} +10 -4
- package/dist/chunks/{index-D2HsxXnK.js.map → index-of6eJn8N.js.map} +1 -1
- package/dist/chunks/{index-CYEise6v.js → index-tdmbNN9b.js} +4 -4
- package/dist/chunks/{index-CYEise6v.js.map → index-tdmbNN9b.js.map} +1 -1
- package/dist/chunks/metadata-comparator-C5zfoYdK.js +137 -0
- package/dist/chunks/metadata-comparator-C5zfoYdK.js.map +1 -0
- package/dist/chunks/{pubmed-client-J18fg3fG.js → pubmed-client-CGReJIOz.js} +2 -2
- package/dist/chunks/{pubmed-client-J18fg3fG.js.map → pubmed-client-CGReJIOz.js.map} +1 -1
- package/dist/chunks/{reference-select-Qpgt9cbN.js → reference-select-i1Cnmc16.js} +3 -3
- package/dist/chunks/{reference-select-Qpgt9cbN.js.map → reference-select-i1Cnmc16.js.map} +1 -1
- package/dist/chunks/{style-select-mEMoWbM2.js → style-select-COnY01qb.js} +3 -3
- package/dist/chunks/{style-select-mEMoWbM2.js.map → style-select-COnY01qb.js.map} +1 -1
- package/dist/cli/commands/check.d.ts +2 -1
- package/dist/cli/commands/check.d.ts.map +1 -1
- package/dist/cli/index.d.ts.map +1 -1
- package/dist/cli.js +1 -1
- package/dist/features/check/checker.d.ts +1 -0
- package/dist/features/check/checker.d.ts.map +1 -1
- package/dist/features/check/crossref-client.d.ts +16 -0
- package/dist/features/check/crossref-client.d.ts.map +1 -1
- package/dist/features/check/fix-actions.d.ts +1 -1
- package/dist/features/check/fix-actions.d.ts.map +1 -1
- package/dist/features/check/fix-interaction.d.ts.map +1 -1
- package/dist/features/check/metadata-comparator.d.ts +37 -0
- package/dist/features/check/metadata-comparator.d.ts.map +1 -0
- package/dist/features/check/metadata-similarity.d.ts +22 -0
- package/dist/features/check/metadata-similarity.d.ts.map +1 -0
- package/dist/features/check/types.d.ts +6 -1
- package/dist/features/check/types.d.ts.map +1 -1
- package/dist/features/operations/check.d.ts +1 -0
- package/dist/features/operations/check.d.ts.map +1 -1
- package/dist/mcp/tools/check.d.ts +1 -0
- package/dist/mcp/tools/check.d.ts.map +1 -1
- package/dist/server/routes/check.d.ts.map +1 -1
- package/dist/server.js +1 -1
- package/package.json +1 -1
- package/dist/chunks/checker-7pzK2XSC.js +0 -92
- package/dist/chunks/checker-7pzK2XSC.js.map +0 -1
- package/dist/chunks/crossref-client-DGNz4PNW.js +0 -52
- package/dist/chunks/crossref-client-DGNz4PNW.js.map +0 -1
- package/dist/chunks/fix-interaction-BpfMLRNY.js.map +0 -1
- package/dist/chunks/index-PQkbePWV.js.map +0 -1
- package/dist/chunks/index-QTYx5RaF.js.map +0 -1
|
@@ -0,0 +1,170 @@
|
|
|
1
|
+
async function checkReference(item, config) {
|
|
2
|
+
const id = item.id;
|
|
3
|
+
const uuid = item.custom?.uuid ?? "";
|
|
4
|
+
const checkedAt = (/* @__PURE__ */ new Date()).toISOString();
|
|
5
|
+
const findings = [];
|
|
6
|
+
const checkedSources = [];
|
|
7
|
+
const hasDoi = !!item.DOI;
|
|
8
|
+
const hasPmid = !!item.PMID;
|
|
9
|
+
if (!hasDoi && !hasPmid) {
|
|
10
|
+
return { id, uuid, status: "skipped", findings: [], checkedAt, checkedSources: [] };
|
|
11
|
+
}
|
|
12
|
+
let crossrefMetadata;
|
|
13
|
+
if (hasDoi) {
|
|
14
|
+
checkedSources.push("crossref");
|
|
15
|
+
const crossrefResult = await checkCrossref(item.DOI, config);
|
|
16
|
+
findings.push(...crossrefResult.findings);
|
|
17
|
+
crossrefMetadata = crossrefResult.metadata;
|
|
18
|
+
}
|
|
19
|
+
if (hasPmid) {
|
|
20
|
+
checkedSources.push("pubmed");
|
|
21
|
+
const pubmedFindings = await checkPubmed(item.PMID, config);
|
|
22
|
+
addUniqueFindings(findings, pubmedFindings);
|
|
23
|
+
}
|
|
24
|
+
const metadataFinding = await checkMetadata(item, config, crossrefMetadata, hasPmid, hasDoi);
|
|
25
|
+
if (metadataFinding) {
|
|
26
|
+
findings.push(metadataFinding);
|
|
27
|
+
}
|
|
28
|
+
const status = findings.length > 0 ? "warning" : "ok";
|
|
29
|
+
return { id, uuid, status, findings, checkedAt, checkedSources };
|
|
30
|
+
}
|
|
31
|
+
function addUniqueFindings(target, source) {
|
|
32
|
+
for (const finding of source) {
|
|
33
|
+
if (!target.some((f) => f.type === finding.type)) {
|
|
34
|
+
target.push(finding);
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
async function checkMetadata(item, config, crossrefMetadata, hasPmid, hasDoi) {
|
|
39
|
+
if (config?.metadata === false) return null;
|
|
40
|
+
if (crossrefMetadata) {
|
|
41
|
+
return compareItemMetadata(item, crossrefMetadata);
|
|
42
|
+
}
|
|
43
|
+
if (hasPmid && !hasDoi) {
|
|
44
|
+
return comparePubmedMetadata(item, config);
|
|
45
|
+
}
|
|
46
|
+
return null;
|
|
47
|
+
}
|
|
48
|
+
async function checkCrossref(doi, config) {
|
|
49
|
+
const { queryCrossref } = await import("./crossref-client-Gs75LMVf.js");
|
|
50
|
+
const crossrefConfig = config?.email ? { email: config.email } : void 0;
|
|
51
|
+
const result = await queryCrossref(doi, crossrefConfig);
|
|
52
|
+
if (!result.success) return { findings: [] };
|
|
53
|
+
const findings = [];
|
|
54
|
+
for (const update of result.updates) {
|
|
55
|
+
const finding = mapCrossrefUpdate(update);
|
|
56
|
+
if (finding) {
|
|
57
|
+
findings.push(finding);
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
return result.metadata ? { findings, metadata: result.metadata } : { findings };
|
|
61
|
+
}
|
|
62
|
+
async function compareItemMetadata(item, remoteMetadata) {
|
|
63
|
+
const { compareMetadata } = await import("./metadata-comparator-C5zfoYdK.js");
|
|
64
|
+
const local = extractLocalMetadata(item);
|
|
65
|
+
const comparison = compareMetadata(local, remoteMetadata);
|
|
66
|
+
if (comparison.classification === "no_change") return null;
|
|
67
|
+
const type = comparison.classification;
|
|
68
|
+
const message = type === "metadata_mismatch" ? "Local metadata significantly differs from the remote record" : "Remote metadata has been updated since import";
|
|
69
|
+
return {
|
|
70
|
+
type,
|
|
71
|
+
message,
|
|
72
|
+
details: {
|
|
73
|
+
updatedFields: comparison.changedFields,
|
|
74
|
+
fieldDiffs: comparison.fieldDiffs
|
|
75
|
+
}
|
|
76
|
+
};
|
|
77
|
+
}
|
|
78
|
+
async function comparePubmedMetadata(item, config) {
|
|
79
|
+
const { fetchPmids } = await import("./index-of6eJn8N.js").then((n) => n.y);
|
|
80
|
+
const pubmedConfig = config?.pubmed ?? {};
|
|
81
|
+
const results = await fetchPmids([item.PMID], pubmedConfig);
|
|
82
|
+
const result = results[0];
|
|
83
|
+
if (!result || !result.success) {
|
|
84
|
+
console.error(
|
|
85
|
+
`PubMed metadata fetch failed for PMID ${item.PMID}: ${result?.error ?? "unknown error"}`
|
|
86
|
+
);
|
|
87
|
+
return null;
|
|
88
|
+
}
|
|
89
|
+
const remoteMetadata = cslItemToRemoteMetadata(result.item);
|
|
90
|
+
return compareItemMetadata(item, remoteMetadata);
|
|
91
|
+
}
|
|
92
|
+
function cslItemToRemoteMetadata(item) {
|
|
93
|
+
const metadata = {};
|
|
94
|
+
if (item.title !== void 0) metadata.title = item.title;
|
|
95
|
+
if (item.author !== void 0) {
|
|
96
|
+
metadata.author = item.author;
|
|
97
|
+
}
|
|
98
|
+
if (item["container-title"] !== void 0) metadata.containerTitle = item["container-title"];
|
|
99
|
+
if (item.type !== void 0) metadata.type = item.type;
|
|
100
|
+
if (item.page !== void 0) metadata.page = item.page;
|
|
101
|
+
if (item.volume !== void 0) metadata.volume = item.volume;
|
|
102
|
+
if (item.issue !== void 0) metadata.issue = item.issue;
|
|
103
|
+
if (item.issued !== void 0) {
|
|
104
|
+
metadata.issued = item.issued;
|
|
105
|
+
}
|
|
106
|
+
return metadata;
|
|
107
|
+
}
|
|
108
|
+
function extractLocalMetadata(item) {
|
|
109
|
+
const local = {};
|
|
110
|
+
if (item.title !== void 0) local.title = item.title;
|
|
111
|
+
if (item.author !== void 0)
|
|
112
|
+
local.author = item.author;
|
|
113
|
+
if (item["container-title"] !== void 0) local["container-title"] = item["container-title"];
|
|
114
|
+
if (item.type !== void 0) local.type = item.type;
|
|
115
|
+
if (item.page !== void 0) local.page = item.page;
|
|
116
|
+
if (item.volume !== void 0) local.volume = item.volume;
|
|
117
|
+
if (item.issue !== void 0) local.issue = item.issue;
|
|
118
|
+
if (item.issued !== void 0) local.issued = item.issued;
|
|
119
|
+
return local;
|
|
120
|
+
}
|
|
121
|
+
async function checkPubmed(pmid, config) {
|
|
122
|
+
const { queryPubmed } = await import("./pubmed-client-CGReJIOz.js");
|
|
123
|
+
const result = await queryPubmed(pmid, config?.pubmed);
|
|
124
|
+
if (!result.success) return [];
|
|
125
|
+
const findings = [];
|
|
126
|
+
if (result.isRetracted) {
|
|
127
|
+
findings.push({
|
|
128
|
+
type: "retracted",
|
|
129
|
+
message: "This article is marked as retracted in PubMed"
|
|
130
|
+
});
|
|
131
|
+
}
|
|
132
|
+
if (result.hasConcern) {
|
|
133
|
+
findings.push({
|
|
134
|
+
type: "concern",
|
|
135
|
+
message: "Expression of concern noted in PubMed"
|
|
136
|
+
});
|
|
137
|
+
}
|
|
138
|
+
return findings;
|
|
139
|
+
}
|
|
140
|
+
function mapCrossrefUpdate(update) {
|
|
141
|
+
const doiDetail = update.doi ? { retractionDoi: update.doi } : {};
|
|
142
|
+
const dateDetail = update.date ? { retractionDate: update.date } : {};
|
|
143
|
+
const newDoiDetail = update.doi ? { newDoi: update.doi } : {};
|
|
144
|
+
switch (update.type) {
|
|
145
|
+
case "retraction":
|
|
146
|
+
return {
|
|
147
|
+
type: "retracted",
|
|
148
|
+
message: update.date ? `This article was retracted on ${update.date}` : "This article was retracted",
|
|
149
|
+
details: { ...doiDetail, ...dateDetail }
|
|
150
|
+
};
|
|
151
|
+
case "expression-of-concern":
|
|
152
|
+
return {
|
|
153
|
+
type: "concern",
|
|
154
|
+
message: update.date ? `Expression of concern issued on ${update.date}` : "Expression of concern issued",
|
|
155
|
+
details: { ...doiDetail, ...dateDetail }
|
|
156
|
+
};
|
|
157
|
+
case "new_version":
|
|
158
|
+
return {
|
|
159
|
+
type: "version_changed",
|
|
160
|
+
message: update.doi ? `Published version available: ${update.doi}` : "Published version available",
|
|
161
|
+
details: newDoiDetail
|
|
162
|
+
};
|
|
163
|
+
default:
|
|
164
|
+
return null;
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
export {
|
|
168
|
+
checkReference
|
|
169
|
+
};
|
|
170
|
+
//# sourceMappingURL=checker-CKfdG8Ia.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"checker-CKfdG8Ia.js","sources":["../../src/features/check/checker.ts"],"sourcesContent":["import type { CslItem } from \"../../core/csl-json/types.js\";\nimport type { CrossrefUpdateInfo, RemoteMetadata } from \"./crossref-client.js\";\nimport type { CheckFinding, CheckResult } from \"./types.js\";\n\nexport interface CheckConfig {\n email?: string;\n pubmed?: { email?: string; apiKey?: string };\n metadata?: boolean;\n}\n\ninterface CrossrefCheckResult {\n findings: CheckFinding[];\n metadata?: RemoteMetadata;\n}\n\n/**\n * Check a single reference against external sources for status changes.\n *\n * @param item - The CSL-JSON item to check\n * @param config - Optional config for API credentials\n * @returns Check result with findings\n */\nexport async function checkReference(item: CslItem, config?: CheckConfig): Promise<CheckResult> {\n const id = item.id;\n const uuid = (item.custom?.uuid as string) ?? \"\";\n const checkedAt = new Date().toISOString();\n const findings: CheckFinding[] = [];\n const checkedSources: string[] = [];\n\n const hasDoi = !!item.DOI;\n const hasPmid = !!item.PMID;\n\n // Skip references with neither DOI nor PMID\n if (!hasDoi && !hasPmid) {\n return { id, uuid, status: \"skipped\", findings: [], checkedAt, checkedSources: [] };\n }\n\n let crossrefMetadata: RemoteMetadata | undefined;\n\n // Query Crossref if DOI is present\n if (hasDoi) {\n checkedSources.push(\"crossref\");\n const crossrefResult = await checkCrossref(item.DOI as string, config);\n findings.push(...crossrefResult.findings);\n crossrefMetadata = crossrefResult.metadata;\n }\n\n // Query PubMed if PMID is present\n if (hasPmid) {\n checkedSources.push(\"pubmed\");\n const pubmedFindings = await checkPubmed(item.PMID as string, config);\n addUniqueFindings(findings, pubmedFindings);\n }\n\n // Metadata comparison (default: enabled)\n const metadataFinding = await checkMetadata(item, config, crossrefMetadata, hasPmid, hasDoi);\n if (metadataFinding) {\n findings.push(metadataFinding);\n }\n\n const status = findings.length > 0 ? \"warning\" : \"ok\";\n return { id, uuid, status, findings, checkedAt, checkedSources };\n}\n\n/**\n * Add findings that aren't already present (by type) to the target list.\n */\nfunction addUniqueFindings(target: CheckFinding[], source: CheckFinding[]): void {\n for (const finding of source) {\n if (!target.some((f) => f.type === finding.type)) {\n target.push(finding);\n }\n }\n}\n\n/**\n * Perform metadata comparison if enabled.\n */\nasync function checkMetadata(\n item: CslItem,\n config: CheckConfig | undefined,\n crossrefMetadata: RemoteMetadata | undefined,\n hasPmid: boolean,\n hasDoi: boolean\n): Promise<CheckFinding | null> {\n if (config?.metadata === false) return null;\n\n if (crossrefMetadata) {\n // DOI-based: compare against Crossref metadata\n return compareItemMetadata(item, crossrefMetadata);\n }\n\n if (hasPmid && !hasDoi) {\n // PubMed-only: fetch remote CSL-JSON via PubMed and compare\n return comparePubmedMetadata(item, config);\n }\n\n return null;\n}\n\n/**\n * Query Crossref and return findings plus metadata.\n */\nasync function checkCrossref(doi: string, config?: CheckConfig): Promise<CrossrefCheckResult> {\n const { queryCrossref } = await import(\"./crossref-client.js\");\n const crossrefConfig = config?.email ? { email: config.email } : undefined;\n const result = await queryCrossref(doi, crossrefConfig);\n if (!result.success) return { findings: [] };\n\n const findings: CheckFinding[] = [];\n for (const update of result.updates) {\n const finding = mapCrossrefUpdate(update);\n if (finding) {\n findings.push(finding);\n }\n }\n return result.metadata ? { findings, metadata: result.metadata } : { findings };\n}\n\n/**\n * Compare item metadata against Crossref metadata.\n */\nasync function compareItemMetadata(\n item: CslItem,\n remoteMetadata: RemoteMetadata\n): Promise<CheckFinding | null> {\n const { compareMetadata } = await import(\"./metadata-comparator.js\");\n\n const local = extractLocalMetadata(item);\n const comparison = compareMetadata(local, remoteMetadata);\n\n if (comparison.classification === \"no_change\") return null;\n\n const type = comparison.classification;\n const message =\n type === \"metadata_mismatch\"\n ? \"Local metadata significantly differs from the remote record\"\n : \"Remote metadata has been updated since import\";\n\n return {\n type,\n message,\n details: {\n updatedFields: comparison.changedFields,\n fieldDiffs: comparison.fieldDiffs,\n },\n };\n}\n\n/**\n * Fetch PubMed CSL-JSON and compare metadata for PMID-only references.\n */\nasync function comparePubmedMetadata(\n item: CslItem,\n config?: CheckConfig\n): Promise<CheckFinding | null> {\n const { fetchPmids } = await import(\"../import/fetcher.js\");\n const pubmedConfig = config?.pubmed ?? {};\n const results = await fetchPmids([item.PMID as string], pubmedConfig);\n const result = results[0];\n if (!result || !result.success) {\n console.error(\n `PubMed metadata fetch failed for PMID ${item.PMID}: ${result?.error ?? \"unknown error\"}`\n );\n return null;\n }\n\n const remoteMetadata = cslItemToRemoteMetadata(result.item);\n return compareItemMetadata(item, remoteMetadata);\n}\n\n/**\n * Convert a CslItem (from PubMed) to RemoteMetadata format for comparison.\n */\nfunction cslItemToRemoteMetadata(item: CslItem): RemoteMetadata {\n const metadata: RemoteMetadata = {};\n if (item.title !== undefined) metadata.title = item.title;\n if (item.author !== undefined) {\n metadata.author = item.author as Array<{ family?: string; given?: string }>;\n }\n if (item[\"container-title\"] !== undefined) metadata.containerTitle = item[\"container-title\"];\n if (item.type !== undefined) metadata.type = item.type;\n if (item.page !== undefined) metadata.page = item.page;\n if (item.volume !== undefined) metadata.volume = item.volume;\n if (item.issue !== undefined) metadata.issue = item.issue;\n if (item.issued !== undefined) {\n metadata.issued = item.issued as { \"date-parts\"?: number[][] };\n }\n return metadata;\n}\n\n/**\n * Extract local metadata fields from a CslItem for comparison.\n */\nfunction extractLocalMetadata(\n item: CslItem\n): import(\"./metadata-comparator.js\").LocalMetadataFields {\n const local: import(\"./metadata-comparator.js\").LocalMetadataFields = {};\n if (item.title !== undefined) local.title = item.title;\n if (item.author !== undefined)\n local.author = item.author as Array<{ family?: string; given?: string }>;\n if (item[\"container-title\"] !== undefined) local[\"container-title\"] = item[\"container-title\"];\n if (item.type !== undefined) local.type = item.type;\n if (item.page !== undefined) local.page = item.page;\n if (item.volume !== undefined) local.volume = item.volume;\n if (item.issue !== undefined) local.issue = item.issue;\n if (item.issued !== undefined) local.issued = item.issued as { \"date-parts\"?: number[][] };\n return local;\n}\n\n/**\n * Query PubMed and return findings.\n */\nasync function checkPubmed(pmid: string, config?: CheckConfig): Promise<CheckFinding[]> {\n const { queryPubmed } = await import(\"./pubmed-client.js\");\n const result = await queryPubmed(pmid, config?.pubmed);\n if (!result.success) return [];\n\n const findings: CheckFinding[] = [];\n if (result.isRetracted) {\n findings.push({\n type: \"retracted\",\n message: \"This article is marked as retracted in PubMed\",\n });\n }\n if (result.hasConcern) {\n findings.push({\n type: \"concern\",\n message: \"Expression of concern noted in PubMed\",\n });\n }\n return findings;\n}\n\n/**\n * Map a Crossref update-to entry to a CheckFinding.\n */\nfunction mapCrossrefUpdate(update: CrossrefUpdateInfo): CheckFinding | null {\n const doiDetail = update.doi ? { retractionDoi: update.doi } : {};\n const dateDetail = update.date ? { retractionDate: update.date } : {};\n const newDoiDetail = update.doi ? { newDoi: update.doi } : {};\n\n switch (update.type) {\n case \"retraction\":\n return {\n type: \"retracted\",\n message: update.date\n ? `This article was retracted on ${update.date}`\n : \"This article was retracted\",\n details: { ...doiDetail, ...dateDetail },\n };\n case \"expression-of-concern\":\n return {\n type: \"concern\",\n message: update.date\n ? `Expression of concern issued on ${update.date}`\n : \"Expression of concern issued\",\n details: { ...doiDetail, ...dateDetail },\n };\n case \"new_version\":\n return {\n type: \"version_changed\",\n message: update.doi\n ? `Published version available: ${update.doi}`\n : \"Published version available\",\n details: newDoiDetail,\n };\n default:\n return null;\n }\n}\n"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,94 @@
|
|
|
1
|
+
import { g as getRateLimiter } from "./index-of6eJn8N.js";
|
|
2
|
+
const CROSSREF_API_BASE = "https://api.crossref.org/works";
|
|
3
|
+
function formatDateParts(updated) {
|
|
4
|
+
if (!updated || typeof updated !== "object") return {};
|
|
5
|
+
const dateParts = updated["date-parts"];
|
|
6
|
+
if (!Array.isArray(dateParts) || dateParts.length === 0) return {};
|
|
7
|
+
const parts = dateParts[0];
|
|
8
|
+
if (!Array.isArray(parts) || parts.length === 0) return {};
|
|
9
|
+
const [year, month, day] = parts;
|
|
10
|
+
const m = String(month ?? 1).padStart(2, "0");
|
|
11
|
+
const d = String(day ?? 1).padStart(2, "0");
|
|
12
|
+
return { date: `${year}-${m}-${d}` };
|
|
13
|
+
}
|
|
14
|
+
function extractMetadata(message) {
|
|
15
|
+
if (!message) return void 0;
|
|
16
|
+
const metadata = {};
|
|
17
|
+
const titleArr = message.title;
|
|
18
|
+
const firstTitle = Array.isArray(titleArr) ? titleArr[0] : void 0;
|
|
19
|
+
if (firstTitle) {
|
|
20
|
+
metadata.title = firstTitle;
|
|
21
|
+
}
|
|
22
|
+
const authorArr = message.author;
|
|
23
|
+
if (Array.isArray(authorArr) && authorArr.length > 0) {
|
|
24
|
+
metadata.author = authorArr.map((a) => ({
|
|
25
|
+
...a.family ? { family: a.family } : {},
|
|
26
|
+
...a.given ? { given: a.given } : {}
|
|
27
|
+
}));
|
|
28
|
+
}
|
|
29
|
+
const containerArr = message["container-title"];
|
|
30
|
+
const firstContainer = Array.isArray(containerArr) ? containerArr[0] : void 0;
|
|
31
|
+
if (firstContainer) {
|
|
32
|
+
metadata.containerTitle = firstContainer;
|
|
33
|
+
}
|
|
34
|
+
if (typeof message.type === "string") {
|
|
35
|
+
metadata.type = message.type;
|
|
36
|
+
}
|
|
37
|
+
if (typeof message.page === "string") metadata.page = message.page;
|
|
38
|
+
if (typeof message.volume === "string") metadata.volume = message.volume;
|
|
39
|
+
if (typeof message.issue === "string") metadata.issue = message.issue;
|
|
40
|
+
const issued = message.issued;
|
|
41
|
+
if (issued && Array.isArray(issued["date-parts"])) {
|
|
42
|
+
metadata.issued = issued;
|
|
43
|
+
}
|
|
44
|
+
if (Object.keys(metadata).length === 0) return void 0;
|
|
45
|
+
return metadata;
|
|
46
|
+
}
|
|
47
|
+
async function queryCrossref(doi, config) {
|
|
48
|
+
const rateLimiter = getRateLimiter("crossref", {});
|
|
49
|
+
await rateLimiter.acquire();
|
|
50
|
+
try {
|
|
51
|
+
const url = new URL(`${CROSSREF_API_BASE}/${encodeURIComponent(doi)}`);
|
|
52
|
+
if (config?.email) {
|
|
53
|
+
url.searchParams.set("mailto", config.email);
|
|
54
|
+
}
|
|
55
|
+
const response = await fetch(url.toString());
|
|
56
|
+
if (!response.ok) {
|
|
57
|
+
return {
|
|
58
|
+
success: false,
|
|
59
|
+
error: `Crossref API returned ${response.status} ${response.statusText}`
|
|
60
|
+
};
|
|
61
|
+
}
|
|
62
|
+
const data = await response.json();
|
|
63
|
+
const message = data.message;
|
|
64
|
+
const updateTo = message?.["update-to"] ?? [];
|
|
65
|
+
const updatedBy = message?.["updated-by"] ?? [];
|
|
66
|
+
const allEntries = [...updateTo, ...updatedBy].map((e) => {
|
|
67
|
+
const datePart = formatDateParts(e.updated);
|
|
68
|
+
return {
|
|
69
|
+
type: String(e.type ?? ""),
|
|
70
|
+
...e.DOI ? { doi: String(e.DOI) } : {},
|
|
71
|
+
...e.label ? { label: String(e.label) } : {},
|
|
72
|
+
...datePart.date ? { date: datePart.date } : {}
|
|
73
|
+
};
|
|
74
|
+
});
|
|
75
|
+
const seen = /* @__PURE__ */ new Set();
|
|
76
|
+
const updates = allEntries.filter((e) => {
|
|
77
|
+
const key = `${e.type}:${e.doi ?? ""}`;
|
|
78
|
+
if (seen.has(key)) return false;
|
|
79
|
+
seen.add(key);
|
|
80
|
+
return true;
|
|
81
|
+
});
|
|
82
|
+
const metadata = extractMetadata(message);
|
|
83
|
+
return metadata ? { success: true, updates, metadata } : { success: true, updates };
|
|
84
|
+
} catch (error) {
|
|
85
|
+
return {
|
|
86
|
+
success: false,
|
|
87
|
+
error: error instanceof Error ? error.message : String(error)
|
|
88
|
+
};
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
export {
|
|
92
|
+
queryCrossref
|
|
93
|
+
};
|
|
94
|
+
//# sourceMappingURL=crossref-client-Gs75LMVf.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"crossref-client-Gs75LMVf.js","sources":["../../src/features/check/crossref-client.ts"],"sourcesContent":["import { getRateLimiter } from \"../import/rate-limiter.js\";\n\nconst CROSSREF_API_BASE = \"https://api.crossref.org/works\";\n\nexport interface CrossrefUpdateInfo {\n type: string;\n doi?: string;\n label?: string;\n date?: string;\n}\n\nexport interface RemoteMetadata {\n title?: string;\n author?: Array<{ family?: string; given?: string }>;\n containerTitle?: string;\n type?: string;\n page?: string;\n volume?: string;\n issue?: string;\n issued?: { \"date-parts\"?: number[][] };\n}\n\nexport type CrossrefResult =\n | { success: true; updates: CrossrefUpdateInfo[]; metadata?: RemoteMetadata }\n | { success: false; error: string };\n\n/**\n * Format date-parts from Crossref API response to ISO date string.\n */\nfunction formatDateParts(updated: unknown): { date?: string } {\n if (!updated || typeof updated !== \"object\") return {};\n const dateParts = (updated as Record<string, unknown>)[\"date-parts\"];\n if (!Array.isArray(dateParts) || dateParts.length === 0) return {};\n const parts = dateParts[0] as number[];\n if (!Array.isArray(parts) || parts.length === 0) return {};\n const [year, month, day] = parts;\n const m = String(month ?? 1).padStart(2, \"0\");\n const d = String(day ?? 1).padStart(2, \"0\");\n return { date: `${year}-${m}-${d}` };\n}\n\n/**\n * Extract comparable metadata fields from a Crossref message object.\n */\nfunction extractMetadata(message: Record<string, unknown> | undefined): RemoteMetadata | undefined {\n if (!message) return undefined;\n\n const metadata: RemoteMetadata = {};\n\n // Title: array in Crossref, take first element\n const titleArr = message.title as string[] | undefined;\n const firstTitle = Array.isArray(titleArr) ? titleArr[0] : undefined;\n if (firstTitle) {\n metadata.title = firstTitle;\n }\n\n // Author\n const authorArr = message.author as Array<{ family?: string; given?: string }> | undefined;\n if (Array.isArray(authorArr) && authorArr.length > 0) {\n metadata.author = authorArr.map((a) => ({\n ...(a.family ? { family: a.family } : {}),\n ...(a.given ? { given: a.given } : {}),\n }));\n }\n\n // Container title: array in Crossref, take first\n const containerArr = message[\"container-title\"] as string[] | undefined;\n const firstContainer = Array.isArray(containerArr) ? containerArr[0] : undefined;\n if (firstContainer) {\n metadata.containerTitle = firstContainer;\n }\n\n // Type\n if (typeof message.type === \"string\") {\n metadata.type = message.type;\n }\n\n // Page, volume, issue\n if (typeof message.page === \"string\") metadata.page = message.page;\n if (typeof message.volume === \"string\") metadata.volume = message.volume;\n if (typeof message.issue === \"string\") metadata.issue = message.issue;\n\n // Issued\n const issued = message.issued as { \"date-parts\"?: number[][] } | undefined;\n if (issued && Array.isArray(issued[\"date-parts\"])) {\n metadata.issued = issued;\n }\n\n // Return undefined if no metadata fields were populated (prevents spurious diffs)\n if (Object.keys(metadata).length === 0) return undefined;\n\n return metadata;\n}\n\n/**\n * Query Crossref REST API for a DOI and extract update-to information.\n *\n * @param doi - The DOI to query\n * @param config - Optional config with email for polite pool\n * @returns Crossref result with update information\n */\nexport async function queryCrossref(\n doi: string,\n config?: { email?: string }\n): Promise<CrossrefResult> {\n const rateLimiter = getRateLimiter(\"crossref\", {});\n await rateLimiter.acquire();\n\n try {\n const url = new URL(`${CROSSREF_API_BASE}/${encodeURIComponent(doi)}`);\n if (config?.email) {\n url.searchParams.set(\"mailto\", config.email);\n }\n const response = await fetch(url.toString());\n\n if (!response.ok) {\n return {\n success: false,\n error: `Crossref API returned ${response.status} ${response.statusText}`,\n };\n }\n\n const data = (await response.json()) as Record<string, unknown>;\n const message = data.message as Record<string, unknown> | undefined;\n\n const updateTo = (message?.[\"update-to\"] ?? []) as Record<string, unknown>[];\n const updatedBy = (message?.[\"updated-by\"] ?? []) as Record<string, unknown>[];\n const allEntries: CrossrefUpdateInfo[] = [...updateTo, ...updatedBy].map((e) => {\n const datePart = formatDateParts(e.updated);\n return {\n type: String(e.type ?? \"\"),\n ...(e.DOI ? { doi: String(e.DOI) } : {}),\n ...(e.label ? { label: String(e.label) } : {}),\n ...(datePart.date ? { date: datePart.date } : {}),\n };\n });\n\n const seen = new Set<string>();\n const updates = allEntries.filter((e) => {\n const key = `${e.type}:${e.doi ?? \"\"}`;\n if (seen.has(key)) return false;\n seen.add(key);\n return true;\n });\n\n const metadata = extractMetadata(message);\n\n return metadata ? { success: true, updates, metadata } : { success: true, updates };\n } catch (error) {\n return {\n success: false,\n error: error instanceof Error ? error.message : String(error),\n };\n }\n}\n"],"names":[],"mappings":"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"}
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import { render } from "ink";
|
|
2
2
|
import { createElement } from "react";
|
|
3
|
-
import { S as Select, r as restoreStdinAfterInk } from "./index-
|
|
3
|
+
import { S as Select, r as restoreStdinAfterInk } from "./index-BEQ4YIXx.js";
|
|
4
4
|
function getFixActionsForFinding(finding) {
|
|
5
5
|
switch (finding.type) {
|
|
6
6
|
case "retracted":
|
|
@@ -22,6 +22,12 @@ function getFixActionsForFinding(finding) {
|
|
|
22
22
|
{ type: "add_concern_note", label: "Add note with concern details" },
|
|
23
23
|
{ type: "skip", label: "Skip" }
|
|
24
24
|
];
|
|
25
|
+
case "metadata_mismatch":
|
|
26
|
+
case "metadata_outdated":
|
|
27
|
+
return [
|
|
28
|
+
{ type: "update_all_fields", label: "Update all changed fields from remote" },
|
|
29
|
+
{ type: "skip", label: "Skip" }
|
|
30
|
+
];
|
|
25
31
|
default:
|
|
26
32
|
return [];
|
|
27
33
|
}
|
|
@@ -66,6 +72,64 @@ async function applyNoteAction(library, item, prefix, finding) {
|
|
|
66
72
|
await library.save();
|
|
67
73
|
return { applied: true, message: `Added note: ${noteText}` };
|
|
68
74
|
}
|
|
75
|
+
async function applyUpdateFromPublished(library, item, finding) {
|
|
76
|
+
const newDoi = finding.details?.newDoi;
|
|
77
|
+
if (!newDoi) {
|
|
78
|
+
return { applied: false, message: "No published DOI available in finding details" };
|
|
79
|
+
}
|
|
80
|
+
const { fetchDoi } = await import("./index-of6eJn8N.js").then((n) => n.y);
|
|
81
|
+
const fetchResult = await fetchDoi(newDoi);
|
|
82
|
+
if (!fetchResult.success) {
|
|
83
|
+
return {
|
|
84
|
+
applied: false,
|
|
85
|
+
message: `Failed to fetch metadata for ${newDoi}: ${fetchResult.error}`
|
|
86
|
+
};
|
|
87
|
+
}
|
|
88
|
+
const { id: _id, custom: _custom, ...metadata } = fetchResult.item;
|
|
89
|
+
await library.update(item.id, metadata, { idType: "id" });
|
|
90
|
+
await library.save();
|
|
91
|
+
return { applied: true, message: `Updated metadata from ${newDoi}` };
|
|
92
|
+
}
|
|
93
|
+
async function applyUpdateAllFields(library, item, finding) {
|
|
94
|
+
let fetchedItem;
|
|
95
|
+
if (item.DOI) {
|
|
96
|
+
const { fetchDoi } = await import("./index-of6eJn8N.js").then((n) => n.y);
|
|
97
|
+
const fetchResult = await fetchDoi(item.DOI);
|
|
98
|
+
if (!fetchResult.success) {
|
|
99
|
+
return {
|
|
100
|
+
applied: false,
|
|
101
|
+
message: `Failed to fetch metadata for DOI ${item.DOI}: ${fetchResult.error}`
|
|
102
|
+
};
|
|
103
|
+
}
|
|
104
|
+
fetchedItem = fetchResult.item;
|
|
105
|
+
} else if (item.PMID) {
|
|
106
|
+
const { fetchPmids } = await import("./index-of6eJn8N.js").then((n) => n.y);
|
|
107
|
+
const results = await fetchPmids([item.PMID], {});
|
|
108
|
+
const result = results[0];
|
|
109
|
+
if (!result || !result.success) {
|
|
110
|
+
return {
|
|
111
|
+
applied: false,
|
|
112
|
+
message: `Failed to fetch metadata for PMID ${item.PMID}: ${result?.error ?? "unknown error"}`
|
|
113
|
+
};
|
|
114
|
+
}
|
|
115
|
+
fetchedItem = result.item;
|
|
116
|
+
} else {
|
|
117
|
+
return { applied: false, message: "No DOI or PMID available for metadata update" };
|
|
118
|
+
}
|
|
119
|
+
const updatedFields = finding.details?.updatedFields ?? [];
|
|
120
|
+
const updates = {};
|
|
121
|
+
for (const field of updatedFields) {
|
|
122
|
+
if (field in fetchedItem) {
|
|
123
|
+
updates[field] = fetchedItem[field];
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
if (Object.keys(updates).length === 0) {
|
|
127
|
+
return { applied: false, message: "No fields to update" };
|
|
128
|
+
}
|
|
129
|
+
await library.update(item.id, updates, { idType: "id" });
|
|
130
|
+
await library.save();
|
|
131
|
+
return { applied: true, message: `Updated fields: ${updatedFields.join(", ")}` };
|
|
132
|
+
}
|
|
69
133
|
async function applyFixAction(library, item, finding, actionType) {
|
|
70
134
|
switch (actionType) {
|
|
71
135
|
case "add_retracted_tag":
|
|
@@ -86,24 +150,15 @@ async function applyFixAction(library, item, finding, actionType) {
|
|
|
86
150
|
await library.save();
|
|
87
151
|
return { applied: true, message: `Removed ${item.id}`, removed: true };
|
|
88
152
|
}
|
|
89
|
-
case "update_from_published":
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
applied: false,
|
|
99
|
-
message: `Failed to fetch metadata for ${newDoi}: ${fetchResult.error}`
|
|
100
|
-
};
|
|
101
|
-
}
|
|
102
|
-
const { id: _id, custom: _custom, ...metadata } = fetchResult.item;
|
|
103
|
-
await library.update(item.id, metadata, { idType: "id" });
|
|
104
|
-
await library.save();
|
|
105
|
-
return { applied: true, message: `Updated metadata from ${newDoi}` };
|
|
106
|
-
}
|
|
153
|
+
case "update_from_published":
|
|
154
|
+
return applyUpdateFromPublished(library, item, finding);
|
|
155
|
+
case "update_all_fields":
|
|
156
|
+
return applyUpdateAllFields(library, item, finding);
|
|
157
|
+
// Placeholder for programmatic API (MCP/server) use.
|
|
158
|
+
// Individual field selection is not available in interactive CLI mode,
|
|
159
|
+
// but can be used via the MCP tool or HTTP server API.
|
|
160
|
+
case "update_selected_fields":
|
|
161
|
+
return { applied: false, message: "Field selection not available in CLI mode" };
|
|
107
162
|
case "skip":
|
|
108
163
|
return { applied: true, message: "Skipped" };
|
|
109
164
|
default:
|
|
@@ -142,6 +197,10 @@ function getStatusLabel(type) {
|
|
|
142
197
|
return "CONCERN";
|
|
143
198
|
case "version_changed":
|
|
144
199
|
return "VERSION";
|
|
200
|
+
case "metadata_mismatch":
|
|
201
|
+
return "MISMATCH";
|
|
202
|
+
case "metadata_outdated":
|
|
203
|
+
return "OUTDATED";
|
|
145
204
|
default:
|
|
146
205
|
return "WARNING";
|
|
147
206
|
}
|
|
@@ -200,4 +259,4 @@ async function runFixInteraction(results, library, findItem) {
|
|
|
200
259
|
export {
|
|
201
260
|
runFixInteraction
|
|
202
261
|
};
|
|
203
|
-
//# sourceMappingURL=fix-interaction-
|
|
262
|
+
//# sourceMappingURL=fix-interaction-DNXbmlPr.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"fix-interaction-DNXbmlPr.js","sources":["../../src/features/check/fix-actions.ts","../../src/features/check/fix-interaction.ts"],"sourcesContent":["import type { CslItem } from \"../../core/csl-json/types.js\";\nimport type { ILibrary } from \"../../core/library-interface.js\";\nimport type { CheckFinding } from \"./types.js\";\n\nexport type FixActionType =\n | \"add_retracted_tag\"\n | \"add_retraction_note\"\n | \"remove_from_library\"\n | \"update_from_published\"\n | \"add_version_tag\"\n | \"add_concern_tag\"\n | \"add_concern_note\"\n | \"update_all_fields\"\n | \"update_selected_fields\"\n | \"skip\";\n\nexport interface FixAction {\n type: FixActionType;\n label: string;\n}\n\nexport interface FixActionResult {\n applied: boolean;\n message: string;\n removed?: boolean;\n}\n\nexport function getFixActionsForFinding(finding: CheckFinding): FixAction[] {\n switch (finding.type) {\n case \"retracted\":\n return [\n { type: \"add_retracted_tag\", label: 'Add tag \"retracted\"' },\n { type: \"add_retraction_note\", label: \"Add note with retraction details\" },\n { type: \"remove_from_library\", label: \"Remove from library\" },\n { type: \"skip\", label: \"Skip\" },\n ];\n case \"version_changed\":\n return [\n { type: \"update_from_published\", label: \"Update metadata from published version\" },\n { type: \"add_version_tag\", label: 'Add tag \"has-published-version\"' },\n { type: \"skip\", label: \"Skip\" },\n ];\n case \"concern\":\n return [\n { type: \"add_concern_tag\", label: 'Add tag \"expression-of-concern\"' },\n { type: \"add_concern_note\", label: \"Add note with concern details\" },\n { type: \"skip\", label: \"Skip\" },\n ];\n case \"metadata_mismatch\":\n case \"metadata_outdated\":\n return [\n { type: \"update_all_fields\", label: \"Update all changed fields from remote\" },\n { type: \"skip\", label: \"Skip\" },\n ];\n default:\n return [];\n }\n}\n\nfunction addTag(item: CslItem, tag: string): string[] {\n const existing = (item.custom?.tags as string[] | undefined) ?? [];\n if (existing.includes(tag)) {\n return existing;\n }\n return [...existing, tag];\n}\n\nfunction buildNoteText(prefix: string, finding: CheckFinding): string {\n const parts = [prefix];\n if (finding.details?.retractionDate) {\n parts.push(`Date: ${finding.details.retractionDate}`);\n }\n if (finding.details?.retractionDoi) {\n parts.push(`DOI: ${finding.details.retractionDoi}`);\n }\n return parts.join(\". \");\n}\n\nfunction appendNote(existingNote: string | undefined, newNote: string): string {\n if (existingNote) {\n return `${existingNote}\\n\\n${newNote}`;\n }\n return newNote;\n}\n\nasync function applyTagAction(\n library: ILibrary,\n item: CslItem,\n tag: string\n): Promise<FixActionResult> {\n const tags = addTag(item, tag);\n await library.update(item.id, { custom: { ...item.custom, tags } } as Partial<CslItem>, {\n idType: \"id\",\n });\n await library.save();\n return { applied: true, message: `Added tag \"${tag}\"` };\n}\n\nasync function applyNoteAction(\n library: ILibrary,\n item: CslItem,\n prefix: string,\n finding: CheckFinding\n): Promise<FixActionResult> {\n const noteText = buildNoteText(prefix, finding);\n const note = appendNote(item.note, noteText);\n await library.update(item.id, { note } as Partial<CslItem>, { idType: \"id\" });\n await library.save();\n return { applied: true, message: `Added note: ${noteText}` };\n}\n\nasync function applyUpdateFromPublished(\n library: ILibrary,\n item: CslItem,\n finding: CheckFinding\n): Promise<FixActionResult> {\n const newDoi = finding.details?.newDoi;\n if (!newDoi) {\n return { applied: false, message: \"No published DOI available in finding details\" };\n }\n const { fetchDoi } = await import(\"../import/fetcher.js\");\n const fetchResult = await fetchDoi(newDoi);\n if (!fetchResult.success) {\n return {\n applied: false,\n message: `Failed to fetch metadata for ${newDoi}: ${fetchResult.error}`,\n };\n }\n const { id: _id, custom: _custom, ...metadata } = fetchResult.item;\n await library.update(item.id, metadata as Partial<CslItem>, { idType: \"id\" });\n await library.save();\n return { applied: true, message: `Updated metadata from ${newDoi}` };\n}\n\nasync function applyUpdateAllFields(\n library: ILibrary,\n item: CslItem,\n finding: CheckFinding\n): Promise<FixActionResult> {\n let fetchedItem: CslItem;\n\n if (item.DOI) {\n const { fetchDoi } = await import(\"../import/fetcher.js\");\n const fetchResult = await fetchDoi(item.DOI);\n if (!fetchResult.success) {\n return {\n applied: false,\n message: `Failed to fetch metadata for DOI ${item.DOI}: ${fetchResult.error}`,\n };\n }\n fetchedItem = fetchResult.item;\n } else if (item.PMID) {\n const { fetchPmids } = await import(\"../import/fetcher.js\");\n const results = await fetchPmids([item.PMID], {});\n const result = results[0];\n if (!result || !result.success) {\n return {\n applied: false,\n message: `Failed to fetch metadata for PMID ${item.PMID}: ${result?.error ?? \"unknown error\"}`,\n };\n }\n fetchedItem = result.item;\n } else {\n return { applied: false, message: \"No DOI or PMID available for metadata update\" };\n }\n\n const updatedFields = finding.details?.updatedFields ?? [];\n const updates: Partial<CslItem> = {};\n for (const field of updatedFields) {\n if (field in fetchedItem) {\n (updates as Record<string, unknown>)[field] = (fetchedItem as Record<string, unknown>)[field];\n }\n }\n if (Object.keys(updates).length === 0) {\n return { applied: false, message: \"No fields to update\" };\n }\n await library.update(item.id, updates, { idType: \"id\" });\n await library.save();\n return { applied: true, message: `Updated fields: ${updatedFields.join(\", \")}` };\n}\n\nexport async function applyFixAction(\n library: ILibrary,\n item: CslItem,\n finding: CheckFinding,\n actionType: FixActionType\n): Promise<FixActionResult> {\n switch (actionType) {\n case \"add_retracted_tag\":\n return applyTagAction(library, item, \"retracted\");\n\n case \"add_retraction_note\":\n return applyNoteAction(library, item, \"RETRACTED\", finding);\n\n case \"add_concern_tag\":\n return applyTagAction(library, item, \"expression-of-concern\");\n\n case \"add_concern_note\":\n return applyNoteAction(library, item, \"EXPRESSION OF CONCERN\", finding);\n\n case \"add_version_tag\":\n return applyTagAction(library, item, \"has-published-version\");\n\n case \"remove_from_library\": {\n const removeResult = await library.remove(item.id, { idType: \"id\" });\n if (!removeResult.removed) {\n return { applied: false, message: `Failed to remove ${item.id}` };\n }\n await library.save();\n return { applied: true, message: `Removed ${item.id}`, removed: true };\n }\n\n case \"update_from_published\":\n return applyUpdateFromPublished(library, item, finding);\n\n case \"update_all_fields\":\n return applyUpdateAllFields(library, item, finding);\n\n // Placeholder for programmatic API (MCP/server) use.\n // Individual field selection is not available in interactive CLI mode,\n // but can be used via the MCP tool or HTTP server API.\n case \"update_selected_fields\":\n return { applied: false, message: \"Field selection not available in CLI mode\" };\n\n case \"skip\":\n return { applied: true, message: \"Skipped\" };\n\n default:\n return { applied: false, message: `Unknown action: ${actionType}` };\n }\n}\n","import { render } from \"ink\";\nimport { createElement } from \"react\";\nimport type { CslItem } from \"../../core/csl-json/types.js\";\nimport type { ILibrary } from \"../../core/library-interface.js\";\nimport { restoreStdinAfterInk } from \"../interactive/alternate-screen.js\";\nimport { Select } from \"../interactive/components/index.js\";\nimport type { SelectOption } from \"../interactive/components/index.js\";\nimport { type FixActionType, applyFixAction, getFixActionsForFinding } from \"./fix-actions.js\";\nimport type { CheckFinding, CheckResult } from \"./types.js\";\n\nexport interface FixInteractionResult {\n totalFindings: number;\n applied: number;\n skipped: number;\n removed: string[];\n}\n\nfunction selectFixAction(\n message: string,\n options: SelectOption<FixActionType>[]\n): Promise<FixActionType | null> {\n return new Promise<FixActionType | null>((resolve) => {\n let selected: FixActionType | null = null;\n\n const { waitUntilExit } = render(\n createElement(Select<FixActionType>, {\n options,\n message,\n onSelect: (value: FixActionType) => {\n selected = value;\n },\n onCancel: () => {\n selected = null;\n },\n })\n );\n\n waitUntilExit()\n .then(() => {\n restoreStdinAfterInk();\n resolve(selected);\n })\n .catch(() => {\n restoreStdinAfterInk();\n resolve(null);\n });\n });\n}\n\nfunction getStatusLabel(type: string): string {\n switch (type) {\n case \"retracted\":\n return \"RETRACTED\";\n case \"concern\":\n return \"CONCERN\";\n case \"version_changed\":\n return \"VERSION\";\n case \"metadata_mismatch\":\n return \"MISMATCH\";\n case \"metadata_outdated\":\n return \"OUTDATED\";\n default:\n return \"WARNING\";\n }\n}\n\nfunction buildSelectOptions(finding: CheckFinding): SelectOption<FixActionType>[] {\n return getFixActionsForFinding(finding).map((a) => ({\n label: a.label,\n value: a.type,\n }));\n}\n\nasync function processFinding(\n result: FixInteractionResult,\n library: ILibrary,\n item: CslItem,\n resultId: string,\n finding: CheckFinding\n): Promise<void> {\n result.totalFindings++;\n const options = buildSelectOptions(finding);\n if (options.length === 0) return;\n\n const label = getStatusLabel(finding.type);\n const message = `[${label}] ${resultId}: ${finding.message}`;\n const selectedAction = await selectFixAction(message, options);\n\n if (selectedAction === null) {\n result.skipped++;\n return;\n }\n\n const actionResult = await applyFixAction(library, item, finding, selectedAction);\n\n if (!actionResult.applied) {\n process.stderr.write(` Error: ${actionResult.message}\\n`);\n return;\n }\n\n process.stderr.write(` ${actionResult.message}\\n`);\n if (selectedAction === \"skip\") {\n result.skipped++;\n } else {\n result.applied++;\n if (actionResult.removed) {\n result.removed.push(resultId);\n }\n }\n}\n\nexport async function runFixInteraction(\n results: CheckResult[],\n library: ILibrary,\n findItem: (id: string) => CslItem | undefined\n): Promise<FixInteractionResult> {\n const interactionResult: FixInteractionResult = {\n totalFindings: 0,\n applied: 0,\n skipped: 0,\n removed: [],\n };\n\n const warningResults = results.filter((r) => r.status === \"warning\");\n\n for (const checkResult of warningResults) {\n const item = findItem(checkResult.id);\n if (!item) continue;\n\n for (const finding of checkResult.findings) {\n await processFinding(interactionResult, library, item, checkResult.id, finding);\n }\n }\n\n return interactionResult;\n}\n"],"names":[],"mappings":";;;AA2BO,SAAS,wBAAwB,SAAoC;AAC1E,UAAQ,QAAQ,MAAA;AAAA,IACd,KAAK;AACH,aAAO;AAAA,QACL,EAAE,MAAM,qBAAqB,OAAO,sBAAA;AAAA,QACpC,EAAE,MAAM,uBAAuB,OAAO,mCAAA;AAAA,QACtC,EAAE,MAAM,uBAAuB,OAAO,sBAAA;AAAA,QACtC,EAAE,MAAM,QAAQ,OAAO,OAAA;AAAA,MAAO;AAAA,IAElC,KAAK;AACH,aAAO;AAAA,QACL,EAAE,MAAM,yBAAyB,OAAO,yCAAA;AAAA,QACxC,EAAE,MAAM,mBAAmB,OAAO,kCAAA;AAAA,QAClC,EAAE,MAAM,QAAQ,OAAO,OAAA;AAAA,MAAO;AAAA,IAElC,KAAK;AACH,aAAO;AAAA,QACL,EAAE,MAAM,mBAAmB,OAAO,kCAAA;AAAA,QAClC,EAAE,MAAM,oBAAoB,OAAO,gCAAA;AAAA,QACnC,EAAE,MAAM,QAAQ,OAAO,OAAA;AAAA,MAAO;AAAA,IAElC,KAAK;AAAA,IACL,KAAK;AACH,aAAO;AAAA,QACL,EAAE,MAAM,qBAAqB,OAAO,wCAAA;AAAA,QACpC,EAAE,MAAM,QAAQ,OAAO,OAAA;AAAA,MAAO;AAAA,IAElC;AACE,aAAO,CAAA;AAAA,EAAC;AAEd;AAEA,SAAS,OAAO,MAAe,KAAuB;AACpD,QAAM,WAAY,KAAK,QAAQ,QAAiC,CAAA;AAChE,MAAI,SAAS,SAAS,GAAG,GAAG;AAC1B,WAAO;AAAA,EACT;AACA,SAAO,CAAC,GAAG,UAAU,GAAG;AAC1B;AAEA,SAAS,cAAc,QAAgB,SAA+B;AACpE,QAAM,QAAQ,CAAC,MAAM;AACrB,MAAI,QAAQ,SAAS,gBAAgB;AACnC,UAAM,KAAK,SAAS,QAAQ,QAAQ,cAAc,EAAE;AAAA,EACtD;AACA,MAAI,QAAQ,SAAS,eAAe;AAClC,UAAM,KAAK,QAAQ,QAAQ,QAAQ,aAAa,EAAE;AAAA,EACpD;AACA,SAAO,MAAM,KAAK,IAAI;AACxB;AAEA,SAAS,WAAW,cAAkC,SAAyB;AAC7E,MAAI,cAAc;AAChB,WAAO,GAAG,YAAY;AAAA;AAAA,EAAO,OAAO;AAAA,EACtC;AACA,SAAO;AACT;AAEA,eAAe,eACb,SACA,MACA,KAC0B;AAC1B,QAAM,OAAO,OAAO,MAAM,GAAG;AAC7B,QAAM,QAAQ,OAAO,KAAK,IAAI,EAAE,QAAQ,EAAE,GAAG,KAAK,QAAQ,KAAA,EAAK,GAAyB;AAAA,IACtF,QAAQ;AAAA,EAAA,CACT;AACD,QAAM,QAAQ,KAAA;AACd,SAAO,EAAE,SAAS,MAAM,SAAS,cAAc,GAAG,IAAA;AACpD;AAEA,eAAe,gBACb,SACA,MACA,QACA,SAC0B;AAC1B,QAAM,WAAW,cAAc,QAAQ,OAAO;AAC9C,QAAM,OAAO,WAAW,KAAK,MAAM,QAAQ;AAC3C,QAAM,QAAQ,OAAO,KAAK,IAAI,EAAE,QAA4B,EAAE,QAAQ,MAAM;AAC5E,QAAM,QAAQ,KAAA;AACd,SAAO,EAAE,SAAS,MAAM,SAAS,eAAe,QAAQ,GAAA;AAC1D;AAEA,eAAe,yBACb,SACA,MACA,SAC0B;AAC1B,QAAM,SAAS,QAAQ,SAAS;AAChC,MAAI,CAAC,QAAQ;AACX,WAAO,EAAE,SAAS,OAAO,SAAS,gDAAA;AAAA,EACpC;AACA,QAAM,EAAE,SAAA,IAAa,MAAM,OAAO,qBAAsB,EAAA,KAAA,OAAA,EAAA,CAAA;AACxD,QAAM,cAAc,MAAM,SAAS,MAAM;AACzC,MAAI,CAAC,YAAY,SAAS;AACxB,WAAO;AAAA,MACL,SAAS;AAAA,MACT,SAAS,gCAAgC,MAAM,KAAK,YAAY,KAAK;AAAA,IAAA;AAAA,EAEzE;AACA,QAAM,EAAE,IAAI,KAAK,QAAQ,SAAS,GAAG,SAAA,IAAa,YAAY;AAC9D,QAAM,QAAQ,OAAO,KAAK,IAAI,UAA8B,EAAE,QAAQ,MAAM;AAC5E,QAAM,QAAQ,KAAA;AACd,SAAO,EAAE,SAAS,MAAM,SAAS,yBAAyB,MAAM,GAAA;AAClE;AAEA,eAAe,qBACb,SACA,MACA,SAC0B;AAC1B,MAAI;AAEJ,MAAI,KAAK,KAAK;AACZ,UAAM,EAAE,SAAA,IAAa,MAAM,OAAO,qBAAsB,EAAA,KAAA,OAAA,EAAA,CAAA;AACxD,UAAM,cAAc,MAAM,SAAS,KAAK,GAAG;AAC3C,QAAI,CAAC,YAAY,SAAS;AACxB,aAAO;AAAA,QACL,SAAS;AAAA,QACT,SAAS,oCAAoC,KAAK,GAAG,KAAK,YAAY,KAAK;AAAA,MAAA;AAAA,IAE/E;AACA,kBAAc,YAAY;AAAA,EAC5B,WAAW,KAAK,MAAM;AACpB,UAAM,EAAE,WAAA,IAAe,MAAM,OAAO,qBAAsB,EAAA,KAAA,OAAA,EAAA,CAAA;AAC1D,UAAM,UAAU,MAAM,WAAW,CAAC,KAAK,IAAI,GAAG,EAAE;AAChD,UAAM,SAAS,QAAQ,CAAC;AACxB,QAAI,CAAC,UAAU,CAAC,OAAO,SAAS;AAC9B,aAAO;AAAA,QACL,SAAS;AAAA,QACT,SAAS,qCAAqC,KAAK,IAAI,KAAK,QAAQ,SAAS,eAAe;AAAA,MAAA;AAAA,IAEhG;AACA,kBAAc,OAAO;AAAA,EACvB,OAAO;AACL,WAAO,EAAE,SAAS,OAAO,SAAS,+CAAA;AAAA,EACpC;AAEA,QAAM,gBAAgB,QAAQ,SAAS,iBAAiB,CAAA;AACxD,QAAM,UAA4B,CAAA;AAClC,aAAW,SAAS,eAAe;AACjC,QAAI,SAAS,aAAa;AACvB,cAAoC,KAAK,IAAK,YAAwC,KAAK;AAAA,IAC9F;AAAA,EACF;AACA,MAAI,OAAO,KAAK,OAAO,EAAE,WAAW,GAAG;AACrC,WAAO,EAAE,SAAS,OAAO,SAAS,sBAAA;AAAA,EACpC;AACA,QAAM,QAAQ,OAAO,KAAK,IAAI,SAAS,EAAE,QAAQ,MAAM;AACvD,QAAM,QAAQ,KAAA;AACd,SAAO,EAAE,SAAS,MAAM,SAAS,mBAAmB,cAAc,KAAK,IAAI,CAAC,GAAA;AAC9E;AAEA,eAAsB,eACpB,SACA,MACA,SACA,YAC0B;AAC1B,UAAQ,YAAA;AAAA,IACN,KAAK;AACH,aAAO,eAAe,SAAS,MAAM,WAAW;AAAA,IAElD,KAAK;AACH,aAAO,gBAAgB,SAAS,MAAM,aAAa,OAAO;AAAA,IAE5D,KAAK;AACH,aAAO,eAAe,SAAS,MAAM,uBAAuB;AAAA,IAE9D,KAAK;AACH,aAAO,gBAAgB,SAAS,MAAM,yBAAyB,OAAO;AAAA,IAExE,KAAK;AACH,aAAO,eAAe,SAAS,MAAM,uBAAuB;AAAA,IAE9D,KAAK,uBAAuB;AAC1B,YAAM,eAAe,MAAM,QAAQ,OAAO,KAAK,IAAI,EAAE,QAAQ,MAAM;AACnE,UAAI,CAAC,aAAa,SAAS;AACzB,eAAO,EAAE,SAAS,OAAO,SAAS,oBAAoB,KAAK,EAAE,GAAA;AAAA,MAC/D;AACA,YAAM,QAAQ,KAAA;AACd,aAAO,EAAE,SAAS,MAAM,SAAS,WAAW,KAAK,EAAE,IAAI,SAAS,KAAA;AAAA,IAClE;AAAA,IAEA,KAAK;AACH,aAAO,yBAAyB,SAAS,MAAM,OAAO;AAAA,IAExD,KAAK;AACH,aAAO,qBAAqB,SAAS,MAAM,OAAO;AAAA;AAAA;AAAA;AAAA,IAKpD,KAAK;AACH,aAAO,EAAE,SAAS,OAAO,SAAS,4CAAA;AAAA,IAEpC,KAAK;AACH,aAAO,EAAE,SAAS,MAAM,SAAS,UAAA;AAAA,IAEnC;AACE,aAAO,EAAE,SAAS,OAAO,SAAS,mBAAmB,UAAU,GAAA;AAAA,EAAG;AAExE;ACrNA,SAAS,gBACP,SACA,SAC+B;AAC/B,SAAO,IAAI,QAA8B,CAAC,YAAY;AACpD,QAAI,WAAiC;AAErC,UAAM,EAAE,kBAAkB;AAAA,MACxB,cAAc,QAAuB;AAAA,QACnC;AAAA,QACA;AAAA,QACA,UAAU,CAAC,UAAyB;AAClC,qBAAW;AAAA,QACb;AAAA,QACA,UAAU,MAAM;AACd,qBAAW;AAAA,QACb;AAAA,MAAA,CACD;AAAA,IAAA;AAGH,kBAAA,EACG,KAAK,MAAM;AACV,2BAAA;AACA,cAAQ,QAAQ;AAAA,IAClB,CAAC,EACA,MAAM,MAAM;AACX,2BAAA;AACA,cAAQ,IAAI;AAAA,IACd,CAAC;AAAA,EACL,CAAC;AACH;AAEA,SAAS,eAAe,MAAsB;AAC5C,UAAQ,MAAA;AAAA,IACN,KAAK;AACH,aAAO;AAAA,IACT,KAAK;AACH,aAAO;AAAA,IACT,KAAK;AACH,aAAO;AAAA,IACT,KAAK;AACH,aAAO;AAAA,IACT,KAAK;AACH,aAAO;AAAA,IACT;AACE,aAAO;AAAA,EAAA;AAEb;AAEA,SAAS,mBAAmB,SAAsD;AAChF,SAAO,wBAAwB,OAAO,EAAE,IAAI,CAAC,OAAO;AAAA,IAClD,OAAO,EAAE;AAAA,IACT,OAAO,EAAE;AAAA,EAAA,EACT;AACJ;AAEA,eAAe,eACb,QACA,SACA,MACA,UACA,SACe;AACf,SAAO;AACP,QAAM,UAAU,mBAAmB,OAAO;AAC1C,MAAI,QAAQ,WAAW,EAAG;AAE1B,QAAM,QAAQ,eAAe,QAAQ,IAAI;AACzC,QAAM,UAAU,IAAI,KAAK,KAAK,QAAQ,KAAK,QAAQ,OAAO;AAC1D,QAAM,iBAAiB,MAAM,gBAAgB,SAAS,OAAO;AAE7D,MAAI,mBAAmB,MAAM;AAC3B,WAAO;AACP;AAAA,EACF;AAEA,QAAM,eAAe,MAAM,eAAe,SAAS,MAAM,SAAS,cAAc;AAEhF,MAAI,CAAC,aAAa,SAAS;AACzB,YAAQ,OAAO,MAAM,YAAY,aAAa,OAAO;AAAA,CAAI;AACzD;AAAA,EACF;AAEA,UAAQ,OAAO,MAAM,KAAK,aAAa,OAAO;AAAA,CAAI;AAClD,MAAI,mBAAmB,QAAQ;AAC7B,WAAO;AAAA,EACT,OAAO;AACL,WAAO;AACP,QAAI,aAAa,SAAS;AACxB,aAAO,QAAQ,KAAK,QAAQ;AAAA,IAC9B;AAAA,EACF;AACF;AAEA,eAAsB,kBACpB,SACA,SACA,UAC+B;AAC/B,QAAM,oBAA0C;AAAA,IAC9C,eAAe;AAAA,IACf,SAAS;AAAA,IACT,SAAS;AAAA,IACT,SAAS,CAAA;AAAA,EAAC;AAGZ,QAAM,iBAAiB,QAAQ,OAAO,CAAC,MAAM,EAAE,WAAW,SAAS;AAEnE,aAAW,eAAe,gBAAgB;AACxC,UAAM,OAAO,SAAS,YAAY,EAAE;AACpC,QAAI,CAAC,KAAM;AAEX,eAAW,WAAW,YAAY,UAAU;AAC1C,YAAM,eAAe,mBAAmB,SAAS,MAAM,YAAY,IAAI,OAAO;AAAA,IAChF;AAAA,EACF;AAEA,SAAO;AACT;"}
|