@ncukondo/search-hub 0.9.3 → 0.9.4
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.
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"jats-parser.d.ts","sourceRoot":"","sources":["../../../src/fulltext/convert/jats-parser.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAGH,OAAO,KAAK,EAEV,YAAY,EACZ,WAAW,EACX,aAAa,EACb,YAAY,EACZ,cAAc,EACd,YAAY,EAEb,MAAM,YAAY,CAAC;AAuMpB;;GAEG;AACH,wBAAgB,iBAAiB,CAAC,GAAG,EAAE,MAAM,GAAG,YAAY,CA2M3D;AAuLD;;;GAGG;AACH,wBAAgB,cAAc,CAAC,GAAG,EAAE,MAAM,GAAG;IAC3C,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,OAAO,EAAE,MAAM,EAAE,CAAC;IAClB,IAAI,EAAE,MAAM,EAAE,EAAE,CAAC;CAClB,CAQA;AA4PD;;GAEG;AACH,wBAAgB,aAAa,CAAC,GAAG,EAAE,MAAM,GAAG,WAAW,EAAE,CAwBxD;AA8JD;;GAEG;AACH,wBAAgB,mBAAmB,CAAC,GAAG,EAAE,MAAM,GAAG,aAAa,EAAE,CAkDhE;AAID,sDAAsD;AACtD,MAAM,WAAW,gBAAgB;IAC/B,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,UAAU,CAAC,EAAE,WAAW,EAAE,CAAC;IAC3B,SAAS,CAAC,EAAE,YAAY,EAAE,CAAC;IAC3B,MAAM,CAAC,EAAE,YAAY,EAAE,CAAC;IACxB,KAAK,CAAC,EAAE,cAAc,EAAE,CAAC;CAC1B;AAED;;;GAGG;AACH,wBAAgB,mBAAmB,CAAC,GAAG,EAAE,MAAM,GAAG,gBAAgB,
|
|
1
|
+
{"version":3,"file":"jats-parser.d.ts","sourceRoot":"","sources":["../../../src/fulltext/convert/jats-parser.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAGH,OAAO,KAAK,EAEV,YAAY,EACZ,WAAW,EACX,aAAa,EACb,YAAY,EACZ,cAAc,EACd,YAAY,EAEb,MAAM,YAAY,CAAC;AAuMpB;;GAEG;AACH,wBAAgB,iBAAiB,CAAC,GAAG,EAAE,MAAM,GAAG,YAAY,CA2M3D;AAuLD;;;GAGG;AACH,wBAAgB,cAAc,CAAC,GAAG,EAAE,MAAM,GAAG;IAC3C,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,OAAO,EAAE,MAAM,EAAE,CAAC;IAClB,IAAI,EAAE,MAAM,EAAE,EAAE,CAAC;CAClB,CAQA;AA4PD;;GAEG;AACH,wBAAgB,aAAa,CAAC,GAAG,EAAE,MAAM,GAAG,WAAW,EAAE,CAwBxD;AA8JD;;GAEG;AACH,wBAAgB,mBAAmB,CAAC,GAAG,EAAE,MAAM,GAAG,aAAa,EAAE,CAkDhE;AAID,sDAAsD;AACtD,MAAM,WAAW,gBAAgB;IAC/B,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,UAAU,CAAC,EAAE,WAAW,EAAE,CAAC;IAC3B,SAAS,CAAC,EAAE,YAAY,EAAE,CAAC;IAC3B,MAAM,CAAC,EAAE,YAAY,EAAE,CAAC;IACxB,KAAK,CAAC,EAAE,cAAc,EAAE,CAAC;CAC1B;AAED;;;GAGG;AACH,wBAAgB,mBAAmB,CAAC,GAAG,EAAE,MAAM,GAAG,gBAAgB,CAuIjE"}
|
|
@@ -815,14 +815,16 @@ function parseJatsBackMatter(xml) {
|
|
|
815
815
|
const notes = [];
|
|
816
816
|
for (const note of notesElements) {
|
|
817
817
|
const secs = findChildren(note.children, "sec");
|
|
818
|
-
|
|
819
|
-
|
|
820
|
-
|
|
821
|
-
|
|
822
|
-
const
|
|
823
|
-
const
|
|
824
|
-
|
|
825
|
-
|
|
818
|
+
const nestedNotes = findChildren(note.children, "notes");
|
|
819
|
+
const subItems = secs.length > 0 ? secs : nestedNotes;
|
|
820
|
+
if (subItems.length > 0) {
|
|
821
|
+
for (const sub of subItems) {
|
|
822
|
+
const subTitleNode = findChild(sub.children, "title");
|
|
823
|
+
const subTitle = subTitleNode ? extractAllText(subTitleNode.children) : "";
|
|
824
|
+
const subParagraphs = findChildren(sub.children, "p");
|
|
825
|
+
const subText = subParagraphs.map((p) => extractAllText(p.children)).join("\n\n");
|
|
826
|
+
if (subTitle || subText) {
|
|
827
|
+
notes.push({ title: subTitle, text: subText });
|
|
826
828
|
}
|
|
827
829
|
}
|
|
828
830
|
} else {
|
|
@@ -839,6 +841,25 @@ function parseJatsBackMatter(xml) {
|
|
|
839
841
|
result.notes = notes;
|
|
840
842
|
}
|
|
841
843
|
}
|
|
844
|
+
const glossaryElements = findChildren(back.children, "glossary");
|
|
845
|
+
for (const glossary of glossaryElements) {
|
|
846
|
+
const titleNode = findChild(glossary.children, "title");
|
|
847
|
+
const title = titleNode ? extractAllText(titleNode.children) : "Glossary";
|
|
848
|
+
const defList = findChild(glossary.children, "def-list");
|
|
849
|
+
if (defList) {
|
|
850
|
+
const defItems = findChildren(defList.children, "def-item");
|
|
851
|
+
const lines = [];
|
|
852
|
+
for (const item of defItems) {
|
|
853
|
+
const termNode = findChild(item.children, "term");
|
|
854
|
+
const defNode = findChild(item.children, "def");
|
|
855
|
+
const term = termNode ? extractAllText(termNode.children) : "";
|
|
856
|
+
const definition = defNode ? extractAllText(defNode.children) : "";
|
|
857
|
+
lines.push(`${term}: ${definition}`);
|
|
858
|
+
}
|
|
859
|
+
if (!result.notes) result.notes = [];
|
|
860
|
+
result.notes.push({ title, text: lines.join("\n") });
|
|
861
|
+
}
|
|
862
|
+
}
|
|
842
863
|
}
|
|
843
864
|
const floatsGroup = findChild(article.children, "floats-group");
|
|
844
865
|
if (floatsGroup) {
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"jats-parser.js","sources":["../../../src/fulltext/convert/jats-parser.ts"],"sourcesContent":["/**\n * JATS XML parser for PMC articles.\n *\n * Parses JATS (Journal Article Tag Suite) XML into an intermediate\n * representation for Markdown conversion.\n *\n * Uses fast-xml-parser with `preserveOrder: true` to maintain document order\n * of interleaved elements (e.g. text, citations, formatting).\n */\n\nimport { XMLParser } from 'fast-xml-parser';\nimport type {\n JatsAuthor,\n JatsMetadata,\n JatsSection,\n JatsReference,\n JatsFootnote,\n BackMatterNote,\n BlockElement,\n InlineContent,\n} from './types.js';\n\n/**\n * A node in the preserveOrder output.\n * Either a text node `{ \"#text\": string | number }` or an element node\n * `{ tagName: OrderedNode[], \":@\"?: { \"@_attr\": value } }`.\n */\ntype OrderedNode = Record<string, unknown>;\n\nconst parser = new XMLParser({\n ignoreAttributes: false,\n attributeNamePrefix: '@_',\n textNodeName: '#text',\n trimValues: false,\n preserveOrder: true,\n processEntities: true,\n htmlEntities: true,\n});\n\n// ─── Navigation Helpers ──────────────────────────────────────────────\n\n/** Get the tag name of an ordered node (the first key that isn't \":@\" or \"#text\"). */\nfunction getTagName(node: OrderedNode): string | undefined {\n for (const key of Object.keys(node)) {\n if (key !== ':@' && key !== '#text') return key;\n }\n return undefined;\n}\n\n/** Get the children array of an element node. */\nfunction getChildren(node: OrderedNode): OrderedNode[] {\n const tag = getTagName(node);\n if (!tag) return [];\n const children = node[tag];\n return Array.isArray(children) ? (children as OrderedNode[]) : [];\n}\n\n/** Get attributes of an element node. */\nfunction getAttr(node: OrderedNode, attrName: string): string | undefined {\n const attrs = node[':@'] as Record<string, unknown> | undefined;\n if (!attrs) return undefined;\n const val = attrs[`@_${attrName}`];\n return val != null ? String(val) : undefined;\n}\n\n/** Get all attributes of an element node (strips @_ prefix for consistency with getAttr). */\nfunction getAttrs(node: OrderedNode): Record<string, string> {\n const attrs = node[':@'] as Record<string, unknown> | undefined;\n if (!attrs) return {};\n const result: Record<string, string> = {};\n for (const [key, value] of Object.entries(attrs)) {\n if (key.startsWith('@_')) {\n result[key.slice(2)] = String(value);\n }\n }\n return result;\n}\n\n/** Find the first child element with the given tag name. */\nfunction findChild(\n children: OrderedNode[],\n tagName: string,\n): { node: OrderedNode; children: OrderedNode[]; attrs: Record<string, string> } | undefined {\n for (const child of children) {\n if (tagName in child) {\n const childArr = child[tagName];\n return {\n node: child,\n children: Array.isArray(childArr) ? (childArr as OrderedNode[]) : [],\n attrs: getAttrs(child),\n };\n }\n }\n return undefined;\n}\n\n/** Find all child elements with the given tag name. */\nfunction findChildren(\n children: OrderedNode[],\n tagName: string,\n): Array<{ node: OrderedNode; children: OrderedNode[]; attrs: Record<string, string> }> {\n const results: Array<{\n node: OrderedNode;\n children: OrderedNode[];\n attrs: Record<string, string>;\n }> = [];\n for (const child of children) {\n if (tagName in child) {\n const childArr = child[tagName];\n results.push({\n node: child,\n children: Array.isArray(childArr) ? (childArr as OrderedNode[]) : [],\n attrs: getAttrs(child),\n });\n }\n }\n return results;\n}\n\n/** Get text content from a #text node. */\nfunction getTextContent(child: OrderedNode): string | undefined {\n if ('#text' in child) {\n const val = child['#text'];\n return val != null ? String(val) : undefined;\n }\n return undefined;\n}\n\n/**\n * Find the <article> element, handling optional <pmc-articleset> wrapper\n * that appears in efetch responses.\n */\nfunction findArticle(\n parsed: OrderedNode[],\n): { node: OrderedNode; children: OrderedNode[]; attrs: Record<string, string> } | undefined {\n const direct = findChild(parsed, 'article');\n if (direct) return direct;\n const wrapper = findChild(parsed, 'pmc-articleset');\n if (wrapper) return findChild(wrapper.children, 'article');\n return undefined;\n}\n\n// ─── Text Extraction ─────────────────────────────────────────────────\n\n/** Tags whose text content should be followed by a space when adjacent to other content. */\nconst SPACE_AFTER_TAGS = new Set([\n 'surname',\n 'given-names',\n 'name',\n 'string-name',\n]);\n\n/**\n * Extract plain text from a node that may contain nested elements.\n * Recursively collects all text content from preserveOrder nodes.\n *\n * When extracting text from inline container elements (e.g. `<name>`,\n * `<string-name>`), inserts a space between adjacent child elements\n * that would otherwise concatenate without whitespace (e.g.\n * `<surname>McGuire</surname><given-names>N</given-names>` → `McGuire N`).\n */\nfunction extractAllText(node: unknown): string {\n if (node == null) return '';\n if (typeof node === 'string') return node;\n if (typeof node === 'number') return String(node);\n if (Array.isArray(node)) {\n return joinChildTexts(node);\n }\n if (typeof node === 'object') {\n const obj = node as OrderedNode;\n // Text node\n const text = getTextContent(obj);\n if (text != null) return text;\n // Element node — recurse into children\n const tag = getTagName(obj);\n if (tag) {\n const children = obj[tag];\n if (Array.isArray(children)) {\n return joinChildTexts(children as OrderedNode[]);\n }\n }\n }\n return '';\n}\n\n/**\n * Join extracted text from an array of child nodes, inserting spaces\n * between adjacent inline elements where no whitespace separator exists.\n */\nfunction joinChildTexts(children: OrderedNode[]): string {\n const parts: string[] = [];\n for (const child of children) {\n const text = extractAllText(child);\n if (!text) continue;\n\n const tag = getTagName(child as OrderedNode);\n\n // If this is a space-after tag and there's previous content that doesn't\n // end with whitespace or punctuation, insert a space before this text.\n if (tag && SPACE_AFTER_TAGS.has(tag) && parts.length > 0) {\n const prev = parts[parts.length - 1]!;\n if (prev && !/[\\s,;.:()\\-/]$/.test(prev)) {\n parts.push(' ');\n }\n }\n\n parts.push(text);\n\n // If this is a space-after tag, check if a space is needed after.\n // We handle this by peeking: space will be inserted before the next\n // element if needed (handled above). But we also need to handle\n // the case where the next sibling is a text node starting without space.\n // That's already handled since text nodes include their own whitespace.\n }\n return parts.join('');\n}\n\n// ─── Metadata Parsing ────────────────────────────────────────────────\n\n/**\n * Parse JATS XML front matter to extract article metadata.\n */\nexport function parseJatsMetadata(xml: string): JatsMetadata {\n const parsed = parser.parse(xml) as OrderedNode[];\n const article = findArticle(parsed);\n if (!article) return { title: '', authors: [] };\n\n const front = findChild(article.children, 'front');\n if (!front) return { title: '', authors: [] };\n\n const articleMeta = findChild(front.children, 'article-meta');\n if (!articleMeta) return { title: '', authors: [] };\n\n const metaChildren = articleMeta.children;\n\n // Title\n const titleGroup = findChild(metaChildren, 'title-group');\n const articleTitle = titleGroup ? findChild(titleGroup.children, 'article-title') : undefined;\n const title = articleTitle ? extractAllText(articleTitle.children) : '';\n\n // Article IDs\n const articleIds = findChildren(metaChildren, 'article-id');\n let doi: string | undefined;\n let pmcid: string | undefined;\n let pmid: string | undefined;\n for (const idEntry of articleIds) {\n const idType = idEntry.attrs['pub-id-type'];\n const idText = extractAllText(idEntry.children);\n if (idType === 'doi') doi = idText;\n if (idType === 'pmc' || idType === 'pmcid') {\n pmcid = idText.replace(/^PMC/, '');\n }\n if (idType === 'pmid') pmid = idText;\n }\n\n // Authors\n const authors: JatsAuthor[] = [];\n const contribGroup = findChild(metaChildren, 'contrib-group');\n if (contribGroup) {\n const contribs = findChildren(contribGroup.children, 'contrib');\n for (const contrib of contribs) {\n if (contrib.attrs['contrib-type'] !== 'author') continue;\n const nameNode = findChild(contrib.children, 'name');\n if (!nameNode) continue;\n const surnameNode = findChild(nameNode.children, 'surname');\n const givenNamesNode = findChild(nameNode.children, 'given-names');\n const author: JatsAuthor = {\n surname: surnameNode ? extractAllText(surnameNode.children) : '',\n };\n const givenNames = givenNamesNode ? extractAllText(givenNamesNode.children) : '';\n if (givenNames) {\n author.givenNames = givenNames;\n }\n authors.push(author);\n }\n }\n\n // Abstract\n const abstractNode = findChild(metaChildren, 'abstract');\n let abstract: string | undefined;\n if (abstractNode) {\n // Structured abstract with <sec> elements\n const sections = findChildren(abstractNode.children, 'sec');\n if (sections.length > 0) {\n const parts: string[] = [];\n for (const sec of sections) {\n const secTitleNode = findChild(sec.children, 'title');\n const secTitle = secTitleNode ? extractAllText(secTitleNode.children) : '';\n const secPs = findChildren(sec.children, 'p');\n const text = secPs.map((p) => extractAllText(p.children)).join(' ');\n if (secTitle) {\n parts.push(`${secTitle}: ${text}`);\n } else {\n parts.push(text);\n }\n }\n abstract = parts.join('\\n\\n');\n } else {\n // Simple abstract with <p>\n const paragraphs = findChildren(abstractNode.children, 'p');\n if (paragraphs.length > 0) {\n abstract = paragraphs.map((p) => extractAllText(p.children)).join('\\n\\n');\n } else {\n const text = extractAllText(abstractNode.children);\n if (text) abstract = text;\n }\n }\n }\n\n // Publication date (from <article-meta>/<pub-date>)\n // Priority: epub > ppub > collection > any other\n const pubDates = findChildren(metaChildren, 'pub-date');\n let publicationDate: { year: string; month?: string; day?: string } | undefined;\n const datePriority: Record<string, number> = { epub: 0, ppub: 1, collection: 2 };\n let bestPriority = Infinity;\n for (const pd of pubDates) {\n // Support both pub-type (NLM/early JATS) and date-type (JATS 1.2+)\n const dateType = pd.attrs['pub-type'] ?? pd.attrs['date-type'] ?? '';\n const priority = datePriority[dateType] ?? 3;\n if (priority < bestPriority) {\n bestPriority = priority;\n const yearNode = findChild(pd.children, 'year');\n if (yearNode) {\n const year = extractAllText(yearNode.children);\n const monthNode = findChild(pd.children, 'month');\n const dayNode = findChild(pd.children, 'day');\n const date: { year: string; month?: string; day?: string } = { year };\n if (monthNode) date.month = extractAllText(monthNode.children);\n if (dayNode) date.day = extractAllText(dayNode.children);\n publicationDate = date;\n }\n }\n }\n // If no prioritized date found, take first available\n if (!publicationDate && pubDates.length > 0) {\n const pd = pubDates[0]!;\n const yearNode = findChild(pd.children, 'year');\n if (yearNode) {\n const year = extractAllText(yearNode.children);\n const monthNode = findChild(pd.children, 'month');\n const dayNode = findChild(pd.children, 'day');\n const date: { year: string; month?: string; day?: string } = { year };\n if (monthNode) date.month = extractAllText(monthNode.children);\n if (dayNode) date.day = extractAllText(dayNode.children);\n publicationDate = date;\n }\n }\n\n // Article type (from root <article> element attribute)\n const articleType = article.attrs['article-type'] || undefined;\n\n // License (from <permissions>/<license>)\n let license: string | undefined;\n const permissions = findChild(metaChildren, 'permissions');\n if (permissions) {\n const licenseNode = findChild(permissions.children, 'license');\n if (licenseNode) {\n // Prefer @xlink:href (standardized URL) over <license-p> (free-text)\n const href = licenseNode.attrs['xlink:href'];\n if (href) {\n license = href;\n } else {\n const licenseP = findChild(licenseNode.children, 'license-p');\n if (licenseP) license = extractAllText(licenseP.children).trim();\n }\n }\n }\n\n // Keywords (from all <kwd-group> elements)\n const kwdGroups = findChildren(metaChildren, 'kwd-group');\n const keywords: string[] = [];\n for (const kwdGroup of kwdGroups) {\n const kwds = findChildren(kwdGroup.children, 'kwd');\n for (const kwd of kwds) {\n const text = extractAllText(kwd.children).trim();\n if (text) keywords.push(text);\n }\n }\n\n // Volume, issue, pages\n const volumeNode = findChild(metaChildren, 'volume');\n const volume = volumeNode ? extractAllText(volumeNode.children) : undefined;\n const issueNode = findChild(metaChildren, 'issue');\n const issue = issueNode ? extractAllText(issueNode.children) : undefined;\n let pages: string | undefined;\n const fpageNode = findChild(metaChildren, 'fpage');\n const lpageNode = findChild(metaChildren, 'lpage');\n if (fpageNode) {\n const fp = extractAllText(fpageNode.children);\n const lp = lpageNode ? extractAllText(lpageNode.children) : '';\n pages = lp ? `${fp}-${lp}` : fp;\n } else {\n const elocationNode = findChild(metaChildren, 'elocation-id');\n if (elocationNode) pages = extractAllText(elocationNode.children);\n }\n\n // Journal name (from <front>/<journal-meta>)\n const journalMeta = findChild(front.children, 'journal-meta');\n let journal: string | undefined;\n if (journalMeta) {\n const titleGroup = findChild(journalMeta.children, 'journal-title-group');\n if (titleGroup) {\n const jTitle = findChild(titleGroup.children, 'journal-title');\n if (jTitle) journal = extractAllText(jTitle.children);\n }\n if (!journal) {\n const jTitle = findChild(journalMeta.children, 'journal-title');\n if (jTitle) journal = extractAllText(jTitle.children);\n }\n }\n\n const result: JatsMetadata = { title, authors };\n if (doi) result.doi = doi;\n if (pmcid) result.pmcid = pmcid;\n if (pmid) result.pmid = pmid;\n if (journal) result.journal = journal;\n if (publicationDate) result.publicationDate = publicationDate;\n if (volume) result.volume = volume;\n if (issue) result.issue = issue;\n if (pages) result.pages = pages;\n if (keywords.length > 0) result.keywords = keywords;\n if (articleType) result.articleType = articleType;\n if (license) result.license = license;\n if (abstract) result.abstract = abstract;\n return result;\n}\n\n// ─── Inline Content Parsing ──────────────────────────────────────────\n\n/**\n * Parse inline content from a paragraph's children array.\n * Iterates in document order to preserve interleaving of text, citations,\n * and formatting elements.\n */\nfunction parseInlineContent(children: OrderedNode[]): InlineContent[] {\n const result: InlineContent[] = [];\n\n for (const child of children) {\n // Text node\n const text = getTextContent(child);\n if (text != null) {\n if (text) result.push({ type: 'text', text });\n continue;\n }\n\n const tag = getTagName(child);\n if (!tag) continue;\n\n const innerChildren = getChildren(child);\n\n if (tag === 'bold') {\n result.push({ type: 'bold', children: parseInlineContent(innerChildren) });\n } else if (tag === 'italic') {\n result.push({ type: 'italic', children: parseInlineContent(innerChildren) });\n } else if (tag === 'sup') {\n result.push({ type: 'superscript', text: extractAllText(innerChildren) });\n } else if (tag === 'sub') {\n result.push({ type: 'subscript', text: extractAllText(innerChildren) });\n } else if (tag === 'inline-formula') {\n // Try to find <tex-math> directly or inside <alternatives>\n let texMath = findChild(innerChildren, 'tex-math');\n if (!texMath) {\n const alternatives = findChild(innerChildren, 'alternatives');\n if (alternatives) {\n texMath = findChild(alternatives.children, 'tex-math');\n }\n }\n const tex = texMath ? extractAllText(texMath.children) : undefined;\n const text = tex || extractAllText(innerChildren);\n const entry: { type: 'inline-formula'; tex?: string; text: string } = {\n type: 'inline-formula',\n text,\n };\n if (tex) entry.tex = tex;\n result.push(entry);\n } else if (tag === 'monospace') {\n result.push({ type: 'code', text: extractAllText(innerChildren) });\n } else if (tag === 'ext-link') {\n const href = getAttr(child, 'xlink:href');\n if (href) {\n result.push({ type: 'link', url: href, children: parseInlineContent(innerChildren) });\n } else {\n const linkText = extractAllText(innerChildren);\n if (linkText) result.push({ type: 'text', text: linkText });\n }\n } else if (tag === 'uri') {\n const href = getAttr(child, 'xlink:href');\n const textContent = extractAllText(innerChildren);\n const url = href || textContent;\n if (url) {\n result.push({ type: 'link', url, children: parseInlineContent(innerChildren) });\n }\n } else if (tag === 'underline' || tag === 'sc') {\n // Pass-through: preserve text content without special formatting\n const passText = extractAllText(innerChildren);\n if (passText) result.push({ type: 'text', text: passText });\n } else if (tag === 'xref') {\n const refType = getAttr(child, 'ref-type');\n if (refType === 'bibr') {\n result.push({\n type: 'citation',\n refId: getAttr(child, 'rid') ?? '',\n text: extractAllText(innerChildren),\n });\n } else {\n const xrefText = extractAllText(innerChildren);\n if (xrefText) result.push({ type: 'text', text: xrefText });\n }\n } else {\n // Unknown inline element — extract text\n const unknownText = extractAllText(innerChildren);\n if (unknownText) result.push({ type: 'text', text: unknownText });\n }\n }\n\n return result;\n}\n\n// ─── Block Content Parsing ───────────────────────────────────────────\n\n/**\n * Parse a <list> element into a BlockElement.\n */\nfunction parseList(listNode: OrderedNode): BlockElement {\n const listType = getAttr(listNode, 'list-type');\n const ordered = listType === 'order';\n const listChildren = getChildren(listNode);\n const listItems = findChildren(listChildren, 'list-item');\n const items: InlineContent[][] = [];\n\n for (const item of listItems) {\n const pNodes = findChildren(item.children, 'p');\n const content = pNodes.flatMap((p) => parseInlineContent(p.children));\n items.push(content);\n }\n\n return { type: 'list', ordered, items };\n}\n\n/**\n * Parse a table row into an array of cell text content.\n */\nfunction parseTableRow(trChildren: OrderedNode[]): string[] {\n const cells: string[] = [];\n for (const child of trChildren) {\n const tag = getTagName(child);\n if (tag === 'th' || tag === 'td') {\n const cellChildren = getChildren(child);\n // Check if cell contains multiple <p> elements\n const paragraphs = findChildren(cellChildren, 'p');\n if (paragraphs.length > 1) {\n cells.push(\n paragraphs.map((p) => extractAllText(p.children)).join('<br>'),\n );\n } else {\n cells.push(extractAllText(cellChildren));\n }\n }\n }\n return cells;\n}\n\n/**\n * Parse an already-parsed table-wrap node.\n */\nfunction parseTableWrap(tableWrapNode: OrderedNode): {\n caption?: string;\n headers: string[];\n rows: string[][];\n} {\n const children = getChildren(tableWrapNode);\n\n // Caption\n const labelNode = findChild(children, 'label');\n const label = labelNode ? extractAllText(labelNode.children) : '';\n const captionNode = findChild(children, 'caption');\n const captionText = captionNode ? extractAllText(captionNode.children) : '';\n const captionStr = [label, captionText].filter(Boolean).join('. ');\n\n const tableNode = findChild(children, 'table');\n const result: { caption?: string; headers: string[]; rows: string[][] } = {\n headers: [],\n rows: [],\n };\n if (captionStr) result.caption = captionStr;\n if (!tableNode) return result;\n\n // Headers from thead\n const thead = findChild(tableNode.children, 'thead');\n if (thead) {\n const headRows = findChildren(thead.children, 'tr');\n if (headRows.length > 0) {\n result.headers.push(...parseTableRow(headRows[0]!.children));\n }\n }\n\n // Body rows\n const tbody = findChild(tableNode.children, 'tbody');\n if (tbody) {\n const bodyRows = findChildren(tbody.children, 'tr');\n for (const row of bodyRows) {\n result.rows.push(parseTableRow(row.children));\n }\n }\n\n return result;\n}\n\n/**\n * Parse a <table-wrap> element into a table block.\n * Exported for standalone use and used internally by parseBlockContent.\n */\nexport function parseJatsTable(xml: string): {\n caption?: string;\n headers: string[];\n rows: string[][];\n} {\n const parsed = parser.parse(xml) as OrderedNode[];\n const tableWrap = findChild(parsed, 'table-wrap');\n if (tableWrap) {\n return parseTableWrap(tableWrap.node);\n }\n // Fallback: if not wrapped, try to find table directly\n return { headers: [], rows: [] };\n}\n\n/**\n * Parse a <boxed-text> element into a boxed-text block.\n * Extracts optional title and recursively parses inner block content.\n */\nfunction parseBoxedText(node: OrderedNode): BlockElement {\n const children = getChildren(node);\n const titleNode = findChild(children, 'title');\n const title = titleNode ? extractAllText(titleNode.children) : undefined;\n const content = parseBlockContent(children);\n const block: BlockElement = { type: 'boxed-text', content };\n if (title) block.title = title;\n return block;\n}\n\n/**\n * Parse a <def-list> element into a def-list block.\n * Extracts optional title and <def-item> pairs with <term> and <def>.\n */\nfunction parseDefList(node: OrderedNode): BlockElement {\n const children = getChildren(node);\n const titleNode = findChild(children, 'title');\n const title = titleNode ? extractAllText(titleNode.children) : undefined;\n const defItems = findChildren(children, 'def-item');\n const items: { term: string; definition: string }[] = [];\n for (const item of defItems) {\n const termNode = findChild(item.children, 'term');\n const defNode = findChild(item.children, 'def');\n const term = termNode ? extractAllText(termNode.children) : '';\n const definition = defNode ? extractAllText(defNode.children) : '';\n items.push({ term, definition });\n }\n const block: BlockElement = { type: 'def-list', items };\n if (title) block.title = title;\n return block;\n}\n\n/**\n * Parse a <disp-formula> element into a formula block.\n * Extracts TeX content from <tex-math> preferentially (inside <alternatives> or direct),\n * falls back to extractAllText for plain text.\n */\nfunction parseDispFormula(node: OrderedNode): BlockElement {\n const children = getChildren(node);\n const id = getAttr(node, 'id');\n const labelNode = findChild(children, 'label');\n const label = labelNode ? extractAllText(labelNode.children) : undefined;\n\n // Try <alternatives> wrapper first\n const alternatives = findChild(children, 'alternatives');\n const searchChildren = alternatives ? alternatives.children : children;\n\n const texMath = findChild(searchChildren, 'tex-math');\n const block: BlockElement = { type: 'formula' };\n if (id) block.id = id;\n if (label) block.label = label;\n\n if (texMath) {\n block.tex = extractAllText(texMath.children);\n } else {\n // Fall back to plain text extraction (skip label)\n const textChildren = children.filter((c) => !('label' in c));\n const text = extractAllText(textChildren).trim();\n if (text) block.text = text;\n }\n\n return block;\n}\n\n/** Tags that represent block-level elements when nested inside <p>. */\nconst BLOCK_TAGS = new Set(['table-wrap', 'fig', 'disp-quote', 'boxed-text']);\n\n/**\n * Parse a <disp-quote> element into a blockquote block.\n * Extracts <p> children and concatenates their inline content.\n */\nfunction parseDispQuote(node: OrderedNode): BlockElement {\n const children = getChildren(node);\n const paragraphs = findChildren(children, 'p');\n const content: InlineContent[] = [];\n for (let i = 0; i < paragraphs.length; i++) {\n if (i > 0) content.push({ type: 'text', text: '\\n\\n' });\n const para = paragraphs[i];\n if (para) content.push(...parseInlineContent(para.children));\n }\n // If no <p> children, extract inline content directly\n if (paragraphs.length === 0) {\n content.push(...parseInlineContent(children));\n }\n return { type: 'blockquote', content };\n}\n\n/**\n * Parse a <table-wrap> node into a table block element.\n */\nfunction parseTableBlock(node: OrderedNode): BlockElement {\n const tableResult = parseTableWrap(node);\n const tableBlock: BlockElement = {\n type: 'table',\n headers: tableResult.headers,\n rows: tableResult.rows,\n };\n if (tableResult.caption) tableBlock.caption = tableResult.caption;\n return tableBlock;\n}\n\n/**\n * Parse a <fig> node into a figure block element.\n */\nfunction parseFigBlock(node: OrderedNode): BlockElement {\n const innerChildren = getChildren(node);\n const figBlock: BlockElement = { type: 'figure' };\n const figId = getAttr(node, 'id');\n if (figId) figBlock.id = figId;\n const figLabel = findChild(innerChildren, 'label');\n if (figLabel) {\n const labelText = extractAllText(figLabel.children);\n if (labelText) figBlock.label = labelText;\n }\n const figCaption = findChild(innerChildren, 'caption');\n if (figCaption) {\n const captionText = extractAllText(figCaption.children);\n if (captionText) figBlock.caption = captionText;\n }\n return figBlock;\n}\n\n/**\n * Parse a <p> element, splitting it if it contains nested block elements\n * (table-wrap, fig, disp-quote). Returns one or more block elements.\n */\nfunction parseParagraph(pChildren: OrderedNode[]): BlockElement[] {\n // Check if <p> contains any nested block elements\n const hasNestedBlocks = pChildren.some((child) => {\n const tag = getTagName(child);\n return tag != null && BLOCK_TAGS.has(tag);\n });\n\n if (!hasNestedBlocks) {\n return [{ type: 'paragraph', content: parseInlineContent(pChildren) }];\n }\n\n // Split into inline runs and block elements\n const blocks: BlockElement[] = [];\n let inlineBuffer: OrderedNode[] = [];\n\n const flushInline = () => {\n if (inlineBuffer.length > 0) {\n const content = parseInlineContent(inlineBuffer);\n // Skip whitespace-only paragraphs created by XML formatting\n const hasNonWhitespace = content.some(\n (c) => c.type !== 'text' || c.text.trim() !== '',\n );\n if (content.length > 0 && hasNonWhitespace) {\n blocks.push({ type: 'paragraph', content });\n }\n inlineBuffer = [];\n }\n };\n\n for (const child of pChildren) {\n const tag = getTagName(child);\n if (tag === 'table-wrap') {\n flushInline();\n blocks.push(parseTableBlock(child));\n } else if (tag === 'fig') {\n flushInline();\n blocks.push(parseFigBlock(child));\n } else if (tag === 'disp-quote') {\n flushInline();\n blocks.push(parseDispQuote(child));\n } else if (tag === 'boxed-text') {\n flushInline();\n blocks.push(parseBoxedText(child));\n } else {\n inlineBuffer.push(child);\n }\n }\n flushInline();\n\n return blocks;\n}\n\n/**\n * Parse block-level content from a section's children.\n * Iterates in document order to preserve ordering of paragraphs, lists,\n * tables, figures, and blockquotes.\n */\nfunction parseBlockContent(sectionChildren: OrderedNode[]): BlockElement[] {\n const blocks: BlockElement[] = [];\n\n for (const child of sectionChildren) {\n const tag = getTagName(child);\n if (!tag) continue;\n\n if (tag === 'p') {\n blocks.push(...parseParagraph(getChildren(child)));\n } else if (tag === 'list') {\n blocks.push(parseList(child));\n } else if (tag === 'table-wrap') {\n blocks.push(parseTableBlock(child));\n } else if (tag === 'fig') {\n blocks.push(parseFigBlock(child));\n } else if (tag === 'disp-quote') {\n blocks.push(parseDispQuote(child));\n } else if (tag === 'boxed-text') {\n blocks.push(parseBoxedText(child));\n } else if (tag === 'def-list') {\n blocks.push(parseDefList(child));\n } else if (tag === 'disp-formula') {\n blocks.push(parseDispFormula(child));\n } else if (tag === 'preformat') {\n const text = extractAllText(getChildren(child));\n blocks.push({ type: 'preformat', text });\n } else if (tag === 'supplementary-material') {\n const innerChildren = getChildren(child);\n const labelNode = findChild(innerChildren, 'label');\n const captionNode = findChild(innerChildren, 'caption');\n const labelText = labelNode ? extractAllText(labelNode.children) : '';\n const captionText = captionNode ? extractAllText(captionNode.children) : '';\n const text = [labelText, captionText].filter(Boolean).join(': ');\n if (text) {\n blocks.push({ type: 'paragraph', content: [{ type: 'text', text }] });\n }\n }\n // Skip title, sec, and other non-block elements\n }\n\n return blocks;\n}\n\n// ─── Section Parsing ─────────────────────────────────────────────────\n\n/**\n * Parse a <sec> element into a JatsSection, recursively handling subsections.\n */\nfunction parseSection(secChildren: OrderedNode[], level: number): JatsSection {\n const titleNode = findChild(secChildren, 'title');\n const title = titleNode ? extractAllText(titleNode.children) : '';\n const content = parseBlockContent(secChildren);\n\n // Nested sections\n const subsections: JatsSection[] = [];\n const nestedSecs = findChildren(secChildren, 'sec');\n for (const sub of nestedSecs) {\n subsections.push(parseSection(sub.children, level + 1));\n }\n\n return { title, level, content, subsections };\n}\n\n/**\n * Parse JATS XML body to extract sections and content.\n */\nexport function parseJatsBody(xml: string): JatsSection[] {\n const parsed = parser.parse(xml) as OrderedNode[];\n const article = findArticle(parsed);\n if (!article) return [];\n\n const body = findChild(article.children, 'body');\n if (!body) return [];\n\n const sections: JatsSection[] = [];\n const secs = findChildren(body.children, 'sec');\n\n if (secs.length > 0) {\n for (const sec of secs) {\n sections.push(parseSection(sec.children, 2));\n }\n } else {\n // Body has paragraphs without sections\n const content = parseBlockContent(body.children);\n if (content.length > 0) {\n sections.push({ title: '', level: 2, content, subsections: [] });\n }\n }\n\n return sections;\n}\n\n// ─── Reference Parsing ───────────────────────────────────────────────\n\n/**\n * Format a structured <element-citation> into a readable reference string.\n * Produces: \"Author1, Author2. Title. Source. Year;Volume:FirstPage-LastPage.\"\n */\nfunction formatElementCitation(children: OrderedNode[]): string {\n const parts: string[] = [];\n\n // Authors from person-group\n const personGroup = findChild(children, 'person-group');\n if (personGroup) {\n const names = findChildren(personGroup.children, 'name');\n const authorParts: string[] = [];\n for (const name of names) {\n const surname = findChild(name.children, 'surname');\n const givenNames = findChild(name.children, 'given-names');\n const surnameText = surname ? extractAllText(surname.children) : '';\n const givenText = givenNames ? extractAllText(givenNames.children) : '';\n if (surnameText && givenText) {\n authorParts.push(`${surnameText} ${givenText}`);\n } else if (surnameText) {\n authorParts.push(surnameText);\n }\n }\n if (authorParts.length > 0) {\n parts.push(authorParts.join(', '));\n }\n }\n\n // Article title\n const articleTitle = findChild(children, 'article-title');\n if (articleTitle) {\n parts.push(extractAllText(articleTitle.children));\n }\n\n // Source (journal name)\n const source = findChild(children, 'source');\n if (source) {\n parts.push(extractAllText(source.children));\n }\n\n // Year, volume, pages\n const year = findChild(children, 'year');\n const volume = findChild(children, 'volume');\n const fpage = findChild(children, 'fpage');\n const lpage = findChild(children, 'lpage');\n\n if (year) {\n let yearStr = extractAllText(year.children);\n if (volume) {\n yearStr += `;${extractAllText(volume.children)}`;\n }\n if (fpage) {\n const fpageText = extractAllText(fpage.children);\n const lpageText = lpage ? extractAllText(lpage.children) : '';\n yearStr += `:${fpageText}${lpageText ? `-${lpageText}` : ''}`;\n }\n parts.push(yearStr);\n }\n\n return parts.join('. ') + '.';\n}\n\n/**\n * Extract text from a <mixed-citation>'s children, deduplicating any\n * <pub-id> content that also appears as inline text.\n *\n * Some publishers include the DOI/PMID both as a text node and inside\n * a <pub-id> element, causing duplication like \"10.1234/x 10.1234/x\".\n */\nfunction extractMixedCitationText(children: OrderedNode[]): string {\n // Collect pub-id values\n const pubIds = findChildren(children, 'pub-id');\n const pubIdValues = pubIds\n .map((p) => extractAllText(p.children).trim())\n .filter(Boolean);\n\n if (pubIdValues.length === 0) {\n return extractAllText(children).trim();\n }\n\n // Extract full text\n const fullText = extractAllText(children).trim();\n\n // For each pub-id value, if it appears more than once, remove extra occurrences\n let result = fullText;\n for (const val of pubIdValues) {\n // Escape regex special characters\n const escaped = val.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&');\n const matches = result.match(new RegExp(escaped, 'g'));\n if (matches && matches.length > 1) {\n // Remove the first occurrence (typically the inline text), keep the last (pub-id element)\n result = result.replace(val, '');\n // Clean up any leftover extra whitespace\n result = result.replace(/\\s{2,}/g, ' ').trim();\n }\n }\n\n return result;\n}\n\n/**\n * Extract structured pub-id values (DOI, PMID, PMCID) from children nodes.\n */\nfunction extractPubIds(children: OrderedNode[]): {\n doi?: string;\n pmid?: string;\n pmcid?: string;\n} {\n const pubIds = findChildren(children, 'pub-id');\n const result: { doi?: string; pmid?: string; pmcid?: string } = {};\n for (const p of pubIds) {\n const idType = p.attrs['pub-id-type'];\n const value = extractAllText(p.children).trim();\n if (!value) continue;\n if (idType === 'doi') result.doi = value;\n if (idType === 'pmid') result.pmid = value;\n if (idType === 'pmc' || idType === 'pmcid') {\n result.pmcid = value.replace(/^PMC/, '');\n }\n }\n return result;\n}\n\n/**\n * Strip extracted pub-id values from reference text to avoid duplication\n * when pub-ids are rendered separately as links.\n */\nfunction stripPubIdValues(\n text: string,\n pubIds: { doi?: string; pmid?: string; pmcid?: string },\n): string {\n let result = text;\n const values = [pubIds.doi, pubIds.pmid, pubIds.pmcid].filter(Boolean) as string[];\n for (const val of values) {\n // Strip common label prefixes (e.g. \"doi: \", \"PMID: \", \"DOI:\", \"pmid:\") followed by the value\n const escaped = val.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&');\n result = result.replace(new RegExp(`(?:doi|PMID|pmid|PMC|pmc)[:\\\\s]*${escaped}`, 'gi'), '');\n // Also strip the bare value itself\n result = result.replace(new RegExp(escaped, 'g'), '');\n }\n // Also strip PMC-prefixed form of pmcid\n if (pubIds.pmcid) {\n const pmcFull = `PMC${pubIds.pmcid}`;\n const escaped = pmcFull.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&');\n result = result.replace(new RegExp(`(?:pmc|pmcid)[:\\\\s]*${escaped}`, 'gi'), '');\n result = result.replace(new RegExp(escaped, 'g'), '');\n }\n // Clean up trailing/leading whitespace and extra spaces\n result = result.replace(/\\s{2,}/g, ' ').trim();\n // Clean up trailing period after stripped content (e.g. \"Title. .\" -> \"Title.\")\n result = result.replace(/\\.\\s*\\.$/, '.');\n return result;\n}\n\n/**\n * Parse JATS XML back matter to extract references.\n */\nexport function parseJatsReferences(xml: string): JatsReference[] {\n const parsed = parser.parse(xml) as OrderedNode[];\n const article = findArticle(parsed);\n if (!article) return [];\n\n const back = findChild(article.children, 'back');\n if (!back) return [];\n\n const refList = findChild(back.children, 'ref-list');\n if (!refList) return [];\n\n const refs = findChildren(refList.children, 'ref');\n const references: JatsReference[] = [];\n\n for (const ref of refs) {\n const id = ref.attrs['id'] ?? '';\n\n // Determine the search scope: if <citation-alternatives> exists, search within it;\n // otherwise search direct children of <ref>\n const citationAlternatives = findChild(ref.children, 'citation-alternatives');\n const searchChildren = citationAlternatives ? citationAlternatives.children : ref.children;\n\n // Try mixed-citation first (already formatted), then element-citation (structured)\n const mixedCitation = findChild(searchChildren, 'mixed-citation');\n if (mixedCitation) {\n const rawText = extractMixedCitationText(mixedCitation.children);\n const pubIds = extractPubIds(mixedCitation.children);\n const text = stripPubIdValues(rawText, pubIds);\n if (id && text) references.push({ id, text, ...pubIds });\n continue;\n }\n\n const elementCitation = findChild(searchChildren, 'element-citation');\n if (elementCitation) {\n const rawText = formatElementCitation(elementCitation.children);\n const pubIds = extractPubIds(elementCitation.children);\n const text = stripPubIdValues(rawText, pubIds);\n if (id && text) references.push({ id, text, ...pubIds });\n continue;\n }\n\n // Fallback: extract all text from ref, skipping <label>\n const childrenWithoutLabel = ref.children.filter((c) => !('label' in c));\n const text = extractAllText(childrenWithoutLabel).trim();\n if (id && text) {\n references.push({ id, text });\n }\n }\n\n return references;\n}\n\n// ─── Back Matter & Floats Parsing ────────────────────────────────────\n\n/** Result of parsing back matter and floats-group. */\nexport interface BackMatterResult {\n acknowledgments?: string;\n appendices?: JatsSection[];\n footnotes?: JatsFootnote[];\n floats?: BlockElement[];\n notes?: BackMatterNote[];\n}\n\n/**\n * Parse JATS XML back matter sections (ack, app-group, fn-group)\n * and top-level floats-group.\n */\nexport function parseJatsBackMatter(xml: string): BackMatterResult {\n const parsed = parser.parse(xml) as OrderedNode[];\n const article = findArticle(parsed);\n if (!article) return {};\n\n const result: BackMatterResult = {};\n\n // Parse <back> children\n const back = findChild(article.children, 'back');\n if (back) {\n // Acknowledgments: <ack>\n const ack = findChild(back.children, 'ack');\n if (ack) {\n const paragraphs = findChildren(ack.children, 'p');\n if (paragraphs.length > 0) {\n result.acknowledgments = paragraphs\n .map((p) => extractAllText(p.children))\n .join('\\n\\n');\n }\n }\n\n // Appendices: <app-group>/<app>\n const appGroup = findChild(back.children, 'app-group');\n if (appGroup) {\n const apps = findChildren(appGroup.children, 'app');\n if (apps.length > 0) {\n result.appendices = apps.map((app) => parseSection(app.children, 2));\n }\n }\n\n // Footnotes: <fn-group>/<fn>\n const fnGroup = findChild(back.children, 'fn-group');\n if (fnGroup) {\n const fns = findChildren(fnGroup.children, 'fn');\n if (fns.length > 0) {\n result.footnotes = fns.map((fn) => {\n const parts: string[] = [];\n // Include <title> if present\n const titleNode = findChild(fn.children, 'title');\n if (titleNode) {\n const titleText = extractAllText(titleNode.children).trim();\n if (titleText) parts.push(titleText);\n }\n // Extract text from each <p> separately and join with space\n const paragraphs = findChildren(fn.children, 'p');\n for (const p of paragraphs) {\n const pText = extractAllText(p.children).trim();\n if (pText) parts.push(pText);\n }\n return {\n id: fn.attrs['id'] ?? '',\n text: parts.join(' '),\n };\n });\n }\n }\n\n // Notes: <notes> (author contributions, funding, data availability, etc.)\n const notesElements = findChildren(back.children, 'notes');\n if (notesElements.length > 0) {\n const notes: BackMatterNote[] = [];\n for (const note of notesElements) {\n // Check if this <notes> contains <sec> children (e.g. Declarations wrapper)\n const secs = findChildren(note.children, 'sec');\n if (secs.length > 0) {\n for (const sec of secs) {\n const secTitleNode = findChild(sec.children, 'title');\n const secTitle = secTitleNode ? extractAllText(secTitleNode.children) : '';\n const secParagraphs = findChildren(sec.children, 'p');\n const secText = secParagraphs\n .map((p) => extractAllText(p.children))\n .join('\\n\\n');\n if (secTitle || secText) {\n notes.push({ title: secTitle, text: secText });\n }\n }\n } else {\n const titleNode = findChild(note.children, 'title');\n const title = titleNode ? extractAllText(titleNode.children) : '';\n const paragraphs = findChildren(note.children, 'p');\n const text = paragraphs\n .map((p) => extractAllText(p.children))\n .join('\\n\\n');\n if (title || text) {\n notes.push({ title, text });\n }\n }\n }\n if (notes.length > 0) {\n result.notes = notes;\n }\n }\n }\n\n // Floats-group: <floats-group> (sibling of <body> and <back>)\n const floatsGroup = findChild(article.children, 'floats-group');\n if (floatsGroup) {\n const blocks: BlockElement[] = [];\n for (const child of floatsGroup.children) {\n const tag = getTagName(child);\n if (tag === 'fig') {\n blocks.push(parseFigBlock(child));\n } else if (tag === 'table-wrap') {\n blocks.push(parseTableBlock(child));\n }\n }\n if (blocks.length > 0) {\n result.floats = blocks;\n }\n }\n\n return result;\n}\n"],"names":["titleGroup","text"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"jats-parser.js","sources":["../../../src/fulltext/convert/jats-parser.ts"],"sourcesContent":["/**\n * JATS XML parser for PMC articles.\n *\n * Parses JATS (Journal Article Tag Suite) XML into an intermediate\n * representation for Markdown conversion.\n *\n * Uses fast-xml-parser with `preserveOrder: true` to maintain document order\n * of interleaved elements (e.g. text, citations, formatting).\n */\n\nimport { XMLParser } from 'fast-xml-parser';\nimport type {\n JatsAuthor,\n JatsMetadata,\n JatsSection,\n JatsReference,\n JatsFootnote,\n BackMatterNote,\n BlockElement,\n InlineContent,\n} from './types.js';\n\n/**\n * A node in the preserveOrder output.\n * Either a text node `{ \"#text\": string | number }` or an element node\n * `{ tagName: OrderedNode[], \":@\"?: { \"@_attr\": value } }`.\n */\ntype OrderedNode = Record<string, unknown>;\n\nconst parser = new XMLParser({\n ignoreAttributes: false,\n attributeNamePrefix: '@_',\n textNodeName: '#text',\n trimValues: false,\n preserveOrder: true,\n processEntities: true,\n htmlEntities: true,\n});\n\n// ─── Navigation Helpers ──────────────────────────────────────────────\n\n/** Get the tag name of an ordered node (the first key that isn't \":@\" or \"#text\"). */\nfunction getTagName(node: OrderedNode): string | undefined {\n for (const key of Object.keys(node)) {\n if (key !== ':@' && key !== '#text') return key;\n }\n return undefined;\n}\n\n/** Get the children array of an element node. */\nfunction getChildren(node: OrderedNode): OrderedNode[] {\n const tag = getTagName(node);\n if (!tag) return [];\n const children = node[tag];\n return Array.isArray(children) ? (children as OrderedNode[]) : [];\n}\n\n/** Get attributes of an element node. */\nfunction getAttr(node: OrderedNode, attrName: string): string | undefined {\n const attrs = node[':@'] as Record<string, unknown> | undefined;\n if (!attrs) return undefined;\n const val = attrs[`@_${attrName}`];\n return val != null ? String(val) : undefined;\n}\n\n/** Get all attributes of an element node (strips @_ prefix for consistency with getAttr). */\nfunction getAttrs(node: OrderedNode): Record<string, string> {\n const attrs = node[':@'] as Record<string, unknown> | undefined;\n if (!attrs) return {};\n const result: Record<string, string> = {};\n for (const [key, value] of Object.entries(attrs)) {\n if (key.startsWith('@_')) {\n result[key.slice(2)] = String(value);\n }\n }\n return result;\n}\n\n/** Find the first child element with the given tag name. */\nfunction findChild(\n children: OrderedNode[],\n tagName: string,\n): { node: OrderedNode; children: OrderedNode[]; attrs: Record<string, string> } | undefined {\n for (const child of children) {\n if (tagName in child) {\n const childArr = child[tagName];\n return {\n node: child,\n children: Array.isArray(childArr) ? (childArr as OrderedNode[]) : [],\n attrs: getAttrs(child),\n };\n }\n }\n return undefined;\n}\n\n/** Find all child elements with the given tag name. */\nfunction findChildren(\n children: OrderedNode[],\n tagName: string,\n): Array<{ node: OrderedNode; children: OrderedNode[]; attrs: Record<string, string> }> {\n const results: Array<{\n node: OrderedNode;\n children: OrderedNode[];\n attrs: Record<string, string>;\n }> = [];\n for (const child of children) {\n if (tagName in child) {\n const childArr = child[tagName];\n results.push({\n node: child,\n children: Array.isArray(childArr) ? (childArr as OrderedNode[]) : [],\n attrs: getAttrs(child),\n });\n }\n }\n return results;\n}\n\n/** Get text content from a #text node. */\nfunction getTextContent(child: OrderedNode): string | undefined {\n if ('#text' in child) {\n const val = child['#text'];\n return val != null ? String(val) : undefined;\n }\n return undefined;\n}\n\n/**\n * Find the <article> element, handling optional <pmc-articleset> wrapper\n * that appears in efetch responses.\n */\nfunction findArticle(\n parsed: OrderedNode[],\n): { node: OrderedNode; children: OrderedNode[]; attrs: Record<string, string> } | undefined {\n const direct = findChild(parsed, 'article');\n if (direct) return direct;\n const wrapper = findChild(parsed, 'pmc-articleset');\n if (wrapper) return findChild(wrapper.children, 'article');\n return undefined;\n}\n\n// ─── Text Extraction ─────────────────────────────────────────────────\n\n/** Tags whose text content should be followed by a space when adjacent to other content. */\nconst SPACE_AFTER_TAGS = new Set([\n 'surname',\n 'given-names',\n 'name',\n 'string-name',\n]);\n\n/**\n * Extract plain text from a node that may contain nested elements.\n * Recursively collects all text content from preserveOrder nodes.\n *\n * When extracting text from inline container elements (e.g. `<name>`,\n * `<string-name>`), inserts a space between adjacent child elements\n * that would otherwise concatenate without whitespace (e.g.\n * `<surname>McGuire</surname><given-names>N</given-names>` → `McGuire N`).\n */\nfunction extractAllText(node: unknown): string {\n if (node == null) return '';\n if (typeof node === 'string') return node;\n if (typeof node === 'number') return String(node);\n if (Array.isArray(node)) {\n return joinChildTexts(node);\n }\n if (typeof node === 'object') {\n const obj = node as OrderedNode;\n // Text node\n const text = getTextContent(obj);\n if (text != null) return text;\n // Element node — recurse into children\n const tag = getTagName(obj);\n if (tag) {\n const children = obj[tag];\n if (Array.isArray(children)) {\n return joinChildTexts(children as OrderedNode[]);\n }\n }\n }\n return '';\n}\n\n/**\n * Join extracted text from an array of child nodes, inserting spaces\n * between adjacent inline elements where no whitespace separator exists.\n */\nfunction joinChildTexts(children: OrderedNode[]): string {\n const parts: string[] = [];\n for (const child of children) {\n const text = extractAllText(child);\n if (!text) continue;\n\n const tag = getTagName(child as OrderedNode);\n\n // If this is a space-after tag and there's previous content that doesn't\n // end with whitespace or punctuation, insert a space before this text.\n if (tag && SPACE_AFTER_TAGS.has(tag) && parts.length > 0) {\n const prev = parts[parts.length - 1]!;\n if (prev && !/[\\s,;.:()\\-/]$/.test(prev)) {\n parts.push(' ');\n }\n }\n\n parts.push(text);\n\n // If this is a space-after tag, check if a space is needed after.\n // We handle this by peeking: space will be inserted before the next\n // element if needed (handled above). But we also need to handle\n // the case where the next sibling is a text node starting without space.\n // That's already handled since text nodes include their own whitespace.\n }\n return parts.join('');\n}\n\n// ─── Metadata Parsing ────────────────────────────────────────────────\n\n/**\n * Parse JATS XML front matter to extract article metadata.\n */\nexport function parseJatsMetadata(xml: string): JatsMetadata {\n const parsed = parser.parse(xml) as OrderedNode[];\n const article = findArticle(parsed);\n if (!article) return { title: '', authors: [] };\n\n const front = findChild(article.children, 'front');\n if (!front) return { title: '', authors: [] };\n\n const articleMeta = findChild(front.children, 'article-meta');\n if (!articleMeta) return { title: '', authors: [] };\n\n const metaChildren = articleMeta.children;\n\n // Title\n const titleGroup = findChild(metaChildren, 'title-group');\n const articleTitle = titleGroup ? findChild(titleGroup.children, 'article-title') : undefined;\n const title = articleTitle ? extractAllText(articleTitle.children) : '';\n\n // Article IDs\n const articleIds = findChildren(metaChildren, 'article-id');\n let doi: string | undefined;\n let pmcid: string | undefined;\n let pmid: string | undefined;\n for (const idEntry of articleIds) {\n const idType = idEntry.attrs['pub-id-type'];\n const idText = extractAllText(idEntry.children);\n if (idType === 'doi') doi = idText;\n if (idType === 'pmc' || idType === 'pmcid') {\n pmcid = idText.replace(/^PMC/, '');\n }\n if (idType === 'pmid') pmid = idText;\n }\n\n // Authors\n const authors: JatsAuthor[] = [];\n const contribGroup = findChild(metaChildren, 'contrib-group');\n if (contribGroup) {\n const contribs = findChildren(contribGroup.children, 'contrib');\n for (const contrib of contribs) {\n if (contrib.attrs['contrib-type'] !== 'author') continue;\n const nameNode = findChild(contrib.children, 'name');\n if (!nameNode) continue;\n const surnameNode = findChild(nameNode.children, 'surname');\n const givenNamesNode = findChild(nameNode.children, 'given-names');\n const author: JatsAuthor = {\n surname: surnameNode ? extractAllText(surnameNode.children) : '',\n };\n const givenNames = givenNamesNode ? extractAllText(givenNamesNode.children) : '';\n if (givenNames) {\n author.givenNames = givenNames;\n }\n authors.push(author);\n }\n }\n\n // Abstract\n const abstractNode = findChild(metaChildren, 'abstract');\n let abstract: string | undefined;\n if (abstractNode) {\n // Structured abstract with <sec> elements\n const sections = findChildren(abstractNode.children, 'sec');\n if (sections.length > 0) {\n const parts: string[] = [];\n for (const sec of sections) {\n const secTitleNode = findChild(sec.children, 'title');\n const secTitle = secTitleNode ? extractAllText(secTitleNode.children) : '';\n const secPs = findChildren(sec.children, 'p');\n const text = secPs.map((p) => extractAllText(p.children)).join(' ');\n if (secTitle) {\n parts.push(`${secTitle}: ${text}`);\n } else {\n parts.push(text);\n }\n }\n abstract = parts.join('\\n\\n');\n } else {\n // Simple abstract with <p>\n const paragraphs = findChildren(abstractNode.children, 'p');\n if (paragraphs.length > 0) {\n abstract = paragraphs.map((p) => extractAllText(p.children)).join('\\n\\n');\n } else {\n const text = extractAllText(abstractNode.children);\n if (text) abstract = text;\n }\n }\n }\n\n // Publication date (from <article-meta>/<pub-date>)\n // Priority: epub > ppub > collection > any other\n const pubDates = findChildren(metaChildren, 'pub-date');\n let publicationDate: { year: string; month?: string; day?: string } | undefined;\n const datePriority: Record<string, number> = { epub: 0, ppub: 1, collection: 2 };\n let bestPriority = Infinity;\n for (const pd of pubDates) {\n // Support both pub-type (NLM/early JATS) and date-type (JATS 1.2+)\n const dateType = pd.attrs['pub-type'] ?? pd.attrs['date-type'] ?? '';\n const priority = datePriority[dateType] ?? 3;\n if (priority < bestPriority) {\n bestPriority = priority;\n const yearNode = findChild(pd.children, 'year');\n if (yearNode) {\n const year = extractAllText(yearNode.children);\n const monthNode = findChild(pd.children, 'month');\n const dayNode = findChild(pd.children, 'day');\n const date: { year: string; month?: string; day?: string } = { year };\n if (monthNode) date.month = extractAllText(monthNode.children);\n if (dayNode) date.day = extractAllText(dayNode.children);\n publicationDate = date;\n }\n }\n }\n // If no prioritized date found, take first available\n if (!publicationDate && pubDates.length > 0) {\n const pd = pubDates[0]!;\n const yearNode = findChild(pd.children, 'year');\n if (yearNode) {\n const year = extractAllText(yearNode.children);\n const monthNode = findChild(pd.children, 'month');\n const dayNode = findChild(pd.children, 'day');\n const date: { year: string; month?: string; day?: string } = { year };\n if (monthNode) date.month = extractAllText(monthNode.children);\n if (dayNode) date.day = extractAllText(dayNode.children);\n publicationDate = date;\n }\n }\n\n // Article type (from root <article> element attribute)\n const articleType = article.attrs['article-type'] || undefined;\n\n // License (from <permissions>/<license>)\n let license: string | undefined;\n const permissions = findChild(metaChildren, 'permissions');\n if (permissions) {\n const licenseNode = findChild(permissions.children, 'license');\n if (licenseNode) {\n // Prefer @xlink:href (standardized URL) over <license-p> (free-text)\n const href = licenseNode.attrs['xlink:href'];\n if (href) {\n license = href;\n } else {\n const licenseP = findChild(licenseNode.children, 'license-p');\n if (licenseP) license = extractAllText(licenseP.children).trim();\n }\n }\n }\n\n // Keywords (from all <kwd-group> elements)\n const kwdGroups = findChildren(metaChildren, 'kwd-group');\n const keywords: string[] = [];\n for (const kwdGroup of kwdGroups) {\n const kwds = findChildren(kwdGroup.children, 'kwd');\n for (const kwd of kwds) {\n const text = extractAllText(kwd.children).trim();\n if (text) keywords.push(text);\n }\n }\n\n // Volume, issue, pages\n const volumeNode = findChild(metaChildren, 'volume');\n const volume = volumeNode ? extractAllText(volumeNode.children) : undefined;\n const issueNode = findChild(metaChildren, 'issue');\n const issue = issueNode ? extractAllText(issueNode.children) : undefined;\n let pages: string | undefined;\n const fpageNode = findChild(metaChildren, 'fpage');\n const lpageNode = findChild(metaChildren, 'lpage');\n if (fpageNode) {\n const fp = extractAllText(fpageNode.children);\n const lp = lpageNode ? extractAllText(lpageNode.children) : '';\n pages = lp ? `${fp}-${lp}` : fp;\n } else {\n const elocationNode = findChild(metaChildren, 'elocation-id');\n if (elocationNode) pages = extractAllText(elocationNode.children);\n }\n\n // Journal name (from <front>/<journal-meta>)\n const journalMeta = findChild(front.children, 'journal-meta');\n let journal: string | undefined;\n if (journalMeta) {\n const titleGroup = findChild(journalMeta.children, 'journal-title-group');\n if (titleGroup) {\n const jTitle = findChild(titleGroup.children, 'journal-title');\n if (jTitle) journal = extractAllText(jTitle.children);\n }\n if (!journal) {\n const jTitle = findChild(journalMeta.children, 'journal-title');\n if (jTitle) journal = extractAllText(jTitle.children);\n }\n }\n\n const result: JatsMetadata = { title, authors };\n if (doi) result.doi = doi;\n if (pmcid) result.pmcid = pmcid;\n if (pmid) result.pmid = pmid;\n if (journal) result.journal = journal;\n if (publicationDate) result.publicationDate = publicationDate;\n if (volume) result.volume = volume;\n if (issue) result.issue = issue;\n if (pages) result.pages = pages;\n if (keywords.length > 0) result.keywords = keywords;\n if (articleType) result.articleType = articleType;\n if (license) result.license = license;\n if (abstract) result.abstract = abstract;\n return result;\n}\n\n// ─── Inline Content Parsing ──────────────────────────────────────────\n\n/**\n * Parse inline content from a paragraph's children array.\n * Iterates in document order to preserve interleaving of text, citations,\n * and formatting elements.\n */\nfunction parseInlineContent(children: OrderedNode[]): InlineContent[] {\n const result: InlineContent[] = [];\n\n for (const child of children) {\n // Text node\n const text = getTextContent(child);\n if (text != null) {\n if (text) result.push({ type: 'text', text });\n continue;\n }\n\n const tag = getTagName(child);\n if (!tag) continue;\n\n const innerChildren = getChildren(child);\n\n if (tag === 'bold') {\n result.push({ type: 'bold', children: parseInlineContent(innerChildren) });\n } else if (tag === 'italic') {\n result.push({ type: 'italic', children: parseInlineContent(innerChildren) });\n } else if (tag === 'sup') {\n result.push({ type: 'superscript', text: extractAllText(innerChildren) });\n } else if (tag === 'sub') {\n result.push({ type: 'subscript', text: extractAllText(innerChildren) });\n } else if (tag === 'inline-formula') {\n // Try to find <tex-math> directly or inside <alternatives>\n let texMath = findChild(innerChildren, 'tex-math');\n if (!texMath) {\n const alternatives = findChild(innerChildren, 'alternatives');\n if (alternatives) {\n texMath = findChild(alternatives.children, 'tex-math');\n }\n }\n const tex = texMath ? extractAllText(texMath.children) : undefined;\n const text = tex || extractAllText(innerChildren);\n const entry: { type: 'inline-formula'; tex?: string; text: string } = {\n type: 'inline-formula',\n text,\n };\n if (tex) entry.tex = tex;\n result.push(entry);\n } else if (tag === 'monospace') {\n result.push({ type: 'code', text: extractAllText(innerChildren) });\n } else if (tag === 'ext-link') {\n const href = getAttr(child, 'xlink:href');\n if (href) {\n result.push({ type: 'link', url: href, children: parseInlineContent(innerChildren) });\n } else {\n const linkText = extractAllText(innerChildren);\n if (linkText) result.push({ type: 'text', text: linkText });\n }\n } else if (tag === 'uri') {\n const href = getAttr(child, 'xlink:href');\n const textContent = extractAllText(innerChildren);\n const url = href || textContent;\n if (url) {\n result.push({ type: 'link', url, children: parseInlineContent(innerChildren) });\n }\n } else if (tag === 'underline' || tag === 'sc') {\n // Pass-through: preserve text content without special formatting\n const passText = extractAllText(innerChildren);\n if (passText) result.push({ type: 'text', text: passText });\n } else if (tag === 'xref') {\n const refType = getAttr(child, 'ref-type');\n if (refType === 'bibr') {\n result.push({\n type: 'citation',\n refId: getAttr(child, 'rid') ?? '',\n text: extractAllText(innerChildren),\n });\n } else {\n const xrefText = extractAllText(innerChildren);\n if (xrefText) result.push({ type: 'text', text: xrefText });\n }\n } else {\n // Unknown inline element — extract text\n const unknownText = extractAllText(innerChildren);\n if (unknownText) result.push({ type: 'text', text: unknownText });\n }\n }\n\n return result;\n}\n\n// ─── Block Content Parsing ───────────────────────────────────────────\n\n/**\n * Parse a <list> element into a BlockElement.\n */\nfunction parseList(listNode: OrderedNode): BlockElement {\n const listType = getAttr(listNode, 'list-type');\n const ordered = listType === 'order';\n const listChildren = getChildren(listNode);\n const listItems = findChildren(listChildren, 'list-item');\n const items: InlineContent[][] = [];\n\n for (const item of listItems) {\n const pNodes = findChildren(item.children, 'p');\n const content = pNodes.flatMap((p) => parseInlineContent(p.children));\n items.push(content);\n }\n\n return { type: 'list', ordered, items };\n}\n\n/**\n * Parse a table row into an array of cell text content.\n */\nfunction parseTableRow(trChildren: OrderedNode[]): string[] {\n const cells: string[] = [];\n for (const child of trChildren) {\n const tag = getTagName(child);\n if (tag === 'th' || tag === 'td') {\n const cellChildren = getChildren(child);\n // Check if cell contains multiple <p> elements\n const paragraphs = findChildren(cellChildren, 'p');\n if (paragraphs.length > 1) {\n cells.push(\n paragraphs.map((p) => extractAllText(p.children)).join('<br>'),\n );\n } else {\n cells.push(extractAllText(cellChildren));\n }\n }\n }\n return cells;\n}\n\n/**\n * Parse an already-parsed table-wrap node.\n */\nfunction parseTableWrap(tableWrapNode: OrderedNode): {\n caption?: string;\n headers: string[];\n rows: string[][];\n} {\n const children = getChildren(tableWrapNode);\n\n // Caption\n const labelNode = findChild(children, 'label');\n const label = labelNode ? extractAllText(labelNode.children) : '';\n const captionNode = findChild(children, 'caption');\n const captionText = captionNode ? extractAllText(captionNode.children) : '';\n const captionStr = [label, captionText].filter(Boolean).join('. ');\n\n const tableNode = findChild(children, 'table');\n const result: { caption?: string; headers: string[]; rows: string[][] } = {\n headers: [],\n rows: [],\n };\n if (captionStr) result.caption = captionStr;\n if (!tableNode) return result;\n\n // Headers from thead\n const thead = findChild(tableNode.children, 'thead');\n if (thead) {\n const headRows = findChildren(thead.children, 'tr');\n if (headRows.length > 0) {\n result.headers.push(...parseTableRow(headRows[0]!.children));\n }\n }\n\n // Body rows\n const tbody = findChild(tableNode.children, 'tbody');\n if (tbody) {\n const bodyRows = findChildren(tbody.children, 'tr');\n for (const row of bodyRows) {\n result.rows.push(parseTableRow(row.children));\n }\n }\n\n return result;\n}\n\n/**\n * Parse a <table-wrap> element into a table block.\n * Exported for standalone use and used internally by parseBlockContent.\n */\nexport function parseJatsTable(xml: string): {\n caption?: string;\n headers: string[];\n rows: string[][];\n} {\n const parsed = parser.parse(xml) as OrderedNode[];\n const tableWrap = findChild(parsed, 'table-wrap');\n if (tableWrap) {\n return parseTableWrap(tableWrap.node);\n }\n // Fallback: if not wrapped, try to find table directly\n return { headers: [], rows: [] };\n}\n\n/**\n * Parse a <boxed-text> element into a boxed-text block.\n * Extracts optional title and recursively parses inner block content.\n */\nfunction parseBoxedText(node: OrderedNode): BlockElement {\n const children = getChildren(node);\n const titleNode = findChild(children, 'title');\n const title = titleNode ? extractAllText(titleNode.children) : undefined;\n const content = parseBlockContent(children);\n const block: BlockElement = { type: 'boxed-text', content };\n if (title) block.title = title;\n return block;\n}\n\n/**\n * Parse a <def-list> element into a def-list block.\n * Extracts optional title and <def-item> pairs with <term> and <def>.\n */\nfunction parseDefList(node: OrderedNode): BlockElement {\n const children = getChildren(node);\n const titleNode = findChild(children, 'title');\n const title = titleNode ? extractAllText(titleNode.children) : undefined;\n const defItems = findChildren(children, 'def-item');\n const items: { term: string; definition: string }[] = [];\n for (const item of defItems) {\n const termNode = findChild(item.children, 'term');\n const defNode = findChild(item.children, 'def');\n const term = termNode ? extractAllText(termNode.children) : '';\n const definition = defNode ? extractAllText(defNode.children) : '';\n items.push({ term, definition });\n }\n const block: BlockElement = { type: 'def-list', items };\n if (title) block.title = title;\n return block;\n}\n\n/**\n * Parse a <disp-formula> element into a formula block.\n * Extracts TeX content from <tex-math> preferentially (inside <alternatives> or direct),\n * falls back to extractAllText for plain text.\n */\nfunction parseDispFormula(node: OrderedNode): BlockElement {\n const children = getChildren(node);\n const id = getAttr(node, 'id');\n const labelNode = findChild(children, 'label');\n const label = labelNode ? extractAllText(labelNode.children) : undefined;\n\n // Try <alternatives> wrapper first\n const alternatives = findChild(children, 'alternatives');\n const searchChildren = alternatives ? alternatives.children : children;\n\n const texMath = findChild(searchChildren, 'tex-math');\n const block: BlockElement = { type: 'formula' };\n if (id) block.id = id;\n if (label) block.label = label;\n\n if (texMath) {\n block.tex = extractAllText(texMath.children);\n } else {\n // Fall back to plain text extraction (skip label)\n const textChildren = children.filter((c) => !('label' in c));\n const text = extractAllText(textChildren).trim();\n if (text) block.text = text;\n }\n\n return block;\n}\n\n/** Tags that represent block-level elements when nested inside <p>. */\nconst BLOCK_TAGS = new Set(['table-wrap', 'fig', 'disp-quote', 'boxed-text']);\n\n/**\n * Parse a <disp-quote> element into a blockquote block.\n * Extracts <p> children and concatenates their inline content.\n */\nfunction parseDispQuote(node: OrderedNode): BlockElement {\n const children = getChildren(node);\n const paragraphs = findChildren(children, 'p');\n const content: InlineContent[] = [];\n for (let i = 0; i < paragraphs.length; i++) {\n if (i > 0) content.push({ type: 'text', text: '\\n\\n' });\n const para = paragraphs[i];\n if (para) content.push(...parseInlineContent(para.children));\n }\n // If no <p> children, extract inline content directly\n if (paragraphs.length === 0) {\n content.push(...parseInlineContent(children));\n }\n return { type: 'blockquote', content };\n}\n\n/**\n * Parse a <table-wrap> node into a table block element.\n */\nfunction parseTableBlock(node: OrderedNode): BlockElement {\n const tableResult = parseTableWrap(node);\n const tableBlock: BlockElement = {\n type: 'table',\n headers: tableResult.headers,\n rows: tableResult.rows,\n };\n if (tableResult.caption) tableBlock.caption = tableResult.caption;\n return tableBlock;\n}\n\n/**\n * Parse a <fig> node into a figure block element.\n */\nfunction parseFigBlock(node: OrderedNode): BlockElement {\n const innerChildren = getChildren(node);\n const figBlock: BlockElement = { type: 'figure' };\n const figId = getAttr(node, 'id');\n if (figId) figBlock.id = figId;\n const figLabel = findChild(innerChildren, 'label');\n if (figLabel) {\n const labelText = extractAllText(figLabel.children);\n if (labelText) figBlock.label = labelText;\n }\n const figCaption = findChild(innerChildren, 'caption');\n if (figCaption) {\n const captionText = extractAllText(figCaption.children);\n if (captionText) figBlock.caption = captionText;\n }\n return figBlock;\n}\n\n/**\n * Parse a <p> element, splitting it if it contains nested block elements\n * (table-wrap, fig, disp-quote). Returns one or more block elements.\n */\nfunction parseParagraph(pChildren: OrderedNode[]): BlockElement[] {\n // Check if <p> contains any nested block elements\n const hasNestedBlocks = pChildren.some((child) => {\n const tag = getTagName(child);\n return tag != null && BLOCK_TAGS.has(tag);\n });\n\n if (!hasNestedBlocks) {\n return [{ type: 'paragraph', content: parseInlineContent(pChildren) }];\n }\n\n // Split into inline runs and block elements\n const blocks: BlockElement[] = [];\n let inlineBuffer: OrderedNode[] = [];\n\n const flushInline = () => {\n if (inlineBuffer.length > 0) {\n const content = parseInlineContent(inlineBuffer);\n // Skip whitespace-only paragraphs created by XML formatting\n const hasNonWhitespace = content.some(\n (c) => c.type !== 'text' || c.text.trim() !== '',\n );\n if (content.length > 0 && hasNonWhitespace) {\n blocks.push({ type: 'paragraph', content });\n }\n inlineBuffer = [];\n }\n };\n\n for (const child of pChildren) {\n const tag = getTagName(child);\n if (tag === 'table-wrap') {\n flushInline();\n blocks.push(parseTableBlock(child));\n } else if (tag === 'fig') {\n flushInline();\n blocks.push(parseFigBlock(child));\n } else if (tag === 'disp-quote') {\n flushInline();\n blocks.push(parseDispQuote(child));\n } else if (tag === 'boxed-text') {\n flushInline();\n blocks.push(parseBoxedText(child));\n } else {\n inlineBuffer.push(child);\n }\n }\n flushInline();\n\n return blocks;\n}\n\n/**\n * Parse block-level content from a section's children.\n * Iterates in document order to preserve ordering of paragraphs, lists,\n * tables, figures, and blockquotes.\n */\nfunction parseBlockContent(sectionChildren: OrderedNode[]): BlockElement[] {\n const blocks: BlockElement[] = [];\n\n for (const child of sectionChildren) {\n const tag = getTagName(child);\n if (!tag) continue;\n\n if (tag === 'p') {\n blocks.push(...parseParagraph(getChildren(child)));\n } else if (tag === 'list') {\n blocks.push(parseList(child));\n } else if (tag === 'table-wrap') {\n blocks.push(parseTableBlock(child));\n } else if (tag === 'fig') {\n blocks.push(parseFigBlock(child));\n } else if (tag === 'disp-quote') {\n blocks.push(parseDispQuote(child));\n } else if (tag === 'boxed-text') {\n blocks.push(parseBoxedText(child));\n } else if (tag === 'def-list') {\n blocks.push(parseDefList(child));\n } else if (tag === 'disp-formula') {\n blocks.push(parseDispFormula(child));\n } else if (tag === 'preformat') {\n const text = extractAllText(getChildren(child));\n blocks.push({ type: 'preformat', text });\n } else if (tag === 'supplementary-material') {\n const innerChildren = getChildren(child);\n const labelNode = findChild(innerChildren, 'label');\n const captionNode = findChild(innerChildren, 'caption');\n const labelText = labelNode ? extractAllText(labelNode.children) : '';\n const captionText = captionNode ? extractAllText(captionNode.children) : '';\n const text = [labelText, captionText].filter(Boolean).join(': ');\n if (text) {\n blocks.push({ type: 'paragraph', content: [{ type: 'text', text }] });\n }\n }\n // Skip title, sec, and other non-block elements\n }\n\n return blocks;\n}\n\n// ─── Section Parsing ─────────────────────────────────────────────────\n\n/**\n * Parse a <sec> element into a JatsSection, recursively handling subsections.\n */\nfunction parseSection(secChildren: OrderedNode[], level: number): JatsSection {\n const titleNode = findChild(secChildren, 'title');\n const title = titleNode ? extractAllText(titleNode.children) : '';\n const content = parseBlockContent(secChildren);\n\n // Nested sections\n const subsections: JatsSection[] = [];\n const nestedSecs = findChildren(secChildren, 'sec');\n for (const sub of nestedSecs) {\n subsections.push(parseSection(sub.children, level + 1));\n }\n\n return { title, level, content, subsections };\n}\n\n/**\n * Parse JATS XML body to extract sections and content.\n */\nexport function parseJatsBody(xml: string): JatsSection[] {\n const parsed = parser.parse(xml) as OrderedNode[];\n const article = findArticle(parsed);\n if (!article) return [];\n\n const body = findChild(article.children, 'body');\n if (!body) return [];\n\n const sections: JatsSection[] = [];\n const secs = findChildren(body.children, 'sec');\n\n if (secs.length > 0) {\n for (const sec of secs) {\n sections.push(parseSection(sec.children, 2));\n }\n } else {\n // Body has paragraphs without sections\n const content = parseBlockContent(body.children);\n if (content.length > 0) {\n sections.push({ title: '', level: 2, content, subsections: [] });\n }\n }\n\n return sections;\n}\n\n// ─── Reference Parsing ───────────────────────────────────────────────\n\n/**\n * Format a structured <element-citation> into a readable reference string.\n * Produces: \"Author1, Author2. Title. Source. Year;Volume:FirstPage-LastPage.\"\n */\nfunction formatElementCitation(children: OrderedNode[]): string {\n const parts: string[] = [];\n\n // Authors from person-group\n const personGroup = findChild(children, 'person-group');\n if (personGroup) {\n const names = findChildren(personGroup.children, 'name');\n const authorParts: string[] = [];\n for (const name of names) {\n const surname = findChild(name.children, 'surname');\n const givenNames = findChild(name.children, 'given-names');\n const surnameText = surname ? extractAllText(surname.children) : '';\n const givenText = givenNames ? extractAllText(givenNames.children) : '';\n if (surnameText && givenText) {\n authorParts.push(`${surnameText} ${givenText}`);\n } else if (surnameText) {\n authorParts.push(surnameText);\n }\n }\n if (authorParts.length > 0) {\n parts.push(authorParts.join(', '));\n }\n }\n\n // Article title\n const articleTitle = findChild(children, 'article-title');\n if (articleTitle) {\n parts.push(extractAllText(articleTitle.children));\n }\n\n // Source (journal name)\n const source = findChild(children, 'source');\n if (source) {\n parts.push(extractAllText(source.children));\n }\n\n // Year, volume, pages\n const year = findChild(children, 'year');\n const volume = findChild(children, 'volume');\n const fpage = findChild(children, 'fpage');\n const lpage = findChild(children, 'lpage');\n\n if (year) {\n let yearStr = extractAllText(year.children);\n if (volume) {\n yearStr += `;${extractAllText(volume.children)}`;\n }\n if (fpage) {\n const fpageText = extractAllText(fpage.children);\n const lpageText = lpage ? extractAllText(lpage.children) : '';\n yearStr += `:${fpageText}${lpageText ? `-${lpageText}` : ''}`;\n }\n parts.push(yearStr);\n }\n\n return parts.join('. ') + '.';\n}\n\n/**\n * Extract text from a <mixed-citation>'s children, deduplicating any\n * <pub-id> content that also appears as inline text.\n *\n * Some publishers include the DOI/PMID both as a text node and inside\n * a <pub-id> element, causing duplication like \"10.1234/x 10.1234/x\".\n */\nfunction extractMixedCitationText(children: OrderedNode[]): string {\n // Collect pub-id values\n const pubIds = findChildren(children, 'pub-id');\n const pubIdValues = pubIds\n .map((p) => extractAllText(p.children).trim())\n .filter(Boolean);\n\n if (pubIdValues.length === 0) {\n return extractAllText(children).trim();\n }\n\n // Extract full text\n const fullText = extractAllText(children).trim();\n\n // For each pub-id value, if it appears more than once, remove extra occurrences\n let result = fullText;\n for (const val of pubIdValues) {\n // Escape regex special characters\n const escaped = val.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&');\n const matches = result.match(new RegExp(escaped, 'g'));\n if (matches && matches.length > 1) {\n // Remove the first occurrence (typically the inline text), keep the last (pub-id element)\n result = result.replace(val, '');\n // Clean up any leftover extra whitespace\n result = result.replace(/\\s{2,}/g, ' ').trim();\n }\n }\n\n return result;\n}\n\n/**\n * Extract structured pub-id values (DOI, PMID, PMCID) from children nodes.\n */\nfunction extractPubIds(children: OrderedNode[]): {\n doi?: string;\n pmid?: string;\n pmcid?: string;\n} {\n const pubIds = findChildren(children, 'pub-id');\n const result: { doi?: string; pmid?: string; pmcid?: string } = {};\n for (const p of pubIds) {\n const idType = p.attrs['pub-id-type'];\n const value = extractAllText(p.children).trim();\n if (!value) continue;\n if (idType === 'doi') result.doi = value;\n if (idType === 'pmid') result.pmid = value;\n if (idType === 'pmc' || idType === 'pmcid') {\n result.pmcid = value.replace(/^PMC/, '');\n }\n }\n return result;\n}\n\n/**\n * Strip extracted pub-id values from reference text to avoid duplication\n * when pub-ids are rendered separately as links.\n */\nfunction stripPubIdValues(\n text: string,\n pubIds: { doi?: string; pmid?: string; pmcid?: string },\n): string {\n let result = text;\n const values = [pubIds.doi, pubIds.pmid, pubIds.pmcid].filter(Boolean) as string[];\n for (const val of values) {\n // Strip common label prefixes (e.g. \"doi: \", \"PMID: \", \"DOI:\", \"pmid:\") followed by the value\n const escaped = val.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&');\n result = result.replace(new RegExp(`(?:doi|PMID|pmid|PMC|pmc)[:\\\\s]*${escaped}`, 'gi'), '');\n // Also strip the bare value itself\n result = result.replace(new RegExp(escaped, 'g'), '');\n }\n // Also strip PMC-prefixed form of pmcid\n if (pubIds.pmcid) {\n const pmcFull = `PMC${pubIds.pmcid}`;\n const escaped = pmcFull.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&');\n result = result.replace(new RegExp(`(?:pmc|pmcid)[:\\\\s]*${escaped}`, 'gi'), '');\n result = result.replace(new RegExp(escaped, 'g'), '');\n }\n // Clean up trailing/leading whitespace and extra spaces\n result = result.replace(/\\s{2,}/g, ' ').trim();\n // Clean up trailing period after stripped content (e.g. \"Title. .\" -> \"Title.\")\n result = result.replace(/\\.\\s*\\.$/, '.');\n return result;\n}\n\n/**\n * Parse JATS XML back matter to extract references.\n */\nexport function parseJatsReferences(xml: string): JatsReference[] {\n const parsed = parser.parse(xml) as OrderedNode[];\n const article = findArticle(parsed);\n if (!article) return [];\n\n const back = findChild(article.children, 'back');\n if (!back) return [];\n\n const refList = findChild(back.children, 'ref-list');\n if (!refList) return [];\n\n const refs = findChildren(refList.children, 'ref');\n const references: JatsReference[] = [];\n\n for (const ref of refs) {\n const id = ref.attrs['id'] ?? '';\n\n // Determine the search scope: if <citation-alternatives> exists, search within it;\n // otherwise search direct children of <ref>\n const citationAlternatives = findChild(ref.children, 'citation-alternatives');\n const searchChildren = citationAlternatives ? citationAlternatives.children : ref.children;\n\n // Try mixed-citation first (already formatted), then element-citation (structured)\n const mixedCitation = findChild(searchChildren, 'mixed-citation');\n if (mixedCitation) {\n const rawText = extractMixedCitationText(mixedCitation.children);\n const pubIds = extractPubIds(mixedCitation.children);\n const text = stripPubIdValues(rawText, pubIds);\n if (id && text) references.push({ id, text, ...pubIds });\n continue;\n }\n\n const elementCitation = findChild(searchChildren, 'element-citation');\n if (elementCitation) {\n const rawText = formatElementCitation(elementCitation.children);\n const pubIds = extractPubIds(elementCitation.children);\n const text = stripPubIdValues(rawText, pubIds);\n if (id && text) references.push({ id, text, ...pubIds });\n continue;\n }\n\n // Fallback: extract all text from ref, skipping <label>\n const childrenWithoutLabel = ref.children.filter((c) => !('label' in c));\n const text = extractAllText(childrenWithoutLabel).trim();\n if (id && text) {\n references.push({ id, text });\n }\n }\n\n return references;\n}\n\n// ─── Back Matter & Floats Parsing ────────────────────────────────────\n\n/** Result of parsing back matter and floats-group. */\nexport interface BackMatterResult {\n acknowledgments?: string;\n appendices?: JatsSection[];\n footnotes?: JatsFootnote[];\n floats?: BlockElement[];\n notes?: BackMatterNote[];\n}\n\n/**\n * Parse JATS XML back matter sections (ack, app-group, fn-group)\n * and top-level floats-group.\n */\nexport function parseJatsBackMatter(xml: string): BackMatterResult {\n const parsed = parser.parse(xml) as OrderedNode[];\n const article = findArticle(parsed);\n if (!article) return {};\n\n const result: BackMatterResult = {};\n\n // Parse <back> children\n const back = findChild(article.children, 'back');\n if (back) {\n // Acknowledgments: <ack>\n const ack = findChild(back.children, 'ack');\n if (ack) {\n const paragraphs = findChildren(ack.children, 'p');\n if (paragraphs.length > 0) {\n result.acknowledgments = paragraphs\n .map((p) => extractAllText(p.children))\n .join('\\n\\n');\n }\n }\n\n // Appendices: <app-group>/<app>\n const appGroup = findChild(back.children, 'app-group');\n if (appGroup) {\n const apps = findChildren(appGroup.children, 'app');\n if (apps.length > 0) {\n result.appendices = apps.map((app) => parseSection(app.children, 2));\n }\n }\n\n // Footnotes: <fn-group>/<fn>\n const fnGroup = findChild(back.children, 'fn-group');\n if (fnGroup) {\n const fns = findChildren(fnGroup.children, 'fn');\n if (fns.length > 0) {\n result.footnotes = fns.map((fn) => {\n const parts: string[] = [];\n // Include <title> if present\n const titleNode = findChild(fn.children, 'title');\n if (titleNode) {\n const titleText = extractAllText(titleNode.children).trim();\n if (titleText) parts.push(titleText);\n }\n // Extract text from each <p> separately and join with space\n const paragraphs = findChildren(fn.children, 'p');\n for (const p of paragraphs) {\n const pText = extractAllText(p.children).trim();\n if (pText) parts.push(pText);\n }\n return {\n id: fn.attrs['id'] ?? '',\n text: parts.join(' '),\n };\n });\n }\n }\n\n // Notes: <notes> (author contributions, funding, data availability, etc.)\n const notesElements = findChildren(back.children, 'notes');\n if (notesElements.length > 0) {\n const notes: BackMatterNote[] = [];\n for (const note of notesElements) {\n // Check if this <notes> contains <sec> or nested <notes> children (e.g. Declarations wrapper)\n const secs = findChildren(note.children, 'sec');\n const nestedNotes = findChildren(note.children, 'notes');\n const subItems = secs.length > 0 ? secs : nestedNotes;\n if (subItems.length > 0) {\n for (const sub of subItems) {\n const subTitleNode = findChild(sub.children, 'title');\n const subTitle = subTitleNode ? extractAllText(subTitleNode.children) : '';\n const subParagraphs = findChildren(sub.children, 'p');\n const subText = subParagraphs\n .map((p) => extractAllText(p.children))\n .join('\\n\\n');\n if (subTitle || subText) {\n notes.push({ title: subTitle, text: subText });\n }\n }\n } else {\n const titleNode = findChild(note.children, 'title');\n const title = titleNode ? extractAllText(titleNode.children) : '';\n const paragraphs = findChildren(note.children, 'p');\n const text = paragraphs\n .map((p) => extractAllText(p.children))\n .join('\\n\\n');\n if (title || text) {\n notes.push({ title, text });\n }\n }\n }\n if (notes.length > 0) {\n result.notes = notes;\n }\n }\n\n // Glossary: <glossary> (abbreviations, definitions)\n const glossaryElements = findChildren(back.children, 'glossary');\n for (const glossary of glossaryElements) {\n const titleNode = findChild(glossary.children, 'title');\n const title = titleNode ? extractAllText(titleNode.children) : 'Glossary';\n const defList = findChild(glossary.children, 'def-list');\n if (defList) {\n const defItems = findChildren(defList.children, 'def-item');\n const lines: string[] = [];\n for (const item of defItems) {\n const termNode = findChild(item.children, 'term');\n const defNode = findChild(item.children, 'def');\n const term = termNode ? extractAllText(termNode.children) : '';\n const definition = defNode ? extractAllText(defNode.children) : '';\n lines.push(`${term}: ${definition}`);\n }\n if (!result.notes) result.notes = [];\n result.notes.push({ title, text: lines.join('\\n') });\n }\n }\n }\n\n // Floats-group: <floats-group> (sibling of <body> and <back>)\n const floatsGroup = findChild(article.children, 'floats-group');\n if (floatsGroup) {\n const blocks: BlockElement[] = [];\n for (const child of floatsGroup.children) {\n const tag = getTagName(child);\n if (tag === 'fig') {\n blocks.push(parseFigBlock(child));\n } else if (tag === 'table-wrap') {\n blocks.push(parseTableBlock(child));\n }\n }\n if (blocks.length > 0) {\n result.floats = blocks;\n }\n }\n\n return result;\n}\n"],"names":["titleGroup","text"],"mappings":"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"}
|