@prose-reader/archive-parser 1.303.0 → 1.304.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/dist/comicInfo/manga.d.ts +0 -5
- package/dist/comicInfo/parse.d.ts +0 -16
- package/dist/index.d.ts +0 -5
- package/dist/index.js +313 -351
- package/dist/index.js.map +1 -1
- package/dist/index.umd.cjs +2 -2
- package/dist/index.umd.cjs.map +1 -1
- package/dist/kobo/parse.d.ts +0 -8
- package/dist/opf/parse.d.ts +0 -29
- package/dist/opf/spineItemrefProperties.d.ts +0 -5
- package/dist/types/archiveResolve.d.ts +0 -49
- package/dist/utils/normalizeGtin.d.ts +0 -4
- package/dist/utils/normalizeIsbn.d.ts +0 -10
- package/dist/utils/parseW3cDtfDate.d.ts +0 -15
- package/dist/utils/tokenizeXmlSpaceSeparatedList.d.ts +0 -4
- package/package.json +2 -2
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","sources":["../src/apple/parse.ts","../src/comicInfo/manga.ts","../src/comicInfo/parse.ts","../src/kobo/parse.ts","../src/utils/tokenizeXmlSpaceSeparatedList.ts","../src/opf/spineItemrefProperties.ts","../src/opf/parse.ts","../src/apple/resolve.ts","../src/utils/normalizeGtin.ts","../src/utils/normalizeIsbn.ts","../src/comicInfo/resolve.ts","../src/kobo/resolve.ts","../src/utils/parseW3cDtfDate.ts","../src/opf/resolve.ts","../src/resolve.ts"],"sourcesContent":["import type { XmlElement } from \"xmldoc\"\nimport { XmlDocument } from \"xmldoc\"\n\nexport const APPLE_IBOOKS_DISPLAY_OPTIONS_FILENAME =\n \"com.apple.ibooks.display-options.xml\"\n\nexport type AppleDisplayOption = {\n readonly name?: string\n readonly value: string\n}\n\nexport type AppleMetadata = {\n readonly kind: \"apple\"\n readonly displayOptions?: {\n readonly platform?: {\n readonly options: ReadonlyArray<AppleDisplayOption>\n }\n }\n}\n\nconst platformOptionsFromElement = (\n platform: XmlElement,\n): ReadonlyArray<AppleDisplayOption> =>\n platform.childrenNamed(\"option\").map((option) => ({\n name: option.attr?.name,\n value: option.val,\n }))\n\nexport const parseAppleDisplayOptionsXml = (xml: string): AppleMetadata => {\n const doc = new XmlDocument(xml)\n const root = doc.name?.toLowerCase()\n\n if (root !== \"display_options\") {\n return { kind: \"apple\" }\n }\n\n const platformEl = doc.childNamed(\"platform\")\n\n return {\n kind: \"apple\",\n displayOptions: {\n ...(platformEl !== undefined\n ? { platform: { options: platformOptionsFromElement(platformEl) } }\n : {}),\n },\n }\n}\n","/**\n * Allowed `Manga` element values (ComicInfo 2.0 `Manga` simpleType).\n *\n * @see https://anansi-project.github.io/docs/comicinfo/documentation#manga\n */\nexport const COMIC_INFO_MANGA_VALUES = [\n \"Unknown\",\n \"No\",\n \"Yes\",\n \"YesAndRightToLeft\",\n] as const\n\nexport type ComicInfoManga = (typeof COMIC_INFO_MANGA_VALUES)[number]\n\nexport const isComicInfoManga = (value: string): value is ComicInfoManga => {\n for (const v of COMIC_INFO_MANGA_VALUES) {\n if (v === value) return true\n }\n return false\n}\n","import type { XmlElement } from \"xmldoc\"\nimport { XmlDocument } from \"xmldoc\"\nimport type { ComicInfoManga } from \"./manga\"\n\n/** Canonical top-level filename; real archives may use any casing. */\nexport const COMIC_INFO_FILENAME = \"ComicInfo.xml\"\n\n/**\n * Parsed `ComicInfo.xml` root: one optional string property per child element,\n * using the same names as in the file (e.g. `Title`, `GTIN`, `LanguageISO`).\n * Nested blocks such as `Pages` are skipped. Other simple child elements are\n * still copied onto this object under their tag name.\n *\n * @see https://anansi-project.github.io/docs/comicinfo/intro\n * @see https://github.com/anansi-project/comicinfo/blob/main/drafts/v2.1/ComicInfo.xsd for schema\n */\nexport interface ComicInfo {\n readonly kind: \"comicInfo\"\n AgeRating?: string\n AlternateCount?: string\n AlternateNumber?: string\n AlternateSeries?: string\n BlackAndWhite?: string\n Characters?: string\n Colorist?: string\n CommunityRating?: string\n Count?: string\n CoverArtist?: string\n Day?: string\n Editor?: string\n Format?: string\n Genre?: string\n GTIN?: string\n Imprint?: string\n Inker?: string\n LanguageISO?: string\n Letterer?: string\n Locations?: string\n MainCharacterOrTeam?: string\n /** Schema literals per {@link ComicInfoManga}; files may still use other strings. */\n Manga?: ComicInfoManga | (string & {})\n Month?: string\n Notes?: string\n Number?: string\n PageCount?: string\n Penciller?: string\n Publisher?: string\n Review?: string\n ScanInformation?: string\n Series?: string\n SeriesGroup?: string\n StoryArc?: string\n StoryArcNumber?: string\n Summary?: string\n Tags?: string\n Teams?: string\n Title?: string\n Translator?: string\n Volume?: string\n Web?: string\n Writer?: string\n Year?: string\n [tag: string]: string | undefined\n}\n\nconst SKIP_ELEMENT_CHILDREN = new Set([\"Pages\", \"ComicInfo\"])\n\nconst hasNestedElement = (el: XmlElement) =>\n el.children.some((c) => c.type === \"element\")\n\nconst trimmedText = (el: XmlElement): string | undefined => {\n const t = el.val.trim()\n return t.length > 0 ? t : undefined\n}\n\n/**\n * Parse a raw `ComicInfo.xml` body. Each direct child element with plain text\n * becomes a property named after that tag. Malformed XML throws; the parser\n * error is attached as `cause`.\n */\nexport const parseComicInfo = (xml: string): ComicInfo => {\n let doc: XmlDocument\n try {\n doc = new XmlDocument(xml)\n } catch (cause) {\n const message = cause instanceof Error ? cause.message : String(cause)\n throw new Error(`${COMIC_INFO_FILENAME} is malformed: ${message}`, {\n cause,\n })\n }\n\n const fields: Record<string, string> = {}\n\n doc.eachChild((child) => {\n if (child.type !== \"element\") return\n if (SKIP_ELEMENT_CHILDREN.has(child.name)) return\n if (hasNestedElement(child)) return\n\n const text = trimmedText(child)\n if (text === undefined) return\n if (fields[child.name] !== undefined) return\n\n fields[child.name] = text\n })\n\n // `as ComicInfo`: TS cannot infer that spreading `Record<string, string>` into `{ kind }` satisfies the named optional fields plus `[tag: string]: string | undefined` on `ComicInfo`.\n return { kind: \"comicInfo\", ...fields } as ComicInfo\n}\n","import { XmlDocument } from \"xmldoc\"\n\nexport const KOBO_DISPLAY_OPTIONS_FILENAME = \"com.kobobooks.display-options.xml\"\n\n/**\n * Normalized fields from Kobo-specific XML sidecars. Grows as new Kobo\n * document kinds are supported; absent keys mean not present or unknown.\n */\nexport type KoboMetadata = {\n readonly kind: \"kobo\"\n renditionLayout?: `reflowable` | `pre-paginated`\n}\n\nconst parseKoboDisplayOptionsDocument = (\n doc: XmlDocument,\n): { renditionLayout?: \"pre-paginated\" } => {\n const platform = doc.childNamed(\"platform\")\n if (!platform) return {}\n\n for (const option of platform.childrenNamed(\"option\")) {\n if (option.attr?.name !== \"fixed-layout\") continue\n if (option.val.trim().toLowerCase() === \"true\") {\n return { renditionLayout: \"pre-paginated\" }\n }\n return {}\n }\n\n return {}\n}\n\n/**\n * Parse a Kobo-related XML document. Unsupported or unrecognized roots\n * yield an empty object. Malformed XML throws from the underlying parser.\n */\nexport const parseKoboXml = (xml: string): KoboMetadata => {\n const doc = new XmlDocument(xml)\n const root = doc.name?.toLowerCase()\n\n if (root === \"display_options\") {\n return { kind: \"kobo\", ...parseKoboDisplayOptionsDocument(doc) }\n }\n\n return { kind: \"kobo\" }\n}\n","/**\n * EPUB/XML space-separated token lists (e.g. `properties` on `item` / `itemref`).\n * Trims the raw string, splits on ASCII whitespace, drops empty segments.\n */\nexport const tokenizeXmlSpaceSeparatedList = (\n raw: string | undefined,\n): readonly string[] => {\n if (raw === undefined || raw.trim().length === 0) {\n return []\n }\n\n return raw\n .trim()\n .split(/\\s+/)\n .filter((t) => t.length > 0)\n}\n","import { tokenizeXmlSpaceSeparatedList } from \"../utils/tokenizeXmlSpaceSeparatedList\"\n\nexport type OpfItemrefLayoutHints = {\n readonly renditionLayout?: `reflowable` | `pre-paginated`\n readonly renditionFlow?:\n | `scrolled-continuous`\n | `scrolled-doc`\n | `paginated`\n | `auto`\n readonly pageSpreadLeft?: true\n readonly pageSpreadRight?: true\n}\n\n/**\n * EPUB `itemref` `properties` attribute (space-separated tokens).\n *\n * @see https://www.w3.org/TR/epub/#attrdef-properties\n */\nexport const layoutHintsFromItemrefProperties = (\n properties: string | undefined,\n): OpfItemrefLayoutHints => {\n const tokens = tokenizeXmlSpaceSeparatedList(properties)\n if (tokens.length === 0) {\n return {}\n }\n\n let renditionLayout: `reflowable` | `pre-paginated` | undefined\n if (tokens.includes(`rendition:layout-reflowable`)) {\n renditionLayout = `reflowable`\n }\n if (tokens.includes(`rendition:layout-pre-paginated`)) {\n renditionLayout = `pre-paginated`\n }\n\n let renditionFlow: OpfItemrefLayoutHints[\"renditionFlow\"]\n if (tokens.includes(`rendition:flow-auto`)) {\n renditionFlow = `auto`\n }\n if (tokens.includes(`rendition:flow-paginated`)) {\n renditionFlow = `paginated`\n }\n if (tokens.includes(`rendition:flow-scrolled-doc`)) {\n renditionFlow = `scrolled-doc`\n }\n if (tokens.includes(`rendition:flow-scrolled-continuous`)) {\n renditionFlow = `scrolled-continuous`\n }\n\n return {\n ...(renditionLayout !== undefined ? { renditionLayout } : {}),\n ...(renditionFlow !== undefined ? { renditionFlow } : {}),\n ...(tokens.includes(`page-spread-left`)\n ? { pageSpreadLeft: true as const }\n : {}),\n ...(tokens.includes(`page-spread-right`)\n ? { pageSpreadRight: true as const }\n : {}),\n }\n}\n","import type { XmlElement, XmlNodeBase } from \"xmldoc\"\nimport { XmlDocument } from \"xmldoc\"\nimport { tokenizeXmlSpaceSeparatedList } from \"../utils/tokenizeXmlSpaceSeparatedList\"\nimport { layoutHintsFromItemrefProperties } from \"./spineItemrefProperties\"\n\nexport type OpfSpineManifestItem = {\n readonly id: string\n readonly href: string\n readonly mediaType?: string\n readonly properties?: string\n}\n\nexport type OpfIdentifier = {\n readonly value: string\n readonly scheme?: string\n}\n\nexport type OpfSpineRow = {\n readonly idref: string\n readonly id: string\n readonly href: string\n readonly mediaType?: string\n readonly properties?: string\n readonly renditionLayout?: `reflowable` | `pre-paginated`\n readonly renditionFlow?:\n | `scrolled-continuous`\n | `scrolled-doc`\n | `paginated`\n | `auto`\n readonly pageSpreadLeft?: true\n readonly pageSpreadRight?: true\n}\n\nexport type OpfGuideReference = {\n readonly href: string\n readonly title: string\n readonly type: string\n}\n\nconst elementLocalName = (name: string): string =>\n name.includes(\":\") ? name.slice(name.lastIndexOf(\":\") + 1) : name\n\nconst localNameEq = (elementName: string, wantLocal: string): boolean =>\n elementLocalName(elementName).toLowerCase() === wantLocal.toLowerCase()\n\nconst isXmlElement = (node: XmlNodeBase): node is XmlElement =>\n node.type === \"element\"\n\nconst childNamedLocal = (\n parent: XmlElement,\n localName: string,\n): XmlElement | undefined => {\n for (const node of parent.children) {\n if (!isXmlElement(node)) continue\n if (localNameEq(node.name, localName)) return node\n }\n return undefined\n}\n\nconst childrenNamedLocal = (\n parent: XmlElement,\n localName: string,\n): XmlElement[] => {\n const out: XmlElement[] = []\n for (const node of parent.children) {\n if (!isXmlElement(node)) continue\n if (localNameEq(node.name, localName)) out.push(node)\n }\n return out\n}\n\nconst identifiersFromMetadata = (metadataEl: XmlElement): OpfIdentifier[] => {\n const identifiers: OpfIdentifier[] = []\n\n metadataEl.eachChild((child) => {\n if (elementLocalName(child.name).toLowerCase() !== \"identifier\") return\n\n const value = child.val.trim()\n if (value.length === 0) return\n\n const scheme =\n child.attr[\"opf:scheme\"] ?? child.attr[\"opf:Scheme\"] ?? child.attr.scheme\n const schemeTrimmed = scheme?.trim()\n\n identifiers.push({\n value,\n ...(schemeTrimmed !== undefined && schemeTrimmed.length > 0\n ? { scheme: schemeTrimmed }\n : {}),\n })\n })\n\n return identifiers\n}\n\nconst firstTextByLocalName = (\n metadataEl: XmlElement,\n localName: string,\n): string | undefined => {\n let found: string | undefined\n metadataEl.eachChild((child) => {\n if (found !== undefined) return\n if (elementLocalName(child.name).toLowerCase() !== localName.toLowerCase())\n return\n const t = child.val.trim()\n if (t.length > 0) found = t\n })\n return found\n}\n\nconst textsByLocalName = (\n metadataEl: XmlElement,\n localName: string,\n): string[] => {\n const out: string[] = []\n metadataEl.eachChild((child) => {\n if (elementLocalName(child.name).toLowerCase() !== localName.toLowerCase())\n return\n const t = child.val.trim()\n if (t.length > 0) out.push(t)\n })\n return out\n}\n\nconst coverContentIdFromMetadata = (\n metadataEl: XmlElement,\n): string | undefined => {\n let coverId: string | undefined\n metadataEl.eachChild((child) => {\n if (coverId !== undefined) return\n if (elementLocalName(child.name).toLowerCase() !== \"meta\") return\n if (child.attr.name?.toLowerCase() !== \"cover\") return\n const content = child.attr.content?.trim()\n if (content !== undefined && content.length > 0) coverId = content\n })\n return coverId\n}\n\n/**\n * EPUB cover image inside the manifest. Resolution order, matching\n * what the spec lays out and what the bulk of EPUB producers in the\n * wild rely on:\n *\n * 1. EPUB 3 — the manifest item carrying the `cover-image` token in\n * its `properties` attribute (§ D.6.1).\n * 2. EPUB 2 — `<meta name=\"cover\" content=\"ID\"/>` in `metadata`,\n * resolved to the manifest item with that `id`.\n * 3. Last-resort fallback — any image manifest item whose `id`\n * contains the substring `cover` (case-insensitive); covers the\n * long tail of producers that emit neither the EPUB 3 property\n * nor the EPUB 2 meta.\n *\n * Each step requires the candidate manifest item to advertise an\n * `image/*` media type so non-image artefacts named `cover` (XHTML\n * cover pages, NCX entries) don't slip through.\n */\nconst coverHrefFromManifestAndMetadata = ({\n manifestItems,\n metadataEl,\n}: {\n manifestItems: ReadonlyArray<OpfSpineManifestItem>\n metadataEl: XmlElement | undefined\n}): string | undefined => {\n const isImage = (item: OpfSpineManifestItem): boolean =>\n item.mediaType?.toLowerCase().includes(\"image/\") === true\n\n const byCoverImageProperty = manifestItems.find((item) => {\n if (!isImage(item)) return false\n return tokenizeXmlSpaceSeparatedList(item.properties).includes(\n \"cover-image\",\n )\n })\n if (byCoverImageProperty !== undefined) return byCoverImageProperty.href\n\n if (metadataEl !== undefined) {\n const coverContentId = coverContentIdFromMetadata(metadataEl)\n if (coverContentId !== undefined) {\n const match = manifestItems.find(\n (item) => item.id === coverContentId && isImage(item),\n )\n if (match !== undefined) return match.href\n }\n }\n\n return manifestItems.find(\n (item) => item.id.toLowerCase().includes(\"cover\") && isImage(item),\n )?.href\n}\n\nconst metaValByProperty = (\n metadataEl: XmlElement,\n property: string,\n): string | undefined => {\n const meta = childrenNamedLocal(metadataEl, \"meta\").find(\n (m) => m.attr.property === property,\n )\n const raw = meta?.val\n if (raw === undefined || raw.trim().length === 0) return undefined\n return raw\n}\n\nconst guideFromPackage = (doc: XmlElement): OpfGuideReference[] => {\n const guideEl = childNamedLocal(doc, \"guide\")\n if (guideEl === undefined) return []\n\n const refs: OpfGuideReference[] = []\n\n for (const ref of childrenNamedLocal(guideEl, \"reference\")) {\n const href = ref.attr.href?.trim()\n if (href === undefined || href.length === 0) continue\n refs.push({\n href,\n title: ref.attr.title?.trim() ?? ``,\n type: ref.attr.type?.trim() ?? ``,\n })\n }\n\n return refs\n}\n\nconst manifestItemFromXmlElement = (\n item: XmlElement,\n): OpfSpineManifestItem | undefined => {\n const id = item.attr.id\n const href = item.attr.href\n if (id === undefined || id.length === 0) return undefined\n if (href === undefined || href.length === 0) return undefined\n\n const mediaType = item.attr[\"media-type\"]\n const properties = item.attr.properties?.trim()\n return {\n id,\n href,\n ...(mediaType !== undefined && mediaType.length > 0 ? { mediaType } : {}),\n ...(properties !== undefined && properties.length > 0\n ? { properties }\n : {}),\n }\n}\n\nconst manifestItemsAndById = (\n manifestEl: XmlElement,\n): {\n items: OpfSpineManifestItem[]\n byId: Map<string, OpfSpineManifestItem>\n} => {\n const items: OpfSpineManifestItem[] = []\n const byId = new Map<string, OpfSpineManifestItem>()\n\n for (const el of childrenNamedLocal(manifestEl, \"item\")) {\n const parsed = manifestItemFromXmlElement(el)\n if (parsed === undefined) continue\n items.push(parsed)\n byId.set(parsed.id, parsed)\n }\n\n return { items, byId }\n}\n\nconst spineRowsFromByIdAndSpine = (\n byId: Map<string, OpfSpineManifestItem>,\n spineEl: XmlElement,\n): OpfSpineRow[] => {\n const rows: OpfSpineRow[] = []\n\n for (const itemref of childrenNamedLocal(spineEl, \"itemref\")) {\n const idref = itemref.attr.idref\n if (idref === undefined || idref.trim().length === 0) continue\n\n const manifestItem = byId.get(idref)\n if (manifestItem === undefined) continue\n\n const hints = layoutHintsFromItemrefProperties(itemref.attr.properties)\n\n rows.push({\n idref,\n id: manifestItem.id,\n href: manifestItem.href,\n ...(manifestItem.mediaType !== undefined\n ? { mediaType: manifestItem.mediaType }\n : {}),\n ...(manifestItem.properties !== undefined\n ? { properties: manifestItem.properties }\n : {}),\n ...(hints.renditionLayout !== undefined\n ? { renditionLayout: hints.renditionLayout }\n : {}),\n ...(hints.renditionFlow !== undefined\n ? { renditionFlow: hints.renditionFlow }\n : {}),\n ...(hints.pageSpreadLeft !== undefined\n ? { pageSpreadLeft: hints.pageSpreadLeft }\n : {}),\n ...(hints.pageSpreadRight !== undefined\n ? { pageSpreadRight: hints.pageSpreadRight }\n : {}),\n })\n }\n\n return rows\n}\n\nexport type OpfMetadata = {\n readonly kind: \"opf\"\n readonly manifestItems: ReadonlyArray<OpfSpineManifestItem>\n readonly spineRows: ReadonlyArray<OpfSpineRow>\n readonly spineTocIdref: string | undefined\n readonly identifiers: ReadonlyArray<OpfIdentifier>\n readonly title: string | undefined\n /** `dc:creator` values, in document order, trimmed; empty when none. */\n readonly creators: ReadonlyArray<string>\n /** First non-empty `dc:publisher`, trimmed. */\n readonly publisher: string | undefined\n /** First non-empty `dc:rights`, trimmed. */\n readonly rights: string | undefined\n /** `dc:language` values, in document order, trimmed; empty when none. */\n readonly languages: ReadonlyArray<string>\n /** `dc:subject` values, in document order, trimmed; empty when none. */\n readonly subjects: ReadonlyArray<string>\n /**\n * Raw `dc:date` value as authored. EPUB 3 requires W3CDTF (a profile\n * of ISO 8601), but real-world publishers also ship free text here,\n * so the value is exposed verbatim and consumers normalize as needed.\n */\n readonly date: string | undefined\n /**\n * Manifest-relative `href` of the cover image, when one can be\n * resolved from `cover-image` properties (EPUB 3), the EPUB 2\n * `<meta name=\"cover\">` convention, or an `id` that contains\n * `cover` on an image manifest item. The href is returned exactly\n * as it appears in the manifest — callers own folder-prefix\n * resolution against the OPF's location in the archive.\n */\n readonly coverHref: string | undefined\n readonly renditionLayoutMeta: string | undefined\n readonly renditionFlowMeta: string | undefined\n readonly renditionSpreadMeta: string | undefined\n readonly pageProgressionDirection: string | undefined\n readonly guide: ReadonlyArray<OpfGuideReference>\n}\n\n/**\n * Parses an EPUB package document (OPF) into structured metadata.\n *\n * Direct children of `package` (`metadata`, `manifest`, `spine`, `guide`) and\n * their structural children (`item`, `itemref`, `reference`, `meta`) are\n * matched by **local name** (ASCII case-insensitive), so prefixed tags such as\n * `opf:manifest` are supported the same as unprefixed `manifest`.\n *\n * Attribute names on `spine` / `itemref` are still read as emitted by xmldoc\n * (no QName normalization).\n */\nexport const parseOpf = (opfXml: string): OpfMetadata => {\n const doc = new XmlDocument(opfXml)\n const manifestEl = childNamedLocal(doc, \"manifest\")\n const spineEl = childNamedLocal(doc, \"spine\")\n const metadataEl = childNamedLocal(doc, \"metadata\")\n\n let manifestItems: OpfSpineManifestItem[] = []\n let spineRows: OpfSpineRow[] = []\n\n if (manifestEl !== undefined) {\n const { items, byId } = manifestItemsAndById(manifestEl)\n manifestItems = items\n if (spineEl !== undefined) {\n spineRows = spineRowsFromByIdAndSpine(byId, spineEl)\n }\n }\n\n const pageProgressionDirectionRaw =\n spineEl?.attr[\"page-progression-direction\"]\n const pageProgressionDirection =\n pageProgressionDirectionRaw !== undefined &&\n pageProgressionDirectionRaw.trim().length > 0\n ? pageProgressionDirectionRaw\n : undefined\n\n const spineTocRaw = spineEl?.attr.toc\n const spineTocIdref =\n spineTocRaw !== undefined && spineTocRaw.trim().length > 0\n ? spineTocRaw.trim()\n : undefined\n\n let title: string | undefined\n let publisher: string | undefined\n let rights: string | undefined\n let date: string | undefined\n let creators: string[] = []\n let languages: string[] = []\n let subjects: string[] = []\n let renditionLayoutMeta: string | undefined\n let renditionFlowMeta: string | undefined\n let renditionSpreadMeta: string | undefined\n const identifiers: OpfIdentifier[] = []\n\n if (metadataEl !== undefined) {\n title = firstTextByLocalName(metadataEl, \"title\")\n publisher = firstTextByLocalName(metadataEl, \"publisher\")\n rights = firstTextByLocalName(metadataEl, \"rights\")\n date = firstTextByLocalName(metadataEl, \"date\")\n creators = textsByLocalName(metadataEl, \"creator\")\n languages = textsByLocalName(metadataEl, \"language\")\n subjects = textsByLocalName(metadataEl, \"subject\")\n renditionLayoutMeta = metaValByProperty(metadataEl, \"rendition:layout\")\n renditionFlowMeta = metaValByProperty(metadataEl, \"rendition:flow\")\n renditionSpreadMeta = metaValByProperty(metadataEl, \"rendition:spread\")\n identifiers.push(...identifiersFromMetadata(metadataEl))\n }\n\n const coverHref = coverHrefFromManifestAndMetadata({\n manifestItems,\n metadataEl,\n })\n\n const guide = guideFromPackage(doc)\n\n return {\n kind: \"opf\",\n manifestItems,\n spineRows,\n spineTocIdref,\n identifiers,\n title,\n creators,\n publisher,\n rights,\n languages,\n subjects,\n date,\n coverHref,\n renditionLayoutMeta,\n renditionFlowMeta,\n renditionSpreadMeta,\n pageProgressionDirection,\n guide,\n }\n}\n","import type { ArchiveResolveResult } from \"../types/archiveResolve\"\nimport type { AppleMetadata } from \"./parse\"\n\nexport const resolveApple = (input: AppleMetadata): ArchiveResolveResult => {\n const fixedLayout = input.displayOptions?.platform?.options?.find(\n (o) => o.name === \"fixed-layout\",\n )?.value\n\n const renditionLayout =\n fixedLayout?.trim().toLowerCase() === \"true\" ? \"pre-paginated\" : undefined\n\n return {\n gtin: undefined,\n isbn: undefined,\n readingDirection: undefined,\n renditionLayout,\n title: undefined,\n authors: undefined,\n publisher: undefined,\n rights: undefined,\n languages: undefined,\n date: undefined,\n subjects: undefined,\n }\n}\n","const GTIN_LENGTHS = new Set([8, 12, 13, 14])\n\n/**\n * Normalize a raw GTIN / EAN / UPC string to digits only when the length\n * matches a GS1 GTIN family size (8, 12, 13, or 14). Does not verify check digits.\n */\nexport const normalizeGtin = (\n raw: string | number | undefined | null,\n): string | undefined => {\n if (raw === undefined || raw === null) return undefined\n\n const digits = String(raw).replace(/\\D/g, \"\")\n if (digits.length === 0 || !GTIN_LENGTHS.has(digits.length)) return undefined\n\n return digits\n}\n","const ISBN_CANDIDATE_PATTERN = /(?:97[89])?\\d{9}[\\dXx]/\n\n/**\n * Normalize a raw ISBN-ish string into a canonical 10- or 13-character\n * form, or `undefined` when no recognisable ISBN can be recovered.\n *\n * - Strips the common `urn:isbn:` / `isbn:` prefixes.\n * - Drops everything that isn't a digit or `X`.\n * - Validates the resulting length (10 or 13).\n * - Falls back to a lax regex scan so publishers that stuff free text\n * around the number still yield a usable value.\n */\nexport const normalizeIsbn = (\n raw: string | number | undefined | null,\n): string | undefined => {\n if (raw === undefined || raw === null) return undefined\n\n const stripped = String(raw)\n .trim()\n .replace(/^urn:isbn:/i, \"\")\n .replace(/^isbn[:\\s-]*/i, \"\")\n\n const digitsOnly = stripped.replace(/[^0-9Xx]/g, \"\")\n\n if (digitsOnly.length === 10 || digitsOnly.length === 13) {\n return digitsOnly.toUpperCase()\n }\n\n const match = stripped.match(ISBN_CANDIDATE_PATTERN)\n if (match) return match[0].toUpperCase()\n\n return undefined\n}\n","import type { ArchiveResolveResult } from \"../types/archiveResolve\"\nimport { normalizeGtin } from \"../utils/normalizeGtin\"\nimport { normalizeIsbn } from \"../utils/normalizeIsbn\"\nimport type { ComicInfo } from \"./parse\"\n\nconst readingDirection = (info: ComicInfo): \"ltr\" | \"rtl\" | undefined => {\n switch (info.Manga) {\n case \"YesAndRightToLeft\":\n return \"rtl\"\n case \"Yes\":\n case \"No\":\n return \"ltr\"\n default:\n return undefined\n }\n}\n\nconst trimToUndefined = (raw: string | undefined): string | undefined => {\n if (raw === undefined) return undefined\n const trimmed = raw.trim()\n return trimmed.length > 0 ? trimmed : undefined\n}\n\n/**\n * Split a comma-separated ComicInfo value into its individual tokens.\n * `Writer`, `Genre`, and `Tags` all follow the same de-facto convention:\n * tokens separated by `,`, with whitespace trimmed and empty tokens\n * dropped (real-world files leave trailing commas around).\n */\nconst splitCommaList = (raw: string | undefined): string[] => {\n if (raw === undefined) return []\n return raw\n .split(\",\")\n .map((token) => token.trim())\n .filter((token) => token.length > 0)\n}\n\nconst parseNonNegativeInt = (raw: string | undefined): number | undefined => {\n const trimmed = trimToUndefined(raw)\n if (trimmed === undefined) return undefined\n if (!/^\\d+$/.test(trimmed)) return undefined\n const n = Number.parseInt(trimmed, 10)\n return Number.isFinite(n) ? n : undefined\n}\n\nconst dateFromYearMonthDay = (\n info: ComicInfo,\n):\n | {\n year?: number\n month?: number\n day?: number\n }\n | undefined => {\n const year = parseNonNegativeInt(info.Year)\n const month = parseNonNegativeInt(info.Month)\n const day = parseNonNegativeInt(info.Day)\n\n if (year === undefined && month === undefined && day === undefined) {\n return undefined\n }\n\n return {\n ...(year !== undefined ? { year } : {}),\n ...(month !== undefined ? { month } : {}),\n ...(day !== undefined ? { day } : {}),\n }\n}\n\nexport const resolveComicInfo = (info: ComicInfo): ArchiveResolveResult => {\n const raw = info.GTIN\n const languageIso = trimToUndefined(info.LanguageISO)\n const authors = splitCommaList(info.Writer)\n const subjects = [...splitCommaList(info.Genre), ...splitCommaList(info.Tags)]\n\n return {\n gtin: normalizeGtin(raw),\n isbn: normalizeIsbn(raw),\n readingDirection: readingDirection(info),\n renditionLayout: undefined,\n title: trimToUndefined(info.Title),\n authors: authors.length > 0 ? authors : undefined,\n publisher: trimToUndefined(info.Publisher),\n rights: undefined,\n languages: languageIso !== undefined ? [languageIso] : undefined,\n date: dateFromYearMonthDay(info),\n subjects: subjects.length > 0 ? subjects : undefined,\n }\n}\n","import type { ArchiveResolveResult } from \"../types/archiveResolve\"\nimport type { KoboMetadata } from \"./parse\"\n\nexport const resolveKobo = (input: KoboMetadata): ArchiveResolveResult => ({\n gtin: undefined,\n isbn: undefined,\n readingDirection: undefined,\n renditionLayout: input.renditionLayout,\n title: undefined,\n authors: undefined,\n publisher: undefined,\n rights: undefined,\n languages: undefined,\n date: undefined,\n subjects: undefined,\n})\n","/**\n * Extract the calendar components of a W3CDTF literal — the\n * date subset of ISO 8601 that EPUB 3.3 § 5.5.3.2.4 mandates for\n * `dc:date`. Accepted forms: `YYYY`, `YYYY-MM`, `YYYY-MM-DD`, and\n * any of the above followed by a `Thh:mm[:ss[.s]][TZD]` time\n * portion that we ignore.\n *\n * Components are returned as plain integers (`month` is 1-12,\n * `day` is 1-31), independent of the host timezone — using\n * `Date.parse` would shift `2011-01-01` by a day in negative-offset\n * locales, which is the exact bug this regex-based approach exists\n * to avoid. Returns `undefined` when the input doesn't even match\n * a leading 4-digit year so consumers can fall back without\n * branching on partial shapes.\n */\nexport const parseW3cDtfDate = (\n raw: string | undefined,\n):\n | {\n year?: number\n month?: number\n day?: number\n }\n | undefined => {\n if (raw === undefined) return undefined\n\n const match = raw.trim().match(/^(\\d{4})(?:-(\\d{2})(?:-(\\d{2}))?)?/)\n if (match === null) return undefined\n\n const [, yearRaw, monthRaw, dayRaw] = match\n\n return {\n ...(yearRaw !== undefined ? { year: Number.parseInt(yearRaw, 10) } : {}),\n ...(monthRaw !== undefined ? { month: Number.parseInt(monthRaw, 10) } : {}),\n ...(dayRaw !== undefined ? { day: Number.parseInt(dayRaw, 10) } : {}),\n }\n}\n","import type { ArchiveResolveResult } from \"../types/archiveResolve\"\nimport { normalizeGtin } from \"../utils/normalizeGtin\"\nimport { normalizeIsbn } from \"../utils/normalizeIsbn\"\nimport { parseW3cDtfDate } from \"../utils/parseW3cDtfDate\"\nimport type { OpfIdentifier, OpfMetadata } from \"./parse\"\n\nconst rawIdentifierValueForIsbn = (\n identifiers: ReadonlyArray<OpfIdentifier>,\n): string | undefined => {\n for (const i of identifiers) {\n if (i.scheme !== undefined && i.scheme.trim().toLowerCase() === \"isbn\") {\n if (normalizeIsbn(i.value) !== undefined) return i.value\n }\n }\n\n for (const i of identifiers) {\n if (normalizeIsbn(i.value) !== undefined) return i.value\n }\n\n return undefined\n}\n\nexport const resolveOpf = (input: OpfMetadata): ArchiveResolveResult => {\n const ppd = input.pageProgressionDirection?.trim().toLowerCase()\n const readingDirection = ppd === \"ltr\" || ppd === \"rtl\" ? ppd : undefined\n\n const rl = input.renditionLayoutMeta?.trim().toLowerCase()\n const renditionLayout =\n rl === \"reflowable\" || rl === \"pre-paginated\" ? rl : undefined\n\n const raw = rawIdentifierValueForIsbn(input.identifiers)\n\n return {\n gtin: normalizeGtin(raw),\n isbn: normalizeIsbn(raw),\n readingDirection,\n renditionLayout,\n title: input.title,\n authors: input.creators.length > 0 ? [...input.creators] : undefined,\n publisher: input.publisher,\n rights: input.rights,\n languages: input.languages.length > 0 ? [...input.languages] : undefined,\n date: parseW3cDtfDate(input.date),\n subjects: input.subjects.length > 0 ? [...input.subjects] : undefined,\n }\n}\n","import type { AppleMetadata } from \"./apple/parse\"\nimport { resolveApple } from \"./apple/resolve\"\nimport type { ComicInfo } from \"./comicInfo/parse\"\nimport { resolveComicInfo } from \"./comicInfo/resolve\"\nimport type { KoboMetadata } from \"./kobo/parse\"\nimport { resolveKobo } from \"./kobo/resolve\"\nimport type { OpfMetadata } from \"./opf/parse\"\nimport { resolveOpf } from \"./opf/resolve\"\nimport type { ArchiveResolveResult } from \"./types/archiveResolve\"\n\nexport type ResolvedArchiveInput =\n | ComicInfo\n | KoboMetadata\n | AppleMetadata\n | OpfMetadata\n\nexport const resolveArchiveMetadata = (\n input: ResolvedArchiveInput,\n): ArchiveResolveResult => {\n if (input.kind === \"comicInfo\") return resolveComicInfo(input)\n if (input.kind === \"kobo\") return resolveKobo(input)\n if (input.kind === \"apple\") return resolveApple(input)\n return resolveOpf(input)\n}\n"],"names":["APPLE_IBOOKS_DISPLAY_OPTIONS_FILENAME","platformOptionsFromElement","platform","option","parseAppleDisplayOptionsXml","xml","doc","XmlDocument","platformEl","COMIC_INFO_MANGA_VALUES","isComicInfoManga","value","v","COMIC_INFO_FILENAME","SKIP_ELEMENT_CHILDREN","hasNestedElement","el","c","trimmedText","parseComicInfo","cause","message","fields","child","text","KOBO_DISPLAY_OPTIONS_FILENAME","parseKoboDisplayOptionsDocument","parseKoboXml","tokenizeXmlSpaceSeparatedList","raw","layoutHintsFromItemrefProperties","properties","tokens","renditionLayout","renditionFlow","elementLocalName","name","localNameEq","elementName","wantLocal","isXmlElement","node","childNamedLocal","parent","localName","childrenNamedLocal","out","identifiersFromMetadata","metadataEl","identifiers","schemeTrimmed","firstTextByLocalName","found","t","textsByLocalName","coverContentIdFromMetadata","coverId","content","coverHrefFromManifestAndMetadata","manifestItems","isImage","item","byCoverImageProperty","coverContentId","match","metaValByProperty","property","m","guideFromPackage","guideEl","refs","ref","href","manifestItemFromXmlElement","id","mediaType","manifestItemsAndById","manifestEl","items","byId","parsed","spineRowsFromByIdAndSpine","spineEl","rows","itemref","idref","manifestItem","hints","parseOpf","opfXml","spineRows","pageProgressionDirectionRaw","pageProgressionDirection","spineTocRaw","spineTocIdref","title","publisher","rights","date","creators","languages","subjects","renditionLayoutMeta","renditionFlowMeta","renditionSpreadMeta","coverHref","guide","resolveApple","input","GTIN_LENGTHS","normalizeGtin","digits","ISBN_CANDIDATE_PATTERN","normalizeIsbn","stripped","digitsOnly","readingDirection","info","trimToUndefined","trimmed","splitCommaList","token","parseNonNegativeInt","n","dateFromYearMonthDay","year","month","day","resolveComicInfo","languageIso","authors","resolveKobo","parseW3cDtfDate","yearRaw","monthRaw","dayRaw","rawIdentifierValueForIsbn","i","resolveOpf","ppd","rl","resolveArchiveMetadata"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"index.js","names":[],"sources":["../src/apple/parse.ts","../src/comicInfo/manga.ts","../src/comicInfo/parse.ts","../src/kobo/parse.ts","../src/utils/tokenizeXmlSpaceSeparatedList.ts","../src/opf/spineItemrefProperties.ts","../src/opf/parse.ts","../src/apple/resolve.ts","../src/utils/normalizeGtin.ts","../src/utils/normalizeIsbn.ts","../src/comicInfo/resolve.ts","../src/kobo/resolve.ts","../src/utils/parseW3cDtfDate.ts","../src/opf/resolve.ts","../src/resolve.ts"],"sourcesContent":["import type { XmlElement } from \"xmldoc\"\nimport { XmlDocument } from \"xmldoc\"\n\nexport const APPLE_IBOOKS_DISPLAY_OPTIONS_FILENAME =\n \"com.apple.ibooks.display-options.xml\"\n\nexport type AppleDisplayOption = {\n readonly name?: string\n readonly value: string\n}\n\nexport type AppleMetadata = {\n readonly kind: \"apple\"\n readonly displayOptions?: {\n readonly platform?: {\n readonly options: ReadonlyArray<AppleDisplayOption>\n }\n }\n}\n\nconst platformOptionsFromElement = (\n platform: XmlElement,\n): ReadonlyArray<AppleDisplayOption> =>\n platform.childrenNamed(\"option\").map((option) => ({\n name: option.attr?.name,\n value: option.val,\n }))\n\nexport const parseAppleDisplayOptionsXml = (xml: string): AppleMetadata => {\n const doc = new XmlDocument(xml)\n const root = doc.name?.toLowerCase()\n\n if (root !== \"display_options\") {\n return { kind: \"apple\" }\n }\n\n const platformEl = doc.childNamed(\"platform\")\n\n return {\n kind: \"apple\",\n displayOptions: {\n ...(platformEl !== undefined\n ? { platform: { options: platformOptionsFromElement(platformEl) } }\n : {}),\n },\n }\n}\n","/**\n * Allowed `Manga` element values (ComicInfo 2.0 `Manga` simpleType).\n *\n * @see https://anansi-project.github.io/docs/comicinfo/documentation#manga\n */\nexport const COMIC_INFO_MANGA_VALUES = [\n \"Unknown\",\n \"No\",\n \"Yes\",\n \"YesAndRightToLeft\",\n] as const\n\nexport type ComicInfoManga = (typeof COMIC_INFO_MANGA_VALUES)[number]\n\nexport const isComicInfoManga = (value: string): value is ComicInfoManga => {\n for (const v of COMIC_INFO_MANGA_VALUES) {\n if (v === value) return true\n }\n return false\n}\n","import type { XmlElement } from \"xmldoc\"\nimport { XmlDocument } from \"xmldoc\"\nimport type { ComicInfoManga } from \"./manga\"\n\n/** Canonical top-level filename; real archives may use any casing. */\nexport const COMIC_INFO_FILENAME = \"ComicInfo.xml\"\n\n/**\n * Parsed `ComicInfo.xml` root: one optional string property per child element,\n * using the same names as in the file (e.g. `Title`, `GTIN`, `LanguageISO`).\n * Nested blocks such as `Pages` are skipped. Other simple child elements are\n * still copied onto this object under their tag name.\n *\n * @see https://anansi-project.github.io/docs/comicinfo/intro\n * @see https://github.com/anansi-project/comicinfo/blob/main/drafts/v2.1/ComicInfo.xsd for schema\n */\nexport interface ComicInfo {\n readonly kind: \"comicInfo\"\n AgeRating?: string\n AlternateCount?: string\n AlternateNumber?: string\n AlternateSeries?: string\n BlackAndWhite?: string\n Characters?: string\n Colorist?: string\n CommunityRating?: string\n Count?: string\n CoverArtist?: string\n Day?: string\n Editor?: string\n Format?: string\n Genre?: string\n GTIN?: string\n Imprint?: string\n Inker?: string\n LanguageISO?: string\n Letterer?: string\n Locations?: string\n MainCharacterOrTeam?: string\n /** Schema literals per {@link ComicInfoManga}; files may still use other strings. */\n Manga?: ComicInfoManga | (string & {})\n Month?: string\n Notes?: string\n Number?: string\n PageCount?: string\n Penciller?: string\n Publisher?: string\n Review?: string\n ScanInformation?: string\n Series?: string\n SeriesGroup?: string\n StoryArc?: string\n StoryArcNumber?: string\n Summary?: string\n Tags?: string\n Teams?: string\n Title?: string\n Translator?: string\n Volume?: string\n Web?: string\n Writer?: string\n Year?: string\n [tag: string]: string | undefined\n}\n\nconst SKIP_ELEMENT_CHILDREN = new Set([\"Pages\", \"ComicInfo\"])\n\nconst hasNestedElement = (el: XmlElement) =>\n el.children.some((c) => c.type === \"element\")\n\nconst trimmedText = (el: XmlElement): string | undefined => {\n const t = el.val.trim()\n return t.length > 0 ? t : undefined\n}\n\n/**\n * Parse a raw `ComicInfo.xml` body. Each direct child element with plain text\n * becomes a property named after that tag. Malformed XML throws; the parser\n * error is attached as `cause`.\n */\nexport const parseComicInfo = (xml: string): ComicInfo => {\n let doc: XmlDocument\n try {\n doc = new XmlDocument(xml)\n } catch (cause) {\n const message = cause instanceof Error ? cause.message : String(cause)\n throw new Error(`${COMIC_INFO_FILENAME} is malformed: ${message}`, {\n cause,\n })\n }\n\n const fields: Record<string, string> = {}\n\n doc.eachChild((child) => {\n if (child.type !== \"element\") return\n if (SKIP_ELEMENT_CHILDREN.has(child.name)) return\n if (hasNestedElement(child)) return\n\n const text = trimmedText(child)\n if (text === undefined) return\n if (fields[child.name] !== undefined) return\n\n fields[child.name] = text\n })\n\n // `as ComicInfo`: TS cannot infer that spreading `Record<string, string>` into `{ kind }` satisfies the named optional fields plus `[tag: string]: string | undefined` on `ComicInfo`.\n return { kind: \"comicInfo\", ...fields } as ComicInfo\n}\n","import { XmlDocument } from \"xmldoc\"\n\nexport const KOBO_DISPLAY_OPTIONS_FILENAME = \"com.kobobooks.display-options.xml\"\n\n/**\n * Normalized fields from Kobo-specific XML sidecars. Grows as new Kobo\n * document kinds are supported; absent keys mean not present or unknown.\n */\nexport type KoboMetadata = {\n readonly kind: \"kobo\"\n renditionLayout?: `reflowable` | `pre-paginated`\n}\n\nconst parseKoboDisplayOptionsDocument = (\n doc: XmlDocument,\n): { renditionLayout?: \"pre-paginated\" } => {\n const platform = doc.childNamed(\"platform\")\n if (!platform) return {}\n\n for (const option of platform.childrenNamed(\"option\")) {\n if (option.attr?.name !== \"fixed-layout\") continue\n if (option.val.trim().toLowerCase() === \"true\") {\n return { renditionLayout: \"pre-paginated\" }\n }\n return {}\n }\n\n return {}\n}\n\n/**\n * Parse a Kobo-related XML document. Unsupported or unrecognized roots\n * yield an empty object. Malformed XML throws from the underlying parser.\n */\nexport const parseKoboXml = (xml: string): KoboMetadata => {\n const doc = new XmlDocument(xml)\n const root = doc.name?.toLowerCase()\n\n if (root === \"display_options\") {\n return { kind: \"kobo\", ...parseKoboDisplayOptionsDocument(doc) }\n }\n\n return { kind: \"kobo\" }\n}\n","/**\n * EPUB/XML space-separated token lists (e.g. `properties` on `item` / `itemref`).\n * Trims the raw string, splits on ASCII whitespace, drops empty segments.\n */\nexport const tokenizeXmlSpaceSeparatedList = (\n raw: string | undefined,\n): readonly string[] => {\n if (raw === undefined || raw.trim().length === 0) {\n return []\n }\n\n return raw\n .trim()\n .split(/\\s+/)\n .filter((t) => t.length > 0)\n}\n","import { tokenizeXmlSpaceSeparatedList } from \"../utils/tokenizeXmlSpaceSeparatedList\"\n\nexport type OpfItemrefLayoutHints = {\n readonly renditionLayout?: `reflowable` | `pre-paginated`\n readonly renditionFlow?:\n | `scrolled-continuous`\n | `scrolled-doc`\n | `paginated`\n | `auto`\n readonly pageSpreadLeft?: true\n readonly pageSpreadRight?: true\n}\n\n/**\n * EPUB `itemref` `properties` attribute (space-separated tokens).\n *\n * @see https://www.w3.org/TR/epub/#attrdef-properties\n */\nexport const layoutHintsFromItemrefProperties = (\n properties: string | undefined,\n): OpfItemrefLayoutHints => {\n const tokens = tokenizeXmlSpaceSeparatedList(properties)\n if (tokens.length === 0) {\n return {}\n }\n\n let renditionLayout: `reflowable` | `pre-paginated` | undefined\n if (tokens.includes(`rendition:layout-reflowable`)) {\n renditionLayout = `reflowable`\n }\n if (tokens.includes(`rendition:layout-pre-paginated`)) {\n renditionLayout = `pre-paginated`\n }\n\n let renditionFlow: OpfItemrefLayoutHints[\"renditionFlow\"]\n if (tokens.includes(`rendition:flow-auto`)) {\n renditionFlow = `auto`\n }\n if (tokens.includes(`rendition:flow-paginated`)) {\n renditionFlow = `paginated`\n }\n if (tokens.includes(`rendition:flow-scrolled-doc`)) {\n renditionFlow = `scrolled-doc`\n }\n if (tokens.includes(`rendition:flow-scrolled-continuous`)) {\n renditionFlow = `scrolled-continuous`\n }\n\n return {\n ...(renditionLayout !== undefined ? { renditionLayout } : {}),\n ...(renditionFlow !== undefined ? { renditionFlow } : {}),\n ...(tokens.includes(`page-spread-left`)\n ? { pageSpreadLeft: true as const }\n : {}),\n ...(tokens.includes(`page-spread-right`)\n ? { pageSpreadRight: true as const }\n : {}),\n }\n}\n","import type { XmlElement, XmlNodeBase } from \"xmldoc\"\nimport { XmlDocument } from \"xmldoc\"\nimport { tokenizeXmlSpaceSeparatedList } from \"../utils/tokenizeXmlSpaceSeparatedList\"\nimport { layoutHintsFromItemrefProperties } from \"./spineItemrefProperties\"\n\nexport type OpfSpineManifestItem = {\n readonly id: string\n readonly href: string\n readonly mediaType?: string\n readonly properties?: string\n}\n\nexport type OpfIdentifier = {\n readonly value: string\n readonly scheme?: string\n}\n\nexport type OpfSpineRow = {\n readonly idref: string\n readonly id: string\n readonly href: string\n readonly mediaType?: string\n readonly properties?: string\n readonly renditionLayout?: `reflowable` | `pre-paginated`\n readonly renditionFlow?:\n | `scrolled-continuous`\n | `scrolled-doc`\n | `paginated`\n | `auto`\n readonly pageSpreadLeft?: true\n readonly pageSpreadRight?: true\n}\n\nexport type OpfGuideReference = {\n readonly href: string\n readonly title: string\n readonly type: string\n}\n\nconst elementLocalName = (name: string): string =>\n name.includes(\":\") ? name.slice(name.lastIndexOf(\":\") + 1) : name\n\nconst localNameEq = (elementName: string, wantLocal: string): boolean =>\n elementLocalName(elementName).toLowerCase() === wantLocal.toLowerCase()\n\nconst isXmlElement = (node: XmlNodeBase): node is XmlElement =>\n node.type === \"element\"\n\nconst childNamedLocal = (\n parent: XmlElement,\n localName: string,\n): XmlElement | undefined => {\n for (const node of parent.children) {\n if (!isXmlElement(node)) continue\n if (localNameEq(node.name, localName)) return node\n }\n return undefined\n}\n\nconst childrenNamedLocal = (\n parent: XmlElement,\n localName: string,\n): XmlElement[] => {\n const out: XmlElement[] = []\n for (const node of parent.children) {\n if (!isXmlElement(node)) continue\n if (localNameEq(node.name, localName)) out.push(node)\n }\n return out\n}\n\nconst identifiersFromMetadata = (metadataEl: XmlElement): OpfIdentifier[] => {\n const identifiers: OpfIdentifier[] = []\n\n metadataEl.eachChild((child) => {\n if (elementLocalName(child.name).toLowerCase() !== \"identifier\") return\n\n const value = child.val.trim()\n if (value.length === 0) return\n\n const scheme =\n child.attr[\"opf:scheme\"] ?? child.attr[\"opf:Scheme\"] ?? child.attr.scheme\n const schemeTrimmed = scheme?.trim()\n\n identifiers.push({\n value,\n ...(schemeTrimmed !== undefined && schemeTrimmed.length > 0\n ? { scheme: schemeTrimmed }\n : {}),\n })\n })\n\n return identifiers\n}\n\nconst firstTextByLocalName = (\n metadataEl: XmlElement,\n localName: string,\n): string | undefined => {\n let found: string | undefined\n metadataEl.eachChild((child) => {\n if (found !== undefined) return\n if (elementLocalName(child.name).toLowerCase() !== localName.toLowerCase())\n return\n const t = child.val.trim()\n if (t.length > 0) found = t\n })\n return found\n}\n\nconst textsByLocalName = (\n metadataEl: XmlElement,\n localName: string,\n): string[] => {\n const out: string[] = []\n metadataEl.eachChild((child) => {\n if (elementLocalName(child.name).toLowerCase() !== localName.toLowerCase())\n return\n const t = child.val.trim()\n if (t.length > 0) out.push(t)\n })\n return out\n}\n\nconst coverContentIdFromMetadata = (\n metadataEl: XmlElement,\n): string | undefined => {\n let coverId: string | undefined\n metadataEl.eachChild((child) => {\n if (coverId !== undefined) return\n if (elementLocalName(child.name).toLowerCase() !== \"meta\") return\n if (child.attr.name?.toLowerCase() !== \"cover\") return\n const content = child.attr.content?.trim()\n if (content !== undefined && content.length > 0) coverId = content\n })\n return coverId\n}\n\n/**\n * EPUB cover image inside the manifest. Resolution order, matching\n * what the spec lays out and what the bulk of EPUB producers in the\n * wild rely on:\n *\n * 1. EPUB 3 — the manifest item carrying the `cover-image` token in\n * its `properties` attribute (§ D.6.1).\n * 2. EPUB 2 — `<meta name=\"cover\" content=\"ID\"/>` in `metadata`,\n * resolved to the manifest item with that `id`.\n * 3. Last-resort fallback — any image manifest item whose `id`\n * contains the substring `cover` (case-insensitive); covers the\n * long tail of producers that emit neither the EPUB 3 property\n * nor the EPUB 2 meta.\n *\n * Each step requires the candidate manifest item to advertise an\n * `image/*` media type so non-image artefacts named `cover` (XHTML\n * cover pages, NCX entries) don't slip through.\n */\nconst coverHrefFromManifestAndMetadata = ({\n manifestItems,\n metadataEl,\n}: {\n manifestItems: ReadonlyArray<OpfSpineManifestItem>\n metadataEl: XmlElement | undefined\n}): string | undefined => {\n const isImage = (item: OpfSpineManifestItem): boolean =>\n item.mediaType?.toLowerCase().includes(\"image/\") === true\n\n const byCoverImageProperty = manifestItems.find((item) => {\n if (!isImage(item)) return false\n return tokenizeXmlSpaceSeparatedList(item.properties).includes(\n \"cover-image\",\n )\n })\n if (byCoverImageProperty !== undefined) return byCoverImageProperty.href\n\n if (metadataEl !== undefined) {\n const coverContentId = coverContentIdFromMetadata(metadataEl)\n if (coverContentId !== undefined) {\n const match = manifestItems.find(\n (item) => item.id === coverContentId && isImage(item),\n )\n if (match !== undefined) return match.href\n }\n }\n\n return manifestItems.find(\n (item) => item.id.toLowerCase().includes(\"cover\") && isImage(item),\n )?.href\n}\n\nconst metaValByProperty = (\n metadataEl: XmlElement,\n property: string,\n): string | undefined => {\n const meta = childrenNamedLocal(metadataEl, \"meta\").find(\n (m) => m.attr.property === property,\n )\n const raw = meta?.val\n if (raw === undefined || raw.trim().length === 0) return undefined\n return raw\n}\n\nconst guideFromPackage = (doc: XmlElement): OpfGuideReference[] => {\n const guideEl = childNamedLocal(doc, \"guide\")\n if (guideEl === undefined) return []\n\n const refs: OpfGuideReference[] = []\n\n for (const ref of childrenNamedLocal(guideEl, \"reference\")) {\n const href = ref.attr.href?.trim()\n if (href === undefined || href.length === 0) continue\n refs.push({\n href,\n title: ref.attr.title?.trim() ?? ``,\n type: ref.attr.type?.trim() ?? ``,\n })\n }\n\n return refs\n}\n\nconst manifestItemFromXmlElement = (\n item: XmlElement,\n): OpfSpineManifestItem | undefined => {\n const id = item.attr.id\n const href = item.attr.href\n if (id === undefined || id.length === 0) return undefined\n if (href === undefined || href.length === 0) return undefined\n\n const mediaType = item.attr[\"media-type\"]\n const properties = item.attr.properties?.trim()\n return {\n id,\n href,\n ...(mediaType !== undefined && mediaType.length > 0 ? { mediaType } : {}),\n ...(properties !== undefined && properties.length > 0\n ? { properties }\n : {}),\n }\n}\n\nconst manifestItemsAndById = (\n manifestEl: XmlElement,\n): {\n items: OpfSpineManifestItem[]\n byId: Map<string, OpfSpineManifestItem>\n} => {\n const items: OpfSpineManifestItem[] = []\n const byId = new Map<string, OpfSpineManifestItem>()\n\n for (const el of childrenNamedLocal(manifestEl, \"item\")) {\n const parsed = manifestItemFromXmlElement(el)\n if (parsed === undefined) continue\n items.push(parsed)\n byId.set(parsed.id, parsed)\n }\n\n return { items, byId }\n}\n\nconst spineRowsFromByIdAndSpine = (\n byId: Map<string, OpfSpineManifestItem>,\n spineEl: XmlElement,\n): OpfSpineRow[] => {\n const rows: OpfSpineRow[] = []\n\n for (const itemref of childrenNamedLocal(spineEl, \"itemref\")) {\n const idref = itemref.attr.idref\n if (idref === undefined || idref.trim().length === 0) continue\n\n const manifestItem = byId.get(idref)\n if (manifestItem === undefined) continue\n\n const hints = layoutHintsFromItemrefProperties(itemref.attr.properties)\n\n rows.push({\n idref,\n id: manifestItem.id,\n href: manifestItem.href,\n ...(manifestItem.mediaType !== undefined\n ? { mediaType: manifestItem.mediaType }\n : {}),\n ...(manifestItem.properties !== undefined\n ? { properties: manifestItem.properties }\n : {}),\n ...(hints.renditionLayout !== undefined\n ? { renditionLayout: hints.renditionLayout }\n : {}),\n ...(hints.renditionFlow !== undefined\n ? { renditionFlow: hints.renditionFlow }\n : {}),\n ...(hints.pageSpreadLeft !== undefined\n ? { pageSpreadLeft: hints.pageSpreadLeft }\n : {}),\n ...(hints.pageSpreadRight !== undefined\n ? { pageSpreadRight: hints.pageSpreadRight }\n : {}),\n })\n }\n\n return rows\n}\n\nexport type OpfMetadata = {\n readonly kind: \"opf\"\n readonly manifestItems: ReadonlyArray<OpfSpineManifestItem>\n readonly spineRows: ReadonlyArray<OpfSpineRow>\n readonly spineTocIdref: string | undefined\n readonly identifiers: ReadonlyArray<OpfIdentifier>\n readonly title: string | undefined\n /** `dc:creator` values, in document order, trimmed; empty when none. */\n readonly creators: ReadonlyArray<string>\n /** First non-empty `dc:publisher`, trimmed. */\n readonly publisher: string | undefined\n /** First non-empty `dc:rights`, trimmed. */\n readonly rights: string | undefined\n /** `dc:language` values, in document order, trimmed; empty when none. */\n readonly languages: ReadonlyArray<string>\n /** `dc:subject` values, in document order, trimmed; empty when none. */\n readonly subjects: ReadonlyArray<string>\n /**\n * Raw `dc:date` value as authored. EPUB 3 requires W3CDTF (a profile\n * of ISO 8601), but real-world publishers also ship free text here,\n * so the value is exposed verbatim and consumers normalize as needed.\n */\n readonly date: string | undefined\n /**\n * Manifest-relative `href` of the cover image, when one can be\n * resolved from `cover-image` properties (EPUB 3), the EPUB 2\n * `<meta name=\"cover\">` convention, or an `id` that contains\n * `cover` on an image manifest item. The href is returned exactly\n * as it appears in the manifest — callers own folder-prefix\n * resolution against the OPF's location in the archive.\n */\n readonly coverHref: string | undefined\n readonly renditionLayoutMeta: string | undefined\n readonly renditionFlowMeta: string | undefined\n readonly renditionSpreadMeta: string | undefined\n readonly pageProgressionDirection: string | undefined\n readonly guide: ReadonlyArray<OpfGuideReference>\n}\n\n/**\n * Parses an EPUB package document (OPF) into structured metadata.\n *\n * Direct children of `package` (`metadata`, `manifest`, `spine`, `guide`) and\n * their structural children (`item`, `itemref`, `reference`, `meta`) are\n * matched by **local name** (ASCII case-insensitive), so prefixed tags such as\n * `opf:manifest` are supported the same as unprefixed `manifest`.\n *\n * Attribute names on `spine` / `itemref` are still read as emitted by xmldoc\n * (no QName normalization).\n */\nexport const parseOpf = (opfXml: string): OpfMetadata => {\n const doc = new XmlDocument(opfXml)\n const manifestEl = childNamedLocal(doc, \"manifest\")\n const spineEl = childNamedLocal(doc, \"spine\")\n const metadataEl = childNamedLocal(doc, \"metadata\")\n\n let manifestItems: OpfSpineManifestItem[] = []\n let spineRows: OpfSpineRow[] = []\n\n if (manifestEl !== undefined) {\n const { items, byId } = manifestItemsAndById(manifestEl)\n manifestItems = items\n if (spineEl !== undefined) {\n spineRows = spineRowsFromByIdAndSpine(byId, spineEl)\n }\n }\n\n const pageProgressionDirectionRaw =\n spineEl?.attr[\"page-progression-direction\"]\n const pageProgressionDirection =\n pageProgressionDirectionRaw !== undefined &&\n pageProgressionDirectionRaw.trim().length > 0\n ? pageProgressionDirectionRaw\n : undefined\n\n const spineTocRaw = spineEl?.attr.toc\n const spineTocIdref =\n spineTocRaw !== undefined && spineTocRaw.trim().length > 0\n ? spineTocRaw.trim()\n : undefined\n\n let title: string | undefined\n let publisher: string | undefined\n let rights: string | undefined\n let date: string | undefined\n let creators: string[] = []\n let languages: string[] = []\n let subjects: string[] = []\n let renditionLayoutMeta: string | undefined\n let renditionFlowMeta: string | undefined\n let renditionSpreadMeta: string | undefined\n const identifiers: OpfIdentifier[] = []\n\n if (metadataEl !== undefined) {\n title = firstTextByLocalName(metadataEl, \"title\")\n publisher = firstTextByLocalName(metadataEl, \"publisher\")\n rights = firstTextByLocalName(metadataEl, \"rights\")\n date = firstTextByLocalName(metadataEl, \"date\")\n creators = textsByLocalName(metadataEl, \"creator\")\n languages = textsByLocalName(metadataEl, \"language\")\n subjects = textsByLocalName(metadataEl, \"subject\")\n renditionLayoutMeta = metaValByProperty(metadataEl, \"rendition:layout\")\n renditionFlowMeta = metaValByProperty(metadataEl, \"rendition:flow\")\n renditionSpreadMeta = metaValByProperty(metadataEl, \"rendition:spread\")\n identifiers.push(...identifiersFromMetadata(metadataEl))\n }\n\n const coverHref = coverHrefFromManifestAndMetadata({\n manifestItems,\n metadataEl,\n })\n\n const guide = guideFromPackage(doc)\n\n return {\n kind: \"opf\",\n manifestItems,\n spineRows,\n spineTocIdref,\n identifiers,\n title,\n creators,\n publisher,\n rights,\n languages,\n subjects,\n date,\n coverHref,\n renditionLayoutMeta,\n renditionFlowMeta,\n renditionSpreadMeta,\n pageProgressionDirection,\n guide,\n }\n}\n","import type { ArchiveResolveResult } from \"../types/archiveResolve\"\nimport type { AppleMetadata } from \"./parse\"\n\nexport const resolveApple = (input: AppleMetadata): ArchiveResolveResult => {\n const fixedLayout = input.displayOptions?.platform?.options?.find(\n (o) => o.name === \"fixed-layout\",\n )?.value\n\n const renditionLayout =\n fixedLayout?.trim().toLowerCase() === \"true\" ? \"pre-paginated\" : undefined\n\n return {\n gtin: undefined,\n isbn: undefined,\n readingDirection: undefined,\n renditionLayout,\n title: undefined,\n authors: undefined,\n publisher: undefined,\n rights: undefined,\n languages: undefined,\n date: undefined,\n subjects: undefined,\n }\n}\n","const GTIN_LENGTHS = new Set([8, 12, 13, 14])\n\n/**\n * Normalize a raw GTIN / EAN / UPC string to digits only when the length\n * matches a GS1 GTIN family size (8, 12, 13, or 14). Does not verify check digits.\n */\nexport const normalizeGtin = (\n raw: string | number | undefined | null,\n): string | undefined => {\n if (raw === undefined || raw === null) return undefined\n\n const digits = String(raw).replace(/\\D/g, \"\")\n if (digits.length === 0 || !GTIN_LENGTHS.has(digits.length)) return undefined\n\n return digits\n}\n","const ISBN_CANDIDATE_PATTERN = /(?:97[89])?\\d{9}[\\dXx]/\n\n/**\n * Normalize a raw ISBN-ish string into a canonical 10- or 13-character\n * form, or `undefined` when no recognisable ISBN can be recovered.\n *\n * - Strips the common `urn:isbn:` / `isbn:` prefixes.\n * - Drops everything that isn't a digit or `X`.\n * - Validates the resulting length (10 or 13).\n * - Falls back to a lax regex scan so publishers that stuff free text\n * around the number still yield a usable value.\n */\nexport const normalizeIsbn = (\n raw: string | number | undefined | null,\n): string | undefined => {\n if (raw === undefined || raw === null) return undefined\n\n const stripped = String(raw)\n .trim()\n .replace(/^urn:isbn:/i, \"\")\n .replace(/^isbn[:\\s-]*/i, \"\")\n\n const digitsOnly = stripped.replace(/[^0-9Xx]/g, \"\")\n\n if (digitsOnly.length === 10 || digitsOnly.length === 13) {\n return digitsOnly.toUpperCase()\n }\n\n const match = stripped.match(ISBN_CANDIDATE_PATTERN)\n if (match) return match[0].toUpperCase()\n\n return undefined\n}\n","import type { ArchiveResolveResult } from \"../types/archiveResolve\"\nimport { normalizeGtin } from \"../utils/normalizeGtin\"\nimport { normalizeIsbn } from \"../utils/normalizeIsbn\"\nimport type { ComicInfo } from \"./parse\"\n\nconst readingDirection = (info: ComicInfo): \"ltr\" | \"rtl\" | undefined => {\n switch (info.Manga) {\n case \"YesAndRightToLeft\":\n return \"rtl\"\n case \"Yes\":\n case \"No\":\n return \"ltr\"\n default:\n return undefined\n }\n}\n\nconst trimToUndefined = (raw: string | undefined): string | undefined => {\n if (raw === undefined) return undefined\n const trimmed = raw.trim()\n return trimmed.length > 0 ? trimmed : undefined\n}\n\n/**\n * Split a comma-separated ComicInfo value into its individual tokens.\n * `Writer`, `Genre`, and `Tags` all follow the same de-facto convention:\n * tokens separated by `,`, with whitespace trimmed and empty tokens\n * dropped (real-world files leave trailing commas around).\n */\nconst splitCommaList = (raw: string | undefined): string[] => {\n if (raw === undefined) return []\n return raw\n .split(\",\")\n .map((token) => token.trim())\n .filter((token) => token.length > 0)\n}\n\nconst parseNonNegativeInt = (raw: string | undefined): number | undefined => {\n const trimmed = trimToUndefined(raw)\n if (trimmed === undefined) return undefined\n if (!/^\\d+$/.test(trimmed)) return undefined\n const n = Number.parseInt(trimmed, 10)\n return Number.isFinite(n) ? n : undefined\n}\n\nconst dateFromYearMonthDay = (\n info: ComicInfo,\n):\n | {\n year?: number\n month?: number\n day?: number\n }\n | undefined => {\n const year = parseNonNegativeInt(info.Year)\n const month = parseNonNegativeInt(info.Month)\n const day = parseNonNegativeInt(info.Day)\n\n if (year === undefined && month === undefined && day === undefined) {\n return undefined\n }\n\n return {\n ...(year !== undefined ? { year } : {}),\n ...(month !== undefined ? { month } : {}),\n ...(day !== undefined ? { day } : {}),\n }\n}\n\nexport const resolveComicInfo = (info: ComicInfo): ArchiveResolveResult => {\n const raw = info.GTIN\n const languageIso = trimToUndefined(info.LanguageISO)\n const authors = splitCommaList(info.Writer)\n const subjects = [...splitCommaList(info.Genre), ...splitCommaList(info.Tags)]\n\n return {\n gtin: normalizeGtin(raw),\n isbn: normalizeIsbn(raw),\n readingDirection: readingDirection(info),\n renditionLayout: undefined,\n title: trimToUndefined(info.Title),\n authors: authors.length > 0 ? authors : undefined,\n publisher: trimToUndefined(info.Publisher),\n rights: undefined,\n languages: languageIso !== undefined ? [languageIso] : undefined,\n date: dateFromYearMonthDay(info),\n subjects: subjects.length > 0 ? subjects : undefined,\n }\n}\n","import type { ArchiveResolveResult } from \"../types/archiveResolve\"\nimport type { KoboMetadata } from \"./parse\"\n\nexport const resolveKobo = (input: KoboMetadata): ArchiveResolveResult => ({\n gtin: undefined,\n isbn: undefined,\n readingDirection: undefined,\n renditionLayout: input.renditionLayout,\n title: undefined,\n authors: undefined,\n publisher: undefined,\n rights: undefined,\n languages: undefined,\n date: undefined,\n subjects: undefined,\n})\n","/**\n * Extract the calendar components of a W3CDTF literal — the\n * date subset of ISO 8601 that EPUB 3.3 § 5.5.3.2.4 mandates for\n * `dc:date`. Accepted forms: `YYYY`, `YYYY-MM`, `YYYY-MM-DD`, and\n * any of the above followed by a `Thh:mm[:ss[.s]][TZD]` time\n * portion that we ignore.\n *\n * Components are returned as plain integers (`month` is 1-12,\n * `day` is 1-31), independent of the host timezone — using\n * `Date.parse` would shift `2011-01-01` by a day in negative-offset\n * locales, which is the exact bug this regex-based approach exists\n * to avoid. Returns `undefined` when the input doesn't even match\n * a leading 4-digit year so consumers can fall back without\n * branching on partial shapes.\n */\nexport const parseW3cDtfDate = (\n raw: string | undefined,\n):\n | {\n year?: number\n month?: number\n day?: number\n }\n | undefined => {\n if (raw === undefined) return undefined\n\n const match = raw.trim().match(/^(\\d{4})(?:-(\\d{2})(?:-(\\d{2}))?)?/)\n if (match === null) return undefined\n\n const [, yearRaw, monthRaw, dayRaw] = match\n\n return {\n ...(yearRaw !== undefined ? { year: Number.parseInt(yearRaw, 10) } : {}),\n ...(monthRaw !== undefined ? { month: Number.parseInt(monthRaw, 10) } : {}),\n ...(dayRaw !== undefined ? { day: Number.parseInt(dayRaw, 10) } : {}),\n }\n}\n","import type { ArchiveResolveResult } from \"../types/archiveResolve\"\nimport { normalizeGtin } from \"../utils/normalizeGtin\"\nimport { normalizeIsbn } from \"../utils/normalizeIsbn\"\nimport { parseW3cDtfDate } from \"../utils/parseW3cDtfDate\"\nimport type { OpfIdentifier, OpfMetadata } from \"./parse\"\n\nconst rawIdentifierValueForIsbn = (\n identifiers: ReadonlyArray<OpfIdentifier>,\n): string | undefined => {\n for (const i of identifiers) {\n if (i.scheme !== undefined && i.scheme.trim().toLowerCase() === \"isbn\") {\n if (normalizeIsbn(i.value) !== undefined) return i.value\n }\n }\n\n for (const i of identifiers) {\n if (normalizeIsbn(i.value) !== undefined) return i.value\n }\n\n return undefined\n}\n\nexport const resolveOpf = (input: OpfMetadata): ArchiveResolveResult => {\n const ppd = input.pageProgressionDirection?.trim().toLowerCase()\n const readingDirection = ppd === \"ltr\" || ppd === \"rtl\" ? ppd : undefined\n\n const rl = input.renditionLayoutMeta?.trim().toLowerCase()\n const renditionLayout =\n rl === \"reflowable\" || rl === \"pre-paginated\" ? rl : undefined\n\n const raw = rawIdentifierValueForIsbn(input.identifiers)\n\n return {\n gtin: normalizeGtin(raw),\n isbn: normalizeIsbn(raw),\n readingDirection,\n renditionLayout,\n title: input.title,\n authors: input.creators.length > 0 ? [...input.creators] : undefined,\n publisher: input.publisher,\n rights: input.rights,\n languages: input.languages.length > 0 ? [...input.languages] : undefined,\n date: parseW3cDtfDate(input.date),\n subjects: input.subjects.length > 0 ? [...input.subjects] : undefined,\n }\n}\n","import type { AppleMetadata } from \"./apple/parse\"\nimport { resolveApple } from \"./apple/resolve\"\nimport type { ComicInfo } from \"./comicInfo/parse\"\nimport { resolveComicInfo } from \"./comicInfo/resolve\"\nimport type { KoboMetadata } from \"./kobo/parse\"\nimport { resolveKobo } from \"./kobo/resolve\"\nimport type { OpfMetadata } from \"./opf/parse\"\nimport { resolveOpf } from \"./opf/resolve\"\nimport type { ArchiveResolveResult } from \"./types/archiveResolve\"\n\nexport type ResolvedArchiveInput =\n | ComicInfo\n | KoboMetadata\n | AppleMetadata\n | OpfMetadata\n\nexport const resolveArchiveMetadata = (\n input: ResolvedArchiveInput,\n): ArchiveResolveResult => {\n if (input.kind === \"comicInfo\") return resolveComicInfo(input)\n if (input.kind === \"kobo\") return resolveKobo(input)\n if (input.kind === \"apple\") return resolveApple(input)\n return resolveOpf(input)\n}\n"],"mappings":"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"}
|
package/dist/index.umd.cjs
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
(function(
|
|
2
|
-
//# sourceMappingURL=index.umd.cjs.map
|
|
1
|
+
(function(e,t){typeof exports==`object`&&typeof module<`u`?t(exports,require("xmldoc")):typeof define==`function`&&define.amd?define([`exports`,`xmldoc`],t):(e=typeof globalThis<`u`?globalThis:e||self,t(e[`prose-reader-archive-parser`]={},e.xmldoc))})(this,function(e,t){Object.defineProperty(e,Symbol.toStringTag,{value:`Module`});var n=`com.apple.ibooks.display-options.xml`,r=e=>e.childrenNamed(`option`).map(e=>({name:e.attr?.name,value:e.val})),i=e=>{let n=new t.XmlDocument(e);if(n.name?.toLowerCase()!==`display_options`)return{kind:`apple`};let i=n.childNamed(`platform`);return{kind:`apple`,displayOptions:{...i===void 0?{}:{platform:{options:r(i)}}}}},a=[`Unknown`,`No`,`Yes`,`YesAndRightToLeft`],o=e=>{for(let t of a)if(t===e)return!0;return!1},s=`ComicInfo.xml`,c=new Set([`Pages`,`ComicInfo`]),l=e=>e.children.some(e=>e.type===`element`),u=e=>{let t=e.val.trim();return t.length>0?t:void 0},d=e=>{let n;try{n=new t.XmlDocument(e)}catch(e){let t=e instanceof Error?e.message:String(e);throw Error(`${s} is malformed: ${t}`,{cause:e})}let r={};return n.eachChild(e=>{if(e.type!==`element`||c.has(e.name)||l(e))return;let t=u(e);t!==void 0&&r[e.name]===void 0&&(r[e.name]=t)}),{kind:`comicInfo`,...r}},f=`com.kobobooks.display-options.xml`,p=e=>{let t=e.childNamed(`platform`);if(!t)return{};for(let e of t.childrenNamed(`option`))if(e.attr?.name===`fixed-layout`)return e.val.trim().toLowerCase()===`true`?{renditionLayout:`pre-paginated`}:{};return{}},m=e=>{let n=new t.XmlDocument(e);return n.name?.toLowerCase()===`display_options`?{kind:`kobo`,...p(n)}:{kind:`kobo`}},h=e=>e===void 0||e.trim().length===0?[]:e.trim().split(/\s+/).filter(e=>e.length>0),g=e=>{let t=h(e);if(t.length===0)return{};let n;t.includes(`rendition:layout-reflowable`)&&(n=`reflowable`),t.includes(`rendition:layout-pre-paginated`)&&(n=`pre-paginated`);let r;return t.includes(`rendition:flow-auto`)&&(r=`auto`),t.includes(`rendition:flow-paginated`)&&(r=`paginated`),t.includes(`rendition:flow-scrolled-doc`)&&(r=`scrolled-doc`),t.includes(`rendition:flow-scrolled-continuous`)&&(r=`scrolled-continuous`),{...n===void 0?{}:{renditionLayout:n},...r===void 0?{}:{renditionFlow:r},...t.includes(`page-spread-left`)?{pageSpreadLeft:!0}:{},...t.includes(`page-spread-right`)?{pageSpreadRight:!0}:{}}},_=e=>e.includes(`:`)?e.slice(e.lastIndexOf(`:`)+1):e,v=(e,t)=>_(e).toLowerCase()===t.toLowerCase(),y=e=>e.type===`element`,b=(e,t)=>{for(let n of e.children)if(y(n)&&v(n.name,t))return n},x=(e,t)=>{let n=[];for(let r of e.children)y(r)&&v(r.name,t)&&n.push(r);return n},S=e=>{let t=[];return e.eachChild(e=>{if(_(e.name).toLowerCase()!==`identifier`)return;let n=e.val.trim();if(n.length===0)return;let r=(e.attr[`opf:scheme`]??e.attr[`opf:Scheme`]??e.attr.scheme)?.trim();t.push({value:n,...r!==void 0&&r.length>0?{scheme:r}:{}})}),t},C=(e,t)=>{let n;return e.eachChild(e=>{if(n!==void 0||_(e.name).toLowerCase()!==t.toLowerCase())return;let r=e.val.trim();r.length>0&&(n=r)}),n},w=(e,t)=>{let n=[];return e.eachChild(e=>{if(_(e.name).toLowerCase()!==t.toLowerCase())return;let r=e.val.trim();r.length>0&&n.push(r)}),n},T=e=>{let t;return e.eachChild(e=>{if(t!==void 0||_(e.name).toLowerCase()!==`meta`||e.attr.name?.toLowerCase()!==`cover`)return;let n=e.attr.content?.trim();n!==void 0&&n.length>0&&(t=n)}),t},E=({manifestItems:e,metadataEl:t})=>{let n=e=>e.mediaType?.toLowerCase().includes(`image/`)===!0,r=e.find(e=>n(e)?h(e.properties).includes(`cover-image`):!1);if(r!==void 0)return r.href;if(t!==void 0){let r=T(t);if(r!==void 0){let t=e.find(e=>e.id===r&&n(e));if(t!==void 0)return t.href}}return e.find(e=>e.id.toLowerCase().includes(`cover`)&&n(e))?.href},D=(e,t)=>{let n=x(e,`meta`).find(e=>e.attr.property===t)?.val;if(!(n===void 0||n.trim().length===0))return n},O=e=>{let t=b(e,`guide`);if(t===void 0)return[];let n=[];for(let e of x(t,`reference`)){let t=e.attr.href?.trim();t===void 0||t.length===0||n.push({href:t,title:e.attr.title?.trim()??``,type:e.attr.type?.trim()??``})}return n},k=e=>{let t=e.attr.id,n=e.attr.href;if(t===void 0||t.length===0||n===void 0||n.length===0)return;let r=e.attr[`media-type`],i=e.attr.properties?.trim();return{id:t,href:n,...r!==void 0&&r.length>0?{mediaType:r}:{},...i!==void 0&&i.length>0?{properties:i}:{}}},A=e=>{let t=[],n=new Map;for(let r of x(e,`item`)){let e=k(r);e!==void 0&&(t.push(e),n.set(e.id,e))}return{items:t,byId:n}},j=(e,t)=>{let n=[];for(let r of x(t,`itemref`)){let t=r.attr.idref;if(t===void 0||t.trim().length===0)continue;let i=e.get(t);if(i===void 0)continue;let a=g(r.attr.properties);n.push({idref:t,id:i.id,href:i.href,...i.mediaType===void 0?{}:{mediaType:i.mediaType},...i.properties===void 0?{}:{properties:i.properties},...a.renditionLayout===void 0?{}:{renditionLayout:a.renditionLayout},...a.renditionFlow===void 0?{}:{renditionFlow:a.renditionFlow},...a.pageSpreadLeft===void 0?{}:{pageSpreadLeft:a.pageSpreadLeft},...a.pageSpreadRight===void 0?{}:{pageSpreadRight:a.pageSpreadRight}})}return n},M=e=>{let n=new t.XmlDocument(e),r=b(n,`manifest`),i=b(n,`spine`),a=b(n,`metadata`),o=[],s=[];if(r!==void 0){let{items:e,byId:t}=A(r);o=e,i!==void 0&&(s=j(t,i))}let c=i?.attr[`page-progression-direction`],l=c!==void 0&&c.trim().length>0?c:void 0,u=i?.attr.toc,d=u!==void 0&&u.trim().length>0?u.trim():void 0,f,p,m,h,g=[],_=[],v=[],y,x,T,k=[];a!==void 0&&(f=C(a,`title`),p=C(a,`publisher`),m=C(a,`rights`),h=C(a,`date`),g=w(a,`creator`),_=w(a,`language`),v=w(a,`subject`),y=D(a,`rendition:layout`),x=D(a,`rendition:flow`),T=D(a,`rendition:spread`),k.push(...S(a)));let M=E({manifestItems:o,metadataEl:a}),N=O(n);return{kind:`opf`,manifestItems:o,spineRows:s,spineTocIdref:d,identifiers:k,title:f,creators:g,publisher:p,rights:m,languages:_,subjects:v,date:h,coverHref:M,renditionLayoutMeta:y,renditionFlowMeta:x,renditionSpreadMeta:T,pageProgressionDirection:l,guide:N}},N=e=>({gtin:void 0,isbn:void 0,readingDirection:void 0,renditionLayout:(e.displayOptions?.platform?.options?.find(e=>e.name===`fixed-layout`)?.value)?.trim().toLowerCase()===`true`?`pre-paginated`:void 0,title:void 0,authors:void 0,publisher:void 0,rights:void 0,languages:void 0,date:void 0,subjects:void 0}),P=new Set([8,12,13,14]),F=e=>{if(e==null)return;let t=String(e).replace(/\D/g,``);if(!(t.length===0||!P.has(t.length)))return t},I=/(?:97[89])?\d{9}[\dXx]/,L=e=>{if(e==null)return;let t=String(e).trim().replace(/^urn:isbn:/i,``).replace(/^isbn[:\s-]*/i,``),n=t.replace(/[^0-9Xx]/g,``);if(n.length===10||n.length===13)return n.toUpperCase();let r=t.match(I);if(r)return r[0].toUpperCase()},R=e=>{switch(e.Manga){case`YesAndRightToLeft`:return`rtl`;case`Yes`:case`No`:return`ltr`;default:return}},z=e=>{if(e===void 0)return;let t=e.trim();return t.length>0?t:void 0},B=e=>e===void 0?[]:e.split(`,`).map(e=>e.trim()).filter(e=>e.length>0),V=e=>{let t=z(e);if(t===void 0||!/^\d+$/.test(t))return;let n=Number.parseInt(t,10);return Number.isFinite(n)?n:void 0},H=e=>{let t=V(e.Year),n=V(e.Month),r=V(e.Day);if(!(t===void 0&&n===void 0&&r===void 0))return{...t===void 0?{}:{year:t},...n===void 0?{}:{month:n},...r===void 0?{}:{day:r}}},U=e=>{let t=e.GTIN,n=z(e.LanguageISO),r=B(e.Writer),i=[...B(e.Genre),...B(e.Tags)];return{gtin:F(t),isbn:L(t),readingDirection:R(e),renditionLayout:void 0,title:z(e.Title),authors:r.length>0?r:void 0,publisher:z(e.Publisher),rights:void 0,languages:n===void 0?void 0:[n],date:H(e),subjects:i.length>0?i:void 0}},W=e=>({gtin:void 0,isbn:void 0,readingDirection:void 0,renditionLayout:e.renditionLayout,title:void 0,authors:void 0,publisher:void 0,rights:void 0,languages:void 0,date:void 0,subjects:void 0}),G=e=>{if(e===void 0)return;let t=e.trim().match(/^(\d{4})(?:-(\d{2})(?:-(\d{2}))?)?/);if(t===null)return;let[,n,r,i]=t;return{...n===void 0?{}:{year:Number.parseInt(n,10)},...r===void 0?{}:{month:Number.parseInt(r,10)},...i===void 0?{}:{day:Number.parseInt(i,10)}}},K=e=>{for(let t of e)if(t.scheme!==void 0&&t.scheme.trim().toLowerCase()===`isbn`&&L(t.value)!==void 0)return t.value;for(let t of e)if(L(t.value)!==void 0)return t.value},q=e=>{let t=e.pageProgressionDirection?.trim().toLowerCase(),n=t===`ltr`||t===`rtl`?t:void 0,r=e.renditionLayoutMeta?.trim().toLowerCase(),i=r===`reflowable`||r===`pre-paginated`?r:void 0,a=K(e.identifiers);return{gtin:F(a),isbn:L(a),readingDirection:n,renditionLayout:i,title:e.title,authors:e.creators.length>0?[...e.creators]:void 0,publisher:e.publisher,rights:e.rights,languages:e.languages.length>0?[...e.languages]:void 0,date:G(e.date),subjects:e.subjects.length>0?[...e.subjects]:void 0}};e.APPLE_IBOOKS_DISPLAY_OPTIONS_FILENAME=n,e.COMIC_INFO_FILENAME=s,e.COMIC_INFO_MANGA_VALUES=a,e.KOBO_DISPLAY_OPTIONS_FILENAME=f,e.isComicInfoManga=o,e.normalizeGtin=F,e.normalizeIsbn=L,e.parseAppleDisplayOptionsXml=i,e.parseComicInfo=d,e.parseKoboXml=m,e.parseOpf=M,e.parseW3cDtfDate=G,e.resolveArchiveMetadata=e=>e.kind===`comicInfo`?U(e):e.kind===`kobo`?W(e):e.kind===`apple`?N(e):q(e),e.tokenizeXmlSpaceSeparatedList=h});
|
|
2
|
+
//# sourceMappingURL=index.umd.cjs.map
|
package/dist/index.umd.cjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.umd.cjs","sources":["../src/apple/parse.ts","../src/comicInfo/manga.ts","../src/comicInfo/parse.ts","../src/kobo/parse.ts","../src/utils/tokenizeXmlSpaceSeparatedList.ts","../src/opf/spineItemrefProperties.ts","../src/opf/parse.ts","../src/apple/resolve.ts","../src/utils/normalizeGtin.ts","../src/utils/normalizeIsbn.ts","../src/comicInfo/resolve.ts","../src/kobo/resolve.ts","../src/utils/parseW3cDtfDate.ts","../src/opf/resolve.ts","../src/resolve.ts"],"sourcesContent":["import type { XmlElement } from \"xmldoc\"\nimport { XmlDocument } from \"xmldoc\"\n\nexport const APPLE_IBOOKS_DISPLAY_OPTIONS_FILENAME =\n \"com.apple.ibooks.display-options.xml\"\n\nexport type AppleDisplayOption = {\n readonly name?: string\n readonly value: string\n}\n\nexport type AppleMetadata = {\n readonly kind: \"apple\"\n readonly displayOptions?: {\n readonly platform?: {\n readonly options: ReadonlyArray<AppleDisplayOption>\n }\n }\n}\n\nconst platformOptionsFromElement = (\n platform: XmlElement,\n): ReadonlyArray<AppleDisplayOption> =>\n platform.childrenNamed(\"option\").map((option) => ({\n name: option.attr?.name,\n value: option.val,\n }))\n\nexport const parseAppleDisplayOptionsXml = (xml: string): AppleMetadata => {\n const doc = new XmlDocument(xml)\n const root = doc.name?.toLowerCase()\n\n if (root !== \"display_options\") {\n return { kind: \"apple\" }\n }\n\n const platformEl = doc.childNamed(\"platform\")\n\n return {\n kind: \"apple\",\n displayOptions: {\n ...(platformEl !== undefined\n ? { platform: { options: platformOptionsFromElement(platformEl) } }\n : {}),\n },\n }\n}\n","/**\n * Allowed `Manga` element values (ComicInfo 2.0 `Manga` simpleType).\n *\n * @see https://anansi-project.github.io/docs/comicinfo/documentation#manga\n */\nexport const COMIC_INFO_MANGA_VALUES = [\n \"Unknown\",\n \"No\",\n \"Yes\",\n \"YesAndRightToLeft\",\n] as const\n\nexport type ComicInfoManga = (typeof COMIC_INFO_MANGA_VALUES)[number]\n\nexport const isComicInfoManga = (value: string): value is ComicInfoManga => {\n for (const v of COMIC_INFO_MANGA_VALUES) {\n if (v === value) return true\n }\n return false\n}\n","import type { XmlElement } from \"xmldoc\"\nimport { XmlDocument } from \"xmldoc\"\nimport type { ComicInfoManga } from \"./manga\"\n\n/** Canonical top-level filename; real archives may use any casing. */\nexport const COMIC_INFO_FILENAME = \"ComicInfo.xml\"\n\n/**\n * Parsed `ComicInfo.xml` root: one optional string property per child element,\n * using the same names as in the file (e.g. `Title`, `GTIN`, `LanguageISO`).\n * Nested blocks such as `Pages` are skipped. Other simple child elements are\n * still copied onto this object under their tag name.\n *\n * @see https://anansi-project.github.io/docs/comicinfo/intro\n * @see https://github.com/anansi-project/comicinfo/blob/main/drafts/v2.1/ComicInfo.xsd for schema\n */\nexport interface ComicInfo {\n readonly kind: \"comicInfo\"\n AgeRating?: string\n AlternateCount?: string\n AlternateNumber?: string\n AlternateSeries?: string\n BlackAndWhite?: string\n Characters?: string\n Colorist?: string\n CommunityRating?: string\n Count?: string\n CoverArtist?: string\n Day?: string\n Editor?: string\n Format?: string\n Genre?: string\n GTIN?: string\n Imprint?: string\n Inker?: string\n LanguageISO?: string\n Letterer?: string\n Locations?: string\n MainCharacterOrTeam?: string\n /** Schema literals per {@link ComicInfoManga}; files may still use other strings. */\n Manga?: ComicInfoManga | (string & {})\n Month?: string\n Notes?: string\n Number?: string\n PageCount?: string\n Penciller?: string\n Publisher?: string\n Review?: string\n ScanInformation?: string\n Series?: string\n SeriesGroup?: string\n StoryArc?: string\n StoryArcNumber?: string\n Summary?: string\n Tags?: string\n Teams?: string\n Title?: string\n Translator?: string\n Volume?: string\n Web?: string\n Writer?: string\n Year?: string\n [tag: string]: string | undefined\n}\n\nconst SKIP_ELEMENT_CHILDREN = new Set([\"Pages\", \"ComicInfo\"])\n\nconst hasNestedElement = (el: XmlElement) =>\n el.children.some((c) => c.type === \"element\")\n\nconst trimmedText = (el: XmlElement): string | undefined => {\n const t = el.val.trim()\n return t.length > 0 ? t : undefined\n}\n\n/**\n * Parse a raw `ComicInfo.xml` body. Each direct child element with plain text\n * becomes a property named after that tag. Malformed XML throws; the parser\n * error is attached as `cause`.\n */\nexport const parseComicInfo = (xml: string): ComicInfo => {\n let doc: XmlDocument\n try {\n doc = new XmlDocument(xml)\n } catch (cause) {\n const message = cause instanceof Error ? cause.message : String(cause)\n throw new Error(`${COMIC_INFO_FILENAME} is malformed: ${message}`, {\n cause,\n })\n }\n\n const fields: Record<string, string> = {}\n\n doc.eachChild((child) => {\n if (child.type !== \"element\") return\n if (SKIP_ELEMENT_CHILDREN.has(child.name)) return\n if (hasNestedElement(child)) return\n\n const text = trimmedText(child)\n if (text === undefined) return\n if (fields[child.name] !== undefined) return\n\n fields[child.name] = text\n })\n\n // `as ComicInfo`: TS cannot infer that spreading `Record<string, string>` into `{ kind }` satisfies the named optional fields plus `[tag: string]: string | undefined` on `ComicInfo`.\n return { kind: \"comicInfo\", ...fields } as ComicInfo\n}\n","import { XmlDocument } from \"xmldoc\"\n\nexport const KOBO_DISPLAY_OPTIONS_FILENAME = \"com.kobobooks.display-options.xml\"\n\n/**\n * Normalized fields from Kobo-specific XML sidecars. Grows as new Kobo\n * document kinds are supported; absent keys mean not present or unknown.\n */\nexport type KoboMetadata = {\n readonly kind: \"kobo\"\n renditionLayout?: `reflowable` | `pre-paginated`\n}\n\nconst parseKoboDisplayOptionsDocument = (\n doc: XmlDocument,\n): { renditionLayout?: \"pre-paginated\" } => {\n const platform = doc.childNamed(\"platform\")\n if (!platform) return {}\n\n for (const option of platform.childrenNamed(\"option\")) {\n if (option.attr?.name !== \"fixed-layout\") continue\n if (option.val.trim().toLowerCase() === \"true\") {\n return { renditionLayout: \"pre-paginated\" }\n }\n return {}\n }\n\n return {}\n}\n\n/**\n * Parse a Kobo-related XML document. Unsupported or unrecognized roots\n * yield an empty object. Malformed XML throws from the underlying parser.\n */\nexport const parseKoboXml = (xml: string): KoboMetadata => {\n const doc = new XmlDocument(xml)\n const root = doc.name?.toLowerCase()\n\n if (root === \"display_options\") {\n return { kind: \"kobo\", ...parseKoboDisplayOptionsDocument(doc) }\n }\n\n return { kind: \"kobo\" }\n}\n","/**\n * EPUB/XML space-separated token lists (e.g. `properties` on `item` / `itemref`).\n * Trims the raw string, splits on ASCII whitespace, drops empty segments.\n */\nexport const tokenizeXmlSpaceSeparatedList = (\n raw: string | undefined,\n): readonly string[] => {\n if (raw === undefined || raw.trim().length === 0) {\n return []\n }\n\n return raw\n .trim()\n .split(/\\s+/)\n .filter((t) => t.length > 0)\n}\n","import { tokenizeXmlSpaceSeparatedList } from \"../utils/tokenizeXmlSpaceSeparatedList\"\n\nexport type OpfItemrefLayoutHints = {\n readonly renditionLayout?: `reflowable` | `pre-paginated`\n readonly renditionFlow?:\n | `scrolled-continuous`\n | `scrolled-doc`\n | `paginated`\n | `auto`\n readonly pageSpreadLeft?: true\n readonly pageSpreadRight?: true\n}\n\n/**\n * EPUB `itemref` `properties` attribute (space-separated tokens).\n *\n * @see https://www.w3.org/TR/epub/#attrdef-properties\n */\nexport const layoutHintsFromItemrefProperties = (\n properties: string | undefined,\n): OpfItemrefLayoutHints => {\n const tokens = tokenizeXmlSpaceSeparatedList(properties)\n if (tokens.length === 0) {\n return {}\n }\n\n let renditionLayout: `reflowable` | `pre-paginated` | undefined\n if (tokens.includes(`rendition:layout-reflowable`)) {\n renditionLayout = `reflowable`\n }\n if (tokens.includes(`rendition:layout-pre-paginated`)) {\n renditionLayout = `pre-paginated`\n }\n\n let renditionFlow: OpfItemrefLayoutHints[\"renditionFlow\"]\n if (tokens.includes(`rendition:flow-auto`)) {\n renditionFlow = `auto`\n }\n if (tokens.includes(`rendition:flow-paginated`)) {\n renditionFlow = `paginated`\n }\n if (tokens.includes(`rendition:flow-scrolled-doc`)) {\n renditionFlow = `scrolled-doc`\n }\n if (tokens.includes(`rendition:flow-scrolled-continuous`)) {\n renditionFlow = `scrolled-continuous`\n }\n\n return {\n ...(renditionLayout !== undefined ? { renditionLayout } : {}),\n ...(renditionFlow !== undefined ? { renditionFlow } : {}),\n ...(tokens.includes(`page-spread-left`)\n ? { pageSpreadLeft: true as const }\n : {}),\n ...(tokens.includes(`page-spread-right`)\n ? { pageSpreadRight: true as const }\n : {}),\n }\n}\n","import type { XmlElement, XmlNodeBase } from \"xmldoc\"\nimport { XmlDocument } from \"xmldoc\"\nimport { tokenizeXmlSpaceSeparatedList } from \"../utils/tokenizeXmlSpaceSeparatedList\"\nimport { layoutHintsFromItemrefProperties } from \"./spineItemrefProperties\"\n\nexport type OpfSpineManifestItem = {\n readonly id: string\n readonly href: string\n readonly mediaType?: string\n readonly properties?: string\n}\n\nexport type OpfIdentifier = {\n readonly value: string\n readonly scheme?: string\n}\n\nexport type OpfSpineRow = {\n readonly idref: string\n readonly id: string\n readonly href: string\n readonly mediaType?: string\n readonly properties?: string\n readonly renditionLayout?: `reflowable` | `pre-paginated`\n readonly renditionFlow?:\n | `scrolled-continuous`\n | `scrolled-doc`\n | `paginated`\n | `auto`\n readonly pageSpreadLeft?: true\n readonly pageSpreadRight?: true\n}\n\nexport type OpfGuideReference = {\n readonly href: string\n readonly title: string\n readonly type: string\n}\n\nconst elementLocalName = (name: string): string =>\n name.includes(\":\") ? name.slice(name.lastIndexOf(\":\") + 1) : name\n\nconst localNameEq = (elementName: string, wantLocal: string): boolean =>\n elementLocalName(elementName).toLowerCase() === wantLocal.toLowerCase()\n\nconst isXmlElement = (node: XmlNodeBase): node is XmlElement =>\n node.type === \"element\"\n\nconst childNamedLocal = (\n parent: XmlElement,\n localName: string,\n): XmlElement | undefined => {\n for (const node of parent.children) {\n if (!isXmlElement(node)) continue\n if (localNameEq(node.name, localName)) return node\n }\n return undefined\n}\n\nconst childrenNamedLocal = (\n parent: XmlElement,\n localName: string,\n): XmlElement[] => {\n const out: XmlElement[] = []\n for (const node of parent.children) {\n if (!isXmlElement(node)) continue\n if (localNameEq(node.name, localName)) out.push(node)\n }\n return out\n}\n\nconst identifiersFromMetadata = (metadataEl: XmlElement): OpfIdentifier[] => {\n const identifiers: OpfIdentifier[] = []\n\n metadataEl.eachChild((child) => {\n if (elementLocalName(child.name).toLowerCase() !== \"identifier\") return\n\n const value = child.val.trim()\n if (value.length === 0) return\n\n const scheme =\n child.attr[\"opf:scheme\"] ?? child.attr[\"opf:Scheme\"] ?? child.attr.scheme\n const schemeTrimmed = scheme?.trim()\n\n identifiers.push({\n value,\n ...(schemeTrimmed !== undefined && schemeTrimmed.length > 0\n ? { scheme: schemeTrimmed }\n : {}),\n })\n })\n\n return identifiers\n}\n\nconst firstTextByLocalName = (\n metadataEl: XmlElement,\n localName: string,\n): string | undefined => {\n let found: string | undefined\n metadataEl.eachChild((child) => {\n if (found !== undefined) return\n if (elementLocalName(child.name).toLowerCase() !== localName.toLowerCase())\n return\n const t = child.val.trim()\n if (t.length > 0) found = t\n })\n return found\n}\n\nconst textsByLocalName = (\n metadataEl: XmlElement,\n localName: string,\n): string[] => {\n const out: string[] = []\n metadataEl.eachChild((child) => {\n if (elementLocalName(child.name).toLowerCase() !== localName.toLowerCase())\n return\n const t = child.val.trim()\n if (t.length > 0) out.push(t)\n })\n return out\n}\n\nconst coverContentIdFromMetadata = (\n metadataEl: XmlElement,\n): string | undefined => {\n let coverId: string | undefined\n metadataEl.eachChild((child) => {\n if (coverId !== undefined) return\n if (elementLocalName(child.name).toLowerCase() !== \"meta\") return\n if (child.attr.name?.toLowerCase() !== \"cover\") return\n const content = child.attr.content?.trim()\n if (content !== undefined && content.length > 0) coverId = content\n })\n return coverId\n}\n\n/**\n * EPUB cover image inside the manifest. Resolution order, matching\n * what the spec lays out and what the bulk of EPUB producers in the\n * wild rely on:\n *\n * 1. EPUB 3 — the manifest item carrying the `cover-image` token in\n * its `properties` attribute (§ D.6.1).\n * 2. EPUB 2 — `<meta name=\"cover\" content=\"ID\"/>` in `metadata`,\n * resolved to the manifest item with that `id`.\n * 3. Last-resort fallback — any image manifest item whose `id`\n * contains the substring `cover` (case-insensitive); covers the\n * long tail of producers that emit neither the EPUB 3 property\n * nor the EPUB 2 meta.\n *\n * Each step requires the candidate manifest item to advertise an\n * `image/*` media type so non-image artefacts named `cover` (XHTML\n * cover pages, NCX entries) don't slip through.\n */\nconst coverHrefFromManifestAndMetadata = ({\n manifestItems,\n metadataEl,\n}: {\n manifestItems: ReadonlyArray<OpfSpineManifestItem>\n metadataEl: XmlElement | undefined\n}): string | undefined => {\n const isImage = (item: OpfSpineManifestItem): boolean =>\n item.mediaType?.toLowerCase().includes(\"image/\") === true\n\n const byCoverImageProperty = manifestItems.find((item) => {\n if (!isImage(item)) return false\n return tokenizeXmlSpaceSeparatedList(item.properties).includes(\n \"cover-image\",\n )\n })\n if (byCoverImageProperty !== undefined) return byCoverImageProperty.href\n\n if (metadataEl !== undefined) {\n const coverContentId = coverContentIdFromMetadata(metadataEl)\n if (coverContentId !== undefined) {\n const match = manifestItems.find(\n (item) => item.id === coverContentId && isImage(item),\n )\n if (match !== undefined) return match.href\n }\n }\n\n return manifestItems.find(\n (item) => item.id.toLowerCase().includes(\"cover\") && isImage(item),\n )?.href\n}\n\nconst metaValByProperty = (\n metadataEl: XmlElement,\n property: string,\n): string | undefined => {\n const meta = childrenNamedLocal(metadataEl, \"meta\").find(\n (m) => m.attr.property === property,\n )\n const raw = meta?.val\n if (raw === undefined || raw.trim().length === 0) return undefined\n return raw\n}\n\nconst guideFromPackage = (doc: XmlElement): OpfGuideReference[] => {\n const guideEl = childNamedLocal(doc, \"guide\")\n if (guideEl === undefined) return []\n\n const refs: OpfGuideReference[] = []\n\n for (const ref of childrenNamedLocal(guideEl, \"reference\")) {\n const href = ref.attr.href?.trim()\n if (href === undefined || href.length === 0) continue\n refs.push({\n href,\n title: ref.attr.title?.trim() ?? ``,\n type: ref.attr.type?.trim() ?? ``,\n })\n }\n\n return refs\n}\n\nconst manifestItemFromXmlElement = (\n item: XmlElement,\n): OpfSpineManifestItem | undefined => {\n const id = item.attr.id\n const href = item.attr.href\n if (id === undefined || id.length === 0) return undefined\n if (href === undefined || href.length === 0) return undefined\n\n const mediaType = item.attr[\"media-type\"]\n const properties = item.attr.properties?.trim()\n return {\n id,\n href,\n ...(mediaType !== undefined && mediaType.length > 0 ? { mediaType } : {}),\n ...(properties !== undefined && properties.length > 0\n ? { properties }\n : {}),\n }\n}\n\nconst manifestItemsAndById = (\n manifestEl: XmlElement,\n): {\n items: OpfSpineManifestItem[]\n byId: Map<string, OpfSpineManifestItem>\n} => {\n const items: OpfSpineManifestItem[] = []\n const byId = new Map<string, OpfSpineManifestItem>()\n\n for (const el of childrenNamedLocal(manifestEl, \"item\")) {\n const parsed = manifestItemFromXmlElement(el)\n if (parsed === undefined) continue\n items.push(parsed)\n byId.set(parsed.id, parsed)\n }\n\n return { items, byId }\n}\n\nconst spineRowsFromByIdAndSpine = (\n byId: Map<string, OpfSpineManifestItem>,\n spineEl: XmlElement,\n): OpfSpineRow[] => {\n const rows: OpfSpineRow[] = []\n\n for (const itemref of childrenNamedLocal(spineEl, \"itemref\")) {\n const idref = itemref.attr.idref\n if (idref === undefined || idref.trim().length === 0) continue\n\n const manifestItem = byId.get(idref)\n if (manifestItem === undefined) continue\n\n const hints = layoutHintsFromItemrefProperties(itemref.attr.properties)\n\n rows.push({\n idref,\n id: manifestItem.id,\n href: manifestItem.href,\n ...(manifestItem.mediaType !== undefined\n ? { mediaType: manifestItem.mediaType }\n : {}),\n ...(manifestItem.properties !== undefined\n ? { properties: manifestItem.properties }\n : {}),\n ...(hints.renditionLayout !== undefined\n ? { renditionLayout: hints.renditionLayout }\n : {}),\n ...(hints.renditionFlow !== undefined\n ? { renditionFlow: hints.renditionFlow }\n : {}),\n ...(hints.pageSpreadLeft !== undefined\n ? { pageSpreadLeft: hints.pageSpreadLeft }\n : {}),\n ...(hints.pageSpreadRight !== undefined\n ? { pageSpreadRight: hints.pageSpreadRight }\n : {}),\n })\n }\n\n return rows\n}\n\nexport type OpfMetadata = {\n readonly kind: \"opf\"\n readonly manifestItems: ReadonlyArray<OpfSpineManifestItem>\n readonly spineRows: ReadonlyArray<OpfSpineRow>\n readonly spineTocIdref: string | undefined\n readonly identifiers: ReadonlyArray<OpfIdentifier>\n readonly title: string | undefined\n /** `dc:creator` values, in document order, trimmed; empty when none. */\n readonly creators: ReadonlyArray<string>\n /** First non-empty `dc:publisher`, trimmed. */\n readonly publisher: string | undefined\n /** First non-empty `dc:rights`, trimmed. */\n readonly rights: string | undefined\n /** `dc:language` values, in document order, trimmed; empty when none. */\n readonly languages: ReadonlyArray<string>\n /** `dc:subject` values, in document order, trimmed; empty when none. */\n readonly subjects: ReadonlyArray<string>\n /**\n * Raw `dc:date` value as authored. EPUB 3 requires W3CDTF (a profile\n * of ISO 8601), but real-world publishers also ship free text here,\n * so the value is exposed verbatim and consumers normalize as needed.\n */\n readonly date: string | undefined\n /**\n * Manifest-relative `href` of the cover image, when one can be\n * resolved from `cover-image` properties (EPUB 3), the EPUB 2\n * `<meta name=\"cover\">` convention, or an `id` that contains\n * `cover` on an image manifest item. The href is returned exactly\n * as it appears in the manifest — callers own folder-prefix\n * resolution against the OPF's location in the archive.\n */\n readonly coverHref: string | undefined\n readonly renditionLayoutMeta: string | undefined\n readonly renditionFlowMeta: string | undefined\n readonly renditionSpreadMeta: string | undefined\n readonly pageProgressionDirection: string | undefined\n readonly guide: ReadonlyArray<OpfGuideReference>\n}\n\n/**\n * Parses an EPUB package document (OPF) into structured metadata.\n *\n * Direct children of `package` (`metadata`, `manifest`, `spine`, `guide`) and\n * their structural children (`item`, `itemref`, `reference`, `meta`) are\n * matched by **local name** (ASCII case-insensitive), so prefixed tags such as\n * `opf:manifest` are supported the same as unprefixed `manifest`.\n *\n * Attribute names on `spine` / `itemref` are still read as emitted by xmldoc\n * (no QName normalization).\n */\nexport const parseOpf = (opfXml: string): OpfMetadata => {\n const doc = new XmlDocument(opfXml)\n const manifestEl = childNamedLocal(doc, \"manifest\")\n const spineEl = childNamedLocal(doc, \"spine\")\n const metadataEl = childNamedLocal(doc, \"metadata\")\n\n let manifestItems: OpfSpineManifestItem[] = []\n let spineRows: OpfSpineRow[] = []\n\n if (manifestEl !== undefined) {\n const { items, byId } = manifestItemsAndById(manifestEl)\n manifestItems = items\n if (spineEl !== undefined) {\n spineRows = spineRowsFromByIdAndSpine(byId, spineEl)\n }\n }\n\n const pageProgressionDirectionRaw =\n spineEl?.attr[\"page-progression-direction\"]\n const pageProgressionDirection =\n pageProgressionDirectionRaw !== undefined &&\n pageProgressionDirectionRaw.trim().length > 0\n ? pageProgressionDirectionRaw\n : undefined\n\n const spineTocRaw = spineEl?.attr.toc\n const spineTocIdref =\n spineTocRaw !== undefined && spineTocRaw.trim().length > 0\n ? spineTocRaw.trim()\n : undefined\n\n let title: string | undefined\n let publisher: string | undefined\n let rights: string | undefined\n let date: string | undefined\n let creators: string[] = []\n let languages: string[] = []\n let subjects: string[] = []\n let renditionLayoutMeta: string | undefined\n let renditionFlowMeta: string | undefined\n let renditionSpreadMeta: string | undefined\n const identifiers: OpfIdentifier[] = []\n\n if (metadataEl !== undefined) {\n title = firstTextByLocalName(metadataEl, \"title\")\n publisher = firstTextByLocalName(metadataEl, \"publisher\")\n rights = firstTextByLocalName(metadataEl, \"rights\")\n date = firstTextByLocalName(metadataEl, \"date\")\n creators = textsByLocalName(metadataEl, \"creator\")\n languages = textsByLocalName(metadataEl, \"language\")\n subjects = textsByLocalName(metadataEl, \"subject\")\n renditionLayoutMeta = metaValByProperty(metadataEl, \"rendition:layout\")\n renditionFlowMeta = metaValByProperty(metadataEl, \"rendition:flow\")\n renditionSpreadMeta = metaValByProperty(metadataEl, \"rendition:spread\")\n identifiers.push(...identifiersFromMetadata(metadataEl))\n }\n\n const coverHref = coverHrefFromManifestAndMetadata({\n manifestItems,\n metadataEl,\n })\n\n const guide = guideFromPackage(doc)\n\n return {\n kind: \"opf\",\n manifestItems,\n spineRows,\n spineTocIdref,\n identifiers,\n title,\n creators,\n publisher,\n rights,\n languages,\n subjects,\n date,\n coverHref,\n renditionLayoutMeta,\n renditionFlowMeta,\n renditionSpreadMeta,\n pageProgressionDirection,\n guide,\n }\n}\n","import type { ArchiveResolveResult } from \"../types/archiveResolve\"\nimport type { AppleMetadata } from \"./parse\"\n\nexport const resolveApple = (input: AppleMetadata): ArchiveResolveResult => {\n const fixedLayout = input.displayOptions?.platform?.options?.find(\n (o) => o.name === \"fixed-layout\",\n )?.value\n\n const renditionLayout =\n fixedLayout?.trim().toLowerCase() === \"true\" ? \"pre-paginated\" : undefined\n\n return {\n gtin: undefined,\n isbn: undefined,\n readingDirection: undefined,\n renditionLayout,\n title: undefined,\n authors: undefined,\n publisher: undefined,\n rights: undefined,\n languages: undefined,\n date: undefined,\n subjects: undefined,\n }\n}\n","const GTIN_LENGTHS = new Set([8, 12, 13, 14])\n\n/**\n * Normalize a raw GTIN / EAN / UPC string to digits only when the length\n * matches a GS1 GTIN family size (8, 12, 13, or 14). Does not verify check digits.\n */\nexport const normalizeGtin = (\n raw: string | number | undefined | null,\n): string | undefined => {\n if (raw === undefined || raw === null) return undefined\n\n const digits = String(raw).replace(/\\D/g, \"\")\n if (digits.length === 0 || !GTIN_LENGTHS.has(digits.length)) return undefined\n\n return digits\n}\n","const ISBN_CANDIDATE_PATTERN = /(?:97[89])?\\d{9}[\\dXx]/\n\n/**\n * Normalize a raw ISBN-ish string into a canonical 10- or 13-character\n * form, or `undefined` when no recognisable ISBN can be recovered.\n *\n * - Strips the common `urn:isbn:` / `isbn:` prefixes.\n * - Drops everything that isn't a digit or `X`.\n * - Validates the resulting length (10 or 13).\n * - Falls back to a lax regex scan so publishers that stuff free text\n * around the number still yield a usable value.\n */\nexport const normalizeIsbn = (\n raw: string | number | undefined | null,\n): string | undefined => {\n if (raw === undefined || raw === null) return undefined\n\n const stripped = String(raw)\n .trim()\n .replace(/^urn:isbn:/i, \"\")\n .replace(/^isbn[:\\s-]*/i, \"\")\n\n const digitsOnly = stripped.replace(/[^0-9Xx]/g, \"\")\n\n if (digitsOnly.length === 10 || digitsOnly.length === 13) {\n return digitsOnly.toUpperCase()\n }\n\n const match = stripped.match(ISBN_CANDIDATE_PATTERN)\n if (match) return match[0].toUpperCase()\n\n return undefined\n}\n","import type { ArchiveResolveResult } from \"../types/archiveResolve\"\nimport { normalizeGtin } from \"../utils/normalizeGtin\"\nimport { normalizeIsbn } from \"../utils/normalizeIsbn\"\nimport type { ComicInfo } from \"./parse\"\n\nconst readingDirection = (info: ComicInfo): \"ltr\" | \"rtl\" | undefined => {\n switch (info.Manga) {\n case \"YesAndRightToLeft\":\n return \"rtl\"\n case \"Yes\":\n case \"No\":\n return \"ltr\"\n default:\n return undefined\n }\n}\n\nconst trimToUndefined = (raw: string | undefined): string | undefined => {\n if (raw === undefined) return undefined\n const trimmed = raw.trim()\n return trimmed.length > 0 ? trimmed : undefined\n}\n\n/**\n * Split a comma-separated ComicInfo value into its individual tokens.\n * `Writer`, `Genre`, and `Tags` all follow the same de-facto convention:\n * tokens separated by `,`, with whitespace trimmed and empty tokens\n * dropped (real-world files leave trailing commas around).\n */\nconst splitCommaList = (raw: string | undefined): string[] => {\n if (raw === undefined) return []\n return raw\n .split(\",\")\n .map((token) => token.trim())\n .filter((token) => token.length > 0)\n}\n\nconst parseNonNegativeInt = (raw: string | undefined): number | undefined => {\n const trimmed = trimToUndefined(raw)\n if (trimmed === undefined) return undefined\n if (!/^\\d+$/.test(trimmed)) return undefined\n const n = Number.parseInt(trimmed, 10)\n return Number.isFinite(n) ? n : undefined\n}\n\nconst dateFromYearMonthDay = (\n info: ComicInfo,\n):\n | {\n year?: number\n month?: number\n day?: number\n }\n | undefined => {\n const year = parseNonNegativeInt(info.Year)\n const month = parseNonNegativeInt(info.Month)\n const day = parseNonNegativeInt(info.Day)\n\n if (year === undefined && month === undefined && day === undefined) {\n return undefined\n }\n\n return {\n ...(year !== undefined ? { year } : {}),\n ...(month !== undefined ? { month } : {}),\n ...(day !== undefined ? { day } : {}),\n }\n}\n\nexport const resolveComicInfo = (info: ComicInfo): ArchiveResolveResult => {\n const raw = info.GTIN\n const languageIso = trimToUndefined(info.LanguageISO)\n const authors = splitCommaList(info.Writer)\n const subjects = [...splitCommaList(info.Genre), ...splitCommaList(info.Tags)]\n\n return {\n gtin: normalizeGtin(raw),\n isbn: normalizeIsbn(raw),\n readingDirection: readingDirection(info),\n renditionLayout: undefined,\n title: trimToUndefined(info.Title),\n authors: authors.length > 0 ? authors : undefined,\n publisher: trimToUndefined(info.Publisher),\n rights: undefined,\n languages: languageIso !== undefined ? [languageIso] : undefined,\n date: dateFromYearMonthDay(info),\n subjects: subjects.length > 0 ? subjects : undefined,\n }\n}\n","import type { ArchiveResolveResult } from \"../types/archiveResolve\"\nimport type { KoboMetadata } from \"./parse\"\n\nexport const resolveKobo = (input: KoboMetadata): ArchiveResolveResult => ({\n gtin: undefined,\n isbn: undefined,\n readingDirection: undefined,\n renditionLayout: input.renditionLayout,\n title: undefined,\n authors: undefined,\n publisher: undefined,\n rights: undefined,\n languages: undefined,\n date: undefined,\n subjects: undefined,\n})\n","/**\n * Extract the calendar components of a W3CDTF literal — the\n * date subset of ISO 8601 that EPUB 3.3 § 5.5.3.2.4 mandates for\n * `dc:date`. Accepted forms: `YYYY`, `YYYY-MM`, `YYYY-MM-DD`, and\n * any of the above followed by a `Thh:mm[:ss[.s]][TZD]` time\n * portion that we ignore.\n *\n * Components are returned as plain integers (`month` is 1-12,\n * `day` is 1-31), independent of the host timezone — using\n * `Date.parse` would shift `2011-01-01` by a day in negative-offset\n * locales, which is the exact bug this regex-based approach exists\n * to avoid. Returns `undefined` when the input doesn't even match\n * a leading 4-digit year so consumers can fall back without\n * branching on partial shapes.\n */\nexport const parseW3cDtfDate = (\n raw: string | undefined,\n):\n | {\n year?: number\n month?: number\n day?: number\n }\n | undefined => {\n if (raw === undefined) return undefined\n\n const match = raw.trim().match(/^(\\d{4})(?:-(\\d{2})(?:-(\\d{2}))?)?/)\n if (match === null) return undefined\n\n const [, yearRaw, monthRaw, dayRaw] = match\n\n return {\n ...(yearRaw !== undefined ? { year: Number.parseInt(yearRaw, 10) } : {}),\n ...(monthRaw !== undefined ? { month: Number.parseInt(monthRaw, 10) } : {}),\n ...(dayRaw !== undefined ? { day: Number.parseInt(dayRaw, 10) } : {}),\n }\n}\n","import type { ArchiveResolveResult } from \"../types/archiveResolve\"\nimport { normalizeGtin } from \"../utils/normalizeGtin\"\nimport { normalizeIsbn } from \"../utils/normalizeIsbn\"\nimport { parseW3cDtfDate } from \"../utils/parseW3cDtfDate\"\nimport type { OpfIdentifier, OpfMetadata } from \"./parse\"\n\nconst rawIdentifierValueForIsbn = (\n identifiers: ReadonlyArray<OpfIdentifier>,\n): string | undefined => {\n for (const i of identifiers) {\n if (i.scheme !== undefined && i.scheme.trim().toLowerCase() === \"isbn\") {\n if (normalizeIsbn(i.value) !== undefined) return i.value\n }\n }\n\n for (const i of identifiers) {\n if (normalizeIsbn(i.value) !== undefined) return i.value\n }\n\n return undefined\n}\n\nexport const resolveOpf = (input: OpfMetadata): ArchiveResolveResult => {\n const ppd = input.pageProgressionDirection?.trim().toLowerCase()\n const readingDirection = ppd === \"ltr\" || ppd === \"rtl\" ? ppd : undefined\n\n const rl = input.renditionLayoutMeta?.trim().toLowerCase()\n const renditionLayout =\n rl === \"reflowable\" || rl === \"pre-paginated\" ? rl : undefined\n\n const raw = rawIdentifierValueForIsbn(input.identifiers)\n\n return {\n gtin: normalizeGtin(raw),\n isbn: normalizeIsbn(raw),\n readingDirection,\n renditionLayout,\n title: input.title,\n authors: input.creators.length > 0 ? [...input.creators] : undefined,\n publisher: input.publisher,\n rights: input.rights,\n languages: input.languages.length > 0 ? [...input.languages] : undefined,\n date: parseW3cDtfDate(input.date),\n subjects: input.subjects.length > 0 ? [...input.subjects] : undefined,\n }\n}\n","import type { AppleMetadata } from \"./apple/parse\"\nimport { resolveApple } from \"./apple/resolve\"\nimport type { ComicInfo } from \"./comicInfo/parse\"\nimport { resolveComicInfo } from \"./comicInfo/resolve\"\nimport type { KoboMetadata } from \"./kobo/parse\"\nimport { resolveKobo } from \"./kobo/resolve\"\nimport type { OpfMetadata } from \"./opf/parse\"\nimport { resolveOpf } from \"./opf/resolve\"\nimport type { ArchiveResolveResult } from \"./types/archiveResolve\"\n\nexport type ResolvedArchiveInput =\n | ComicInfo\n | KoboMetadata\n | AppleMetadata\n | OpfMetadata\n\nexport const resolveArchiveMetadata = (\n input: ResolvedArchiveInput,\n): ArchiveResolveResult => {\n if (input.kind === \"comicInfo\") return resolveComicInfo(input)\n if (input.kind === \"kobo\") return resolveKobo(input)\n if (input.kind === \"apple\") return resolveApple(input)\n return resolveOpf(input)\n}\n"],"names":["APPLE_IBOOKS_DISPLAY_OPTIONS_FILENAME","platformOptionsFromElement","platform","option","parseAppleDisplayOptionsXml","xml","doc","XmlDocument","platformEl","COMIC_INFO_MANGA_VALUES","isComicInfoManga","value","v","COMIC_INFO_FILENAME","SKIP_ELEMENT_CHILDREN","hasNestedElement","el","c","trimmedText","parseComicInfo","cause","message","fields","child","text","KOBO_DISPLAY_OPTIONS_FILENAME","parseKoboDisplayOptionsDocument","parseKoboXml","tokenizeXmlSpaceSeparatedList","raw","layoutHintsFromItemrefProperties","properties","tokens","renditionLayout","renditionFlow","elementLocalName","name","localNameEq","elementName","wantLocal","isXmlElement","node","childNamedLocal","parent","localName","childrenNamedLocal","out","identifiersFromMetadata","metadataEl","identifiers","schemeTrimmed","firstTextByLocalName","found","t","textsByLocalName","coverContentIdFromMetadata","coverId","content","coverHrefFromManifestAndMetadata","manifestItems","isImage","item","byCoverImageProperty","coverContentId","match","metaValByProperty","property","m","guideFromPackage","guideEl","refs","ref","href","manifestItemFromXmlElement","id","mediaType","manifestItemsAndById","manifestEl","items","byId","parsed","spineRowsFromByIdAndSpine","spineEl","rows","itemref","idref","manifestItem","hints","parseOpf","opfXml","spineRows","pageProgressionDirectionRaw","pageProgressionDirection","spineTocRaw","spineTocIdref","title","publisher","rights","date","creators","languages","subjects","renditionLayoutMeta","renditionFlowMeta","renditionSpreadMeta","coverHref","guide","resolveApple","input","GTIN_LENGTHS","normalizeGtin","digits","ISBN_CANDIDATE_PATTERN","normalizeIsbn","stripped","digitsOnly","readingDirection","info","trimToUndefined","trimmed","splitCommaList","token","parseNonNegativeInt","dateFromYearMonthDay","year","month","day","resolveComicInfo","languageIso","authors","resolveKobo","parseW3cDtfDate","yearRaw","monthRaw","dayRaw","rawIdentifierValueForIsbn","i","resolveOpf","ppd","rl","resolveArchiveMetadata"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"index.umd.cjs","names":[],"sources":["../src/apple/parse.ts","../src/comicInfo/manga.ts","../src/comicInfo/parse.ts","../src/kobo/parse.ts","../src/utils/tokenizeXmlSpaceSeparatedList.ts","../src/opf/spineItemrefProperties.ts","../src/opf/parse.ts","../src/apple/resolve.ts","../src/utils/normalizeGtin.ts","../src/utils/normalizeIsbn.ts","../src/comicInfo/resolve.ts","../src/kobo/resolve.ts","../src/utils/parseW3cDtfDate.ts","../src/opf/resolve.ts","../src/resolve.ts"],"sourcesContent":["import type { XmlElement } from \"xmldoc\"\nimport { XmlDocument } from \"xmldoc\"\n\nexport const APPLE_IBOOKS_DISPLAY_OPTIONS_FILENAME =\n \"com.apple.ibooks.display-options.xml\"\n\nexport type AppleDisplayOption = {\n readonly name?: string\n readonly value: string\n}\n\nexport type AppleMetadata = {\n readonly kind: \"apple\"\n readonly displayOptions?: {\n readonly platform?: {\n readonly options: ReadonlyArray<AppleDisplayOption>\n }\n }\n}\n\nconst platformOptionsFromElement = (\n platform: XmlElement,\n): ReadonlyArray<AppleDisplayOption> =>\n platform.childrenNamed(\"option\").map((option) => ({\n name: option.attr?.name,\n value: option.val,\n }))\n\nexport const parseAppleDisplayOptionsXml = (xml: string): AppleMetadata => {\n const doc = new XmlDocument(xml)\n const root = doc.name?.toLowerCase()\n\n if (root !== \"display_options\") {\n return { kind: \"apple\" }\n }\n\n const platformEl = doc.childNamed(\"platform\")\n\n return {\n kind: \"apple\",\n displayOptions: {\n ...(platformEl !== undefined\n ? { platform: { options: platformOptionsFromElement(platformEl) } }\n : {}),\n },\n }\n}\n","/**\n * Allowed `Manga` element values (ComicInfo 2.0 `Manga` simpleType).\n *\n * @see https://anansi-project.github.io/docs/comicinfo/documentation#manga\n */\nexport const COMIC_INFO_MANGA_VALUES = [\n \"Unknown\",\n \"No\",\n \"Yes\",\n \"YesAndRightToLeft\",\n] as const\n\nexport type ComicInfoManga = (typeof COMIC_INFO_MANGA_VALUES)[number]\n\nexport const isComicInfoManga = (value: string): value is ComicInfoManga => {\n for (const v of COMIC_INFO_MANGA_VALUES) {\n if (v === value) return true\n }\n return false\n}\n","import type { XmlElement } from \"xmldoc\"\nimport { XmlDocument } from \"xmldoc\"\nimport type { ComicInfoManga } from \"./manga\"\n\n/** Canonical top-level filename; real archives may use any casing. */\nexport const COMIC_INFO_FILENAME = \"ComicInfo.xml\"\n\n/**\n * Parsed `ComicInfo.xml` root: one optional string property per child element,\n * using the same names as in the file (e.g. `Title`, `GTIN`, `LanguageISO`).\n * Nested blocks such as `Pages` are skipped. Other simple child elements are\n * still copied onto this object under their tag name.\n *\n * @see https://anansi-project.github.io/docs/comicinfo/intro\n * @see https://github.com/anansi-project/comicinfo/blob/main/drafts/v2.1/ComicInfo.xsd for schema\n */\nexport interface ComicInfo {\n readonly kind: \"comicInfo\"\n AgeRating?: string\n AlternateCount?: string\n AlternateNumber?: string\n AlternateSeries?: string\n BlackAndWhite?: string\n Characters?: string\n Colorist?: string\n CommunityRating?: string\n Count?: string\n CoverArtist?: string\n Day?: string\n Editor?: string\n Format?: string\n Genre?: string\n GTIN?: string\n Imprint?: string\n Inker?: string\n LanguageISO?: string\n Letterer?: string\n Locations?: string\n MainCharacterOrTeam?: string\n /** Schema literals per {@link ComicInfoManga}; files may still use other strings. */\n Manga?: ComicInfoManga | (string & {})\n Month?: string\n Notes?: string\n Number?: string\n PageCount?: string\n Penciller?: string\n Publisher?: string\n Review?: string\n ScanInformation?: string\n Series?: string\n SeriesGroup?: string\n StoryArc?: string\n StoryArcNumber?: string\n Summary?: string\n Tags?: string\n Teams?: string\n Title?: string\n Translator?: string\n Volume?: string\n Web?: string\n Writer?: string\n Year?: string\n [tag: string]: string | undefined\n}\n\nconst SKIP_ELEMENT_CHILDREN = new Set([\"Pages\", \"ComicInfo\"])\n\nconst hasNestedElement = (el: XmlElement) =>\n el.children.some((c) => c.type === \"element\")\n\nconst trimmedText = (el: XmlElement): string | undefined => {\n const t = el.val.trim()\n return t.length > 0 ? t : undefined\n}\n\n/**\n * Parse a raw `ComicInfo.xml` body. Each direct child element with plain text\n * becomes a property named after that tag. Malformed XML throws; the parser\n * error is attached as `cause`.\n */\nexport const parseComicInfo = (xml: string): ComicInfo => {\n let doc: XmlDocument\n try {\n doc = new XmlDocument(xml)\n } catch (cause) {\n const message = cause instanceof Error ? cause.message : String(cause)\n throw new Error(`${COMIC_INFO_FILENAME} is malformed: ${message}`, {\n cause,\n })\n }\n\n const fields: Record<string, string> = {}\n\n doc.eachChild((child) => {\n if (child.type !== \"element\") return\n if (SKIP_ELEMENT_CHILDREN.has(child.name)) return\n if (hasNestedElement(child)) return\n\n const text = trimmedText(child)\n if (text === undefined) return\n if (fields[child.name] !== undefined) return\n\n fields[child.name] = text\n })\n\n // `as ComicInfo`: TS cannot infer that spreading `Record<string, string>` into `{ kind }` satisfies the named optional fields plus `[tag: string]: string | undefined` on `ComicInfo`.\n return { kind: \"comicInfo\", ...fields } as ComicInfo\n}\n","import { XmlDocument } from \"xmldoc\"\n\nexport const KOBO_DISPLAY_OPTIONS_FILENAME = \"com.kobobooks.display-options.xml\"\n\n/**\n * Normalized fields from Kobo-specific XML sidecars. Grows as new Kobo\n * document kinds are supported; absent keys mean not present or unknown.\n */\nexport type KoboMetadata = {\n readonly kind: \"kobo\"\n renditionLayout?: `reflowable` | `pre-paginated`\n}\n\nconst parseKoboDisplayOptionsDocument = (\n doc: XmlDocument,\n): { renditionLayout?: \"pre-paginated\" } => {\n const platform = doc.childNamed(\"platform\")\n if (!platform) return {}\n\n for (const option of platform.childrenNamed(\"option\")) {\n if (option.attr?.name !== \"fixed-layout\") continue\n if (option.val.trim().toLowerCase() === \"true\") {\n return { renditionLayout: \"pre-paginated\" }\n }\n return {}\n }\n\n return {}\n}\n\n/**\n * Parse a Kobo-related XML document. Unsupported or unrecognized roots\n * yield an empty object. Malformed XML throws from the underlying parser.\n */\nexport const parseKoboXml = (xml: string): KoboMetadata => {\n const doc = new XmlDocument(xml)\n const root = doc.name?.toLowerCase()\n\n if (root === \"display_options\") {\n return { kind: \"kobo\", ...parseKoboDisplayOptionsDocument(doc) }\n }\n\n return { kind: \"kobo\" }\n}\n","/**\n * EPUB/XML space-separated token lists (e.g. `properties` on `item` / `itemref`).\n * Trims the raw string, splits on ASCII whitespace, drops empty segments.\n */\nexport const tokenizeXmlSpaceSeparatedList = (\n raw: string | undefined,\n): readonly string[] => {\n if (raw === undefined || raw.trim().length === 0) {\n return []\n }\n\n return raw\n .trim()\n .split(/\\s+/)\n .filter((t) => t.length > 0)\n}\n","import { tokenizeXmlSpaceSeparatedList } from \"../utils/tokenizeXmlSpaceSeparatedList\"\n\nexport type OpfItemrefLayoutHints = {\n readonly renditionLayout?: `reflowable` | `pre-paginated`\n readonly renditionFlow?:\n | `scrolled-continuous`\n | `scrolled-doc`\n | `paginated`\n | `auto`\n readonly pageSpreadLeft?: true\n readonly pageSpreadRight?: true\n}\n\n/**\n * EPUB `itemref` `properties` attribute (space-separated tokens).\n *\n * @see https://www.w3.org/TR/epub/#attrdef-properties\n */\nexport const layoutHintsFromItemrefProperties = (\n properties: string | undefined,\n): OpfItemrefLayoutHints => {\n const tokens = tokenizeXmlSpaceSeparatedList(properties)\n if (tokens.length === 0) {\n return {}\n }\n\n let renditionLayout: `reflowable` | `pre-paginated` | undefined\n if (tokens.includes(`rendition:layout-reflowable`)) {\n renditionLayout = `reflowable`\n }\n if (tokens.includes(`rendition:layout-pre-paginated`)) {\n renditionLayout = `pre-paginated`\n }\n\n let renditionFlow: OpfItemrefLayoutHints[\"renditionFlow\"]\n if (tokens.includes(`rendition:flow-auto`)) {\n renditionFlow = `auto`\n }\n if (tokens.includes(`rendition:flow-paginated`)) {\n renditionFlow = `paginated`\n }\n if (tokens.includes(`rendition:flow-scrolled-doc`)) {\n renditionFlow = `scrolled-doc`\n }\n if (tokens.includes(`rendition:flow-scrolled-continuous`)) {\n renditionFlow = `scrolled-continuous`\n }\n\n return {\n ...(renditionLayout !== undefined ? { renditionLayout } : {}),\n ...(renditionFlow !== undefined ? { renditionFlow } : {}),\n ...(tokens.includes(`page-spread-left`)\n ? { pageSpreadLeft: true as const }\n : {}),\n ...(tokens.includes(`page-spread-right`)\n ? { pageSpreadRight: true as const }\n : {}),\n }\n}\n","import type { XmlElement, XmlNodeBase } from \"xmldoc\"\nimport { XmlDocument } from \"xmldoc\"\nimport { tokenizeXmlSpaceSeparatedList } from \"../utils/tokenizeXmlSpaceSeparatedList\"\nimport { layoutHintsFromItemrefProperties } from \"./spineItemrefProperties\"\n\nexport type OpfSpineManifestItem = {\n readonly id: string\n readonly href: string\n readonly mediaType?: string\n readonly properties?: string\n}\n\nexport type OpfIdentifier = {\n readonly value: string\n readonly scheme?: string\n}\n\nexport type OpfSpineRow = {\n readonly idref: string\n readonly id: string\n readonly href: string\n readonly mediaType?: string\n readonly properties?: string\n readonly renditionLayout?: `reflowable` | `pre-paginated`\n readonly renditionFlow?:\n | `scrolled-continuous`\n | `scrolled-doc`\n | `paginated`\n | `auto`\n readonly pageSpreadLeft?: true\n readonly pageSpreadRight?: true\n}\n\nexport type OpfGuideReference = {\n readonly href: string\n readonly title: string\n readonly type: string\n}\n\nconst elementLocalName = (name: string): string =>\n name.includes(\":\") ? name.slice(name.lastIndexOf(\":\") + 1) : name\n\nconst localNameEq = (elementName: string, wantLocal: string): boolean =>\n elementLocalName(elementName).toLowerCase() === wantLocal.toLowerCase()\n\nconst isXmlElement = (node: XmlNodeBase): node is XmlElement =>\n node.type === \"element\"\n\nconst childNamedLocal = (\n parent: XmlElement,\n localName: string,\n): XmlElement | undefined => {\n for (const node of parent.children) {\n if (!isXmlElement(node)) continue\n if (localNameEq(node.name, localName)) return node\n }\n return undefined\n}\n\nconst childrenNamedLocal = (\n parent: XmlElement,\n localName: string,\n): XmlElement[] => {\n const out: XmlElement[] = []\n for (const node of parent.children) {\n if (!isXmlElement(node)) continue\n if (localNameEq(node.name, localName)) out.push(node)\n }\n return out\n}\n\nconst identifiersFromMetadata = (metadataEl: XmlElement): OpfIdentifier[] => {\n const identifiers: OpfIdentifier[] = []\n\n metadataEl.eachChild((child) => {\n if (elementLocalName(child.name).toLowerCase() !== \"identifier\") return\n\n const value = child.val.trim()\n if (value.length === 0) return\n\n const scheme =\n child.attr[\"opf:scheme\"] ?? child.attr[\"opf:Scheme\"] ?? child.attr.scheme\n const schemeTrimmed = scheme?.trim()\n\n identifiers.push({\n value,\n ...(schemeTrimmed !== undefined && schemeTrimmed.length > 0\n ? { scheme: schemeTrimmed }\n : {}),\n })\n })\n\n return identifiers\n}\n\nconst firstTextByLocalName = (\n metadataEl: XmlElement,\n localName: string,\n): string | undefined => {\n let found: string | undefined\n metadataEl.eachChild((child) => {\n if (found !== undefined) return\n if (elementLocalName(child.name).toLowerCase() !== localName.toLowerCase())\n return\n const t = child.val.trim()\n if (t.length > 0) found = t\n })\n return found\n}\n\nconst textsByLocalName = (\n metadataEl: XmlElement,\n localName: string,\n): string[] => {\n const out: string[] = []\n metadataEl.eachChild((child) => {\n if (elementLocalName(child.name).toLowerCase() !== localName.toLowerCase())\n return\n const t = child.val.trim()\n if (t.length > 0) out.push(t)\n })\n return out\n}\n\nconst coverContentIdFromMetadata = (\n metadataEl: XmlElement,\n): string | undefined => {\n let coverId: string | undefined\n metadataEl.eachChild((child) => {\n if (coverId !== undefined) return\n if (elementLocalName(child.name).toLowerCase() !== \"meta\") return\n if (child.attr.name?.toLowerCase() !== \"cover\") return\n const content = child.attr.content?.trim()\n if (content !== undefined && content.length > 0) coverId = content\n })\n return coverId\n}\n\n/**\n * EPUB cover image inside the manifest. Resolution order, matching\n * what the spec lays out and what the bulk of EPUB producers in the\n * wild rely on:\n *\n * 1. EPUB 3 — the manifest item carrying the `cover-image` token in\n * its `properties` attribute (§ D.6.1).\n * 2. EPUB 2 — `<meta name=\"cover\" content=\"ID\"/>` in `metadata`,\n * resolved to the manifest item with that `id`.\n * 3. Last-resort fallback — any image manifest item whose `id`\n * contains the substring `cover` (case-insensitive); covers the\n * long tail of producers that emit neither the EPUB 3 property\n * nor the EPUB 2 meta.\n *\n * Each step requires the candidate manifest item to advertise an\n * `image/*` media type so non-image artefacts named `cover` (XHTML\n * cover pages, NCX entries) don't slip through.\n */\nconst coverHrefFromManifestAndMetadata = ({\n manifestItems,\n metadataEl,\n}: {\n manifestItems: ReadonlyArray<OpfSpineManifestItem>\n metadataEl: XmlElement | undefined\n}): string | undefined => {\n const isImage = (item: OpfSpineManifestItem): boolean =>\n item.mediaType?.toLowerCase().includes(\"image/\") === true\n\n const byCoverImageProperty = manifestItems.find((item) => {\n if (!isImage(item)) return false\n return tokenizeXmlSpaceSeparatedList(item.properties).includes(\n \"cover-image\",\n )\n })\n if (byCoverImageProperty !== undefined) return byCoverImageProperty.href\n\n if (metadataEl !== undefined) {\n const coverContentId = coverContentIdFromMetadata(metadataEl)\n if (coverContentId !== undefined) {\n const match = manifestItems.find(\n (item) => item.id === coverContentId && isImage(item),\n )\n if (match !== undefined) return match.href\n }\n }\n\n return manifestItems.find(\n (item) => item.id.toLowerCase().includes(\"cover\") && isImage(item),\n )?.href\n}\n\nconst metaValByProperty = (\n metadataEl: XmlElement,\n property: string,\n): string | undefined => {\n const meta = childrenNamedLocal(metadataEl, \"meta\").find(\n (m) => m.attr.property === property,\n )\n const raw = meta?.val\n if (raw === undefined || raw.trim().length === 0) return undefined\n return raw\n}\n\nconst guideFromPackage = (doc: XmlElement): OpfGuideReference[] => {\n const guideEl = childNamedLocal(doc, \"guide\")\n if (guideEl === undefined) return []\n\n const refs: OpfGuideReference[] = []\n\n for (const ref of childrenNamedLocal(guideEl, \"reference\")) {\n const href = ref.attr.href?.trim()\n if (href === undefined || href.length === 0) continue\n refs.push({\n href,\n title: ref.attr.title?.trim() ?? ``,\n type: ref.attr.type?.trim() ?? ``,\n })\n }\n\n return refs\n}\n\nconst manifestItemFromXmlElement = (\n item: XmlElement,\n): OpfSpineManifestItem | undefined => {\n const id = item.attr.id\n const href = item.attr.href\n if (id === undefined || id.length === 0) return undefined\n if (href === undefined || href.length === 0) return undefined\n\n const mediaType = item.attr[\"media-type\"]\n const properties = item.attr.properties?.trim()\n return {\n id,\n href,\n ...(mediaType !== undefined && mediaType.length > 0 ? { mediaType } : {}),\n ...(properties !== undefined && properties.length > 0\n ? { properties }\n : {}),\n }\n}\n\nconst manifestItemsAndById = (\n manifestEl: XmlElement,\n): {\n items: OpfSpineManifestItem[]\n byId: Map<string, OpfSpineManifestItem>\n} => {\n const items: OpfSpineManifestItem[] = []\n const byId = new Map<string, OpfSpineManifestItem>()\n\n for (const el of childrenNamedLocal(manifestEl, \"item\")) {\n const parsed = manifestItemFromXmlElement(el)\n if (parsed === undefined) continue\n items.push(parsed)\n byId.set(parsed.id, parsed)\n }\n\n return { items, byId }\n}\n\nconst spineRowsFromByIdAndSpine = (\n byId: Map<string, OpfSpineManifestItem>,\n spineEl: XmlElement,\n): OpfSpineRow[] => {\n const rows: OpfSpineRow[] = []\n\n for (const itemref of childrenNamedLocal(spineEl, \"itemref\")) {\n const idref = itemref.attr.idref\n if (idref === undefined || idref.trim().length === 0) continue\n\n const manifestItem = byId.get(idref)\n if (manifestItem === undefined) continue\n\n const hints = layoutHintsFromItemrefProperties(itemref.attr.properties)\n\n rows.push({\n idref,\n id: manifestItem.id,\n href: manifestItem.href,\n ...(manifestItem.mediaType !== undefined\n ? { mediaType: manifestItem.mediaType }\n : {}),\n ...(manifestItem.properties !== undefined\n ? { properties: manifestItem.properties }\n : {}),\n ...(hints.renditionLayout !== undefined\n ? { renditionLayout: hints.renditionLayout }\n : {}),\n ...(hints.renditionFlow !== undefined\n ? { renditionFlow: hints.renditionFlow }\n : {}),\n ...(hints.pageSpreadLeft !== undefined\n ? { pageSpreadLeft: hints.pageSpreadLeft }\n : {}),\n ...(hints.pageSpreadRight !== undefined\n ? { pageSpreadRight: hints.pageSpreadRight }\n : {}),\n })\n }\n\n return rows\n}\n\nexport type OpfMetadata = {\n readonly kind: \"opf\"\n readonly manifestItems: ReadonlyArray<OpfSpineManifestItem>\n readonly spineRows: ReadonlyArray<OpfSpineRow>\n readonly spineTocIdref: string | undefined\n readonly identifiers: ReadonlyArray<OpfIdentifier>\n readonly title: string | undefined\n /** `dc:creator` values, in document order, trimmed; empty when none. */\n readonly creators: ReadonlyArray<string>\n /** First non-empty `dc:publisher`, trimmed. */\n readonly publisher: string | undefined\n /** First non-empty `dc:rights`, trimmed. */\n readonly rights: string | undefined\n /** `dc:language` values, in document order, trimmed; empty when none. */\n readonly languages: ReadonlyArray<string>\n /** `dc:subject` values, in document order, trimmed; empty when none. */\n readonly subjects: ReadonlyArray<string>\n /**\n * Raw `dc:date` value as authored. EPUB 3 requires W3CDTF (a profile\n * of ISO 8601), but real-world publishers also ship free text here,\n * so the value is exposed verbatim and consumers normalize as needed.\n */\n readonly date: string | undefined\n /**\n * Manifest-relative `href` of the cover image, when one can be\n * resolved from `cover-image` properties (EPUB 3), the EPUB 2\n * `<meta name=\"cover\">` convention, or an `id` that contains\n * `cover` on an image manifest item. The href is returned exactly\n * as it appears in the manifest — callers own folder-prefix\n * resolution against the OPF's location in the archive.\n */\n readonly coverHref: string | undefined\n readonly renditionLayoutMeta: string | undefined\n readonly renditionFlowMeta: string | undefined\n readonly renditionSpreadMeta: string | undefined\n readonly pageProgressionDirection: string | undefined\n readonly guide: ReadonlyArray<OpfGuideReference>\n}\n\n/**\n * Parses an EPUB package document (OPF) into structured metadata.\n *\n * Direct children of `package` (`metadata`, `manifest`, `spine`, `guide`) and\n * their structural children (`item`, `itemref`, `reference`, `meta`) are\n * matched by **local name** (ASCII case-insensitive), so prefixed tags such as\n * `opf:manifest` are supported the same as unprefixed `manifest`.\n *\n * Attribute names on `spine` / `itemref` are still read as emitted by xmldoc\n * (no QName normalization).\n */\nexport const parseOpf = (opfXml: string): OpfMetadata => {\n const doc = new XmlDocument(opfXml)\n const manifestEl = childNamedLocal(doc, \"manifest\")\n const spineEl = childNamedLocal(doc, \"spine\")\n const metadataEl = childNamedLocal(doc, \"metadata\")\n\n let manifestItems: OpfSpineManifestItem[] = []\n let spineRows: OpfSpineRow[] = []\n\n if (manifestEl !== undefined) {\n const { items, byId } = manifestItemsAndById(manifestEl)\n manifestItems = items\n if (spineEl !== undefined) {\n spineRows = spineRowsFromByIdAndSpine(byId, spineEl)\n }\n }\n\n const pageProgressionDirectionRaw =\n spineEl?.attr[\"page-progression-direction\"]\n const pageProgressionDirection =\n pageProgressionDirectionRaw !== undefined &&\n pageProgressionDirectionRaw.trim().length > 0\n ? pageProgressionDirectionRaw\n : undefined\n\n const spineTocRaw = spineEl?.attr.toc\n const spineTocIdref =\n spineTocRaw !== undefined && spineTocRaw.trim().length > 0\n ? spineTocRaw.trim()\n : undefined\n\n let title: string | undefined\n let publisher: string | undefined\n let rights: string | undefined\n let date: string | undefined\n let creators: string[] = []\n let languages: string[] = []\n let subjects: string[] = []\n let renditionLayoutMeta: string | undefined\n let renditionFlowMeta: string | undefined\n let renditionSpreadMeta: string | undefined\n const identifiers: OpfIdentifier[] = []\n\n if (metadataEl !== undefined) {\n title = firstTextByLocalName(metadataEl, \"title\")\n publisher = firstTextByLocalName(metadataEl, \"publisher\")\n rights = firstTextByLocalName(metadataEl, \"rights\")\n date = firstTextByLocalName(metadataEl, \"date\")\n creators = textsByLocalName(metadataEl, \"creator\")\n languages = textsByLocalName(metadataEl, \"language\")\n subjects = textsByLocalName(metadataEl, \"subject\")\n renditionLayoutMeta = metaValByProperty(metadataEl, \"rendition:layout\")\n renditionFlowMeta = metaValByProperty(metadataEl, \"rendition:flow\")\n renditionSpreadMeta = metaValByProperty(metadataEl, \"rendition:spread\")\n identifiers.push(...identifiersFromMetadata(metadataEl))\n }\n\n const coverHref = coverHrefFromManifestAndMetadata({\n manifestItems,\n metadataEl,\n })\n\n const guide = guideFromPackage(doc)\n\n return {\n kind: \"opf\",\n manifestItems,\n spineRows,\n spineTocIdref,\n identifiers,\n title,\n creators,\n publisher,\n rights,\n languages,\n subjects,\n date,\n coverHref,\n renditionLayoutMeta,\n renditionFlowMeta,\n renditionSpreadMeta,\n pageProgressionDirection,\n guide,\n }\n}\n","import type { ArchiveResolveResult } from \"../types/archiveResolve\"\nimport type { AppleMetadata } from \"./parse\"\n\nexport const resolveApple = (input: AppleMetadata): ArchiveResolveResult => {\n const fixedLayout = input.displayOptions?.platform?.options?.find(\n (o) => o.name === \"fixed-layout\",\n )?.value\n\n const renditionLayout =\n fixedLayout?.trim().toLowerCase() === \"true\" ? \"pre-paginated\" : undefined\n\n return {\n gtin: undefined,\n isbn: undefined,\n readingDirection: undefined,\n renditionLayout,\n title: undefined,\n authors: undefined,\n publisher: undefined,\n rights: undefined,\n languages: undefined,\n date: undefined,\n subjects: undefined,\n }\n}\n","const GTIN_LENGTHS = new Set([8, 12, 13, 14])\n\n/**\n * Normalize a raw GTIN / EAN / UPC string to digits only when the length\n * matches a GS1 GTIN family size (8, 12, 13, or 14). Does not verify check digits.\n */\nexport const normalizeGtin = (\n raw: string | number | undefined | null,\n): string | undefined => {\n if (raw === undefined || raw === null) return undefined\n\n const digits = String(raw).replace(/\\D/g, \"\")\n if (digits.length === 0 || !GTIN_LENGTHS.has(digits.length)) return undefined\n\n return digits\n}\n","const ISBN_CANDIDATE_PATTERN = /(?:97[89])?\\d{9}[\\dXx]/\n\n/**\n * Normalize a raw ISBN-ish string into a canonical 10- or 13-character\n * form, or `undefined` when no recognisable ISBN can be recovered.\n *\n * - Strips the common `urn:isbn:` / `isbn:` prefixes.\n * - Drops everything that isn't a digit or `X`.\n * - Validates the resulting length (10 or 13).\n * - Falls back to a lax regex scan so publishers that stuff free text\n * around the number still yield a usable value.\n */\nexport const normalizeIsbn = (\n raw: string | number | undefined | null,\n): string | undefined => {\n if (raw === undefined || raw === null) return undefined\n\n const stripped = String(raw)\n .trim()\n .replace(/^urn:isbn:/i, \"\")\n .replace(/^isbn[:\\s-]*/i, \"\")\n\n const digitsOnly = stripped.replace(/[^0-9Xx]/g, \"\")\n\n if (digitsOnly.length === 10 || digitsOnly.length === 13) {\n return digitsOnly.toUpperCase()\n }\n\n const match = stripped.match(ISBN_CANDIDATE_PATTERN)\n if (match) return match[0].toUpperCase()\n\n return undefined\n}\n","import type { ArchiveResolveResult } from \"../types/archiveResolve\"\nimport { normalizeGtin } from \"../utils/normalizeGtin\"\nimport { normalizeIsbn } from \"../utils/normalizeIsbn\"\nimport type { ComicInfo } from \"./parse\"\n\nconst readingDirection = (info: ComicInfo): \"ltr\" | \"rtl\" | undefined => {\n switch (info.Manga) {\n case \"YesAndRightToLeft\":\n return \"rtl\"\n case \"Yes\":\n case \"No\":\n return \"ltr\"\n default:\n return undefined\n }\n}\n\nconst trimToUndefined = (raw: string | undefined): string | undefined => {\n if (raw === undefined) return undefined\n const trimmed = raw.trim()\n return trimmed.length > 0 ? trimmed : undefined\n}\n\n/**\n * Split a comma-separated ComicInfo value into its individual tokens.\n * `Writer`, `Genre`, and `Tags` all follow the same de-facto convention:\n * tokens separated by `,`, with whitespace trimmed and empty tokens\n * dropped (real-world files leave trailing commas around).\n */\nconst splitCommaList = (raw: string | undefined): string[] => {\n if (raw === undefined) return []\n return raw\n .split(\",\")\n .map((token) => token.trim())\n .filter((token) => token.length > 0)\n}\n\nconst parseNonNegativeInt = (raw: string | undefined): number | undefined => {\n const trimmed = trimToUndefined(raw)\n if (trimmed === undefined) return undefined\n if (!/^\\d+$/.test(trimmed)) return undefined\n const n = Number.parseInt(trimmed, 10)\n return Number.isFinite(n) ? n : undefined\n}\n\nconst dateFromYearMonthDay = (\n info: ComicInfo,\n):\n | {\n year?: number\n month?: number\n day?: number\n }\n | undefined => {\n const year = parseNonNegativeInt(info.Year)\n const month = parseNonNegativeInt(info.Month)\n const day = parseNonNegativeInt(info.Day)\n\n if (year === undefined && month === undefined && day === undefined) {\n return undefined\n }\n\n return {\n ...(year !== undefined ? { year } : {}),\n ...(month !== undefined ? { month } : {}),\n ...(day !== undefined ? { day } : {}),\n }\n}\n\nexport const resolveComicInfo = (info: ComicInfo): ArchiveResolveResult => {\n const raw = info.GTIN\n const languageIso = trimToUndefined(info.LanguageISO)\n const authors = splitCommaList(info.Writer)\n const subjects = [...splitCommaList(info.Genre), ...splitCommaList(info.Tags)]\n\n return {\n gtin: normalizeGtin(raw),\n isbn: normalizeIsbn(raw),\n readingDirection: readingDirection(info),\n renditionLayout: undefined,\n title: trimToUndefined(info.Title),\n authors: authors.length > 0 ? authors : undefined,\n publisher: trimToUndefined(info.Publisher),\n rights: undefined,\n languages: languageIso !== undefined ? [languageIso] : undefined,\n date: dateFromYearMonthDay(info),\n subjects: subjects.length > 0 ? subjects : undefined,\n }\n}\n","import type { ArchiveResolveResult } from \"../types/archiveResolve\"\nimport type { KoboMetadata } from \"./parse\"\n\nexport const resolveKobo = (input: KoboMetadata): ArchiveResolveResult => ({\n gtin: undefined,\n isbn: undefined,\n readingDirection: undefined,\n renditionLayout: input.renditionLayout,\n title: undefined,\n authors: undefined,\n publisher: undefined,\n rights: undefined,\n languages: undefined,\n date: undefined,\n subjects: undefined,\n})\n","/**\n * Extract the calendar components of a W3CDTF literal — the\n * date subset of ISO 8601 that EPUB 3.3 § 5.5.3.2.4 mandates for\n * `dc:date`. Accepted forms: `YYYY`, `YYYY-MM`, `YYYY-MM-DD`, and\n * any of the above followed by a `Thh:mm[:ss[.s]][TZD]` time\n * portion that we ignore.\n *\n * Components are returned as plain integers (`month` is 1-12,\n * `day` is 1-31), independent of the host timezone — using\n * `Date.parse` would shift `2011-01-01` by a day in negative-offset\n * locales, which is the exact bug this regex-based approach exists\n * to avoid. Returns `undefined` when the input doesn't even match\n * a leading 4-digit year so consumers can fall back without\n * branching on partial shapes.\n */\nexport const parseW3cDtfDate = (\n raw: string | undefined,\n):\n | {\n year?: number\n month?: number\n day?: number\n }\n | undefined => {\n if (raw === undefined) return undefined\n\n const match = raw.trim().match(/^(\\d{4})(?:-(\\d{2})(?:-(\\d{2}))?)?/)\n if (match === null) return undefined\n\n const [, yearRaw, monthRaw, dayRaw] = match\n\n return {\n ...(yearRaw !== undefined ? { year: Number.parseInt(yearRaw, 10) } : {}),\n ...(monthRaw !== undefined ? { month: Number.parseInt(monthRaw, 10) } : {}),\n ...(dayRaw !== undefined ? { day: Number.parseInt(dayRaw, 10) } : {}),\n }\n}\n","import type { ArchiveResolveResult } from \"../types/archiveResolve\"\nimport { normalizeGtin } from \"../utils/normalizeGtin\"\nimport { normalizeIsbn } from \"../utils/normalizeIsbn\"\nimport { parseW3cDtfDate } from \"../utils/parseW3cDtfDate\"\nimport type { OpfIdentifier, OpfMetadata } from \"./parse\"\n\nconst rawIdentifierValueForIsbn = (\n identifiers: ReadonlyArray<OpfIdentifier>,\n): string | undefined => {\n for (const i of identifiers) {\n if (i.scheme !== undefined && i.scheme.trim().toLowerCase() === \"isbn\") {\n if (normalizeIsbn(i.value) !== undefined) return i.value\n }\n }\n\n for (const i of identifiers) {\n if (normalizeIsbn(i.value) !== undefined) return i.value\n }\n\n return undefined\n}\n\nexport const resolveOpf = (input: OpfMetadata): ArchiveResolveResult => {\n const ppd = input.pageProgressionDirection?.trim().toLowerCase()\n const readingDirection = ppd === \"ltr\" || ppd === \"rtl\" ? ppd : undefined\n\n const rl = input.renditionLayoutMeta?.trim().toLowerCase()\n const renditionLayout =\n rl === \"reflowable\" || rl === \"pre-paginated\" ? rl : undefined\n\n const raw = rawIdentifierValueForIsbn(input.identifiers)\n\n return {\n gtin: normalizeGtin(raw),\n isbn: normalizeIsbn(raw),\n readingDirection,\n renditionLayout,\n title: input.title,\n authors: input.creators.length > 0 ? [...input.creators] : undefined,\n publisher: input.publisher,\n rights: input.rights,\n languages: input.languages.length > 0 ? [...input.languages] : undefined,\n date: parseW3cDtfDate(input.date),\n subjects: input.subjects.length > 0 ? [...input.subjects] : undefined,\n }\n}\n","import type { AppleMetadata } from \"./apple/parse\"\nimport { resolveApple } from \"./apple/resolve\"\nimport type { ComicInfo } from \"./comicInfo/parse\"\nimport { resolveComicInfo } from \"./comicInfo/resolve\"\nimport type { KoboMetadata } from \"./kobo/parse\"\nimport { resolveKobo } from \"./kobo/resolve\"\nimport type { OpfMetadata } from \"./opf/parse\"\nimport { resolveOpf } from \"./opf/resolve\"\nimport type { ArchiveResolveResult } from \"./types/archiveResolve\"\n\nexport type ResolvedArchiveInput =\n | ComicInfo\n | KoboMetadata\n | AppleMetadata\n | OpfMetadata\n\nexport const resolveArchiveMetadata = (\n input: ResolvedArchiveInput,\n): ArchiveResolveResult => {\n if (input.kind === \"comicInfo\") return resolveComicInfo(input)\n if (input.kind === \"kobo\") return resolveKobo(input)\n if (input.kind === \"apple\") return resolveApple(input)\n return resolveOpf(input)\n}\n"],"mappings":"4UAGA,IAAa,EACX,uCAgBI,EACJ,GAEA,EAAS,cAAc,QAAQ,EAAE,IAAK,IAAY,CAChD,KAAM,EAAO,MAAM,KACnB,MAAO,EAAO,GAChB,EAAE,EAES,EAA+B,GAA+B,CACzE,IAAM,EAAM,IAAI,EAAA,YAAY,CAAG,EAG/B,GAFa,EAAI,MAAM,YAAY,IAEtB,kBACX,MAAO,CAAE,KAAM,OAAQ,EAGzB,IAAM,EAAa,EAAI,WAAW,UAAU,EAE5C,MAAO,CACL,KAAM,QACN,eAAgB,CACd,GAAI,IAAe,IAAA,GAEf,CAAC,EADD,CAAE,SAAU,CAAE,QAAS,EAA2B,CAAU,CAAE,CAAE,CAEtE,CACF,CACF,ECzCa,EAA0B,CACrC,UACA,KACA,MACA,mBACF,EAIa,EAAoB,GAA2C,CAC1E,IAAK,IAAM,KAAK,EACd,GAAI,IAAM,EAAO,MAAO,GAE1B,MAAO,EACT,ECda,EAAsB,gBA4D7B,EAAwB,IAAI,IAAI,CAAC,QAAS,WAAW,CAAC,EAEtD,EAAoB,GACxB,EAAG,SAAS,KAAM,GAAM,EAAE,OAAS,SAAS,EAExC,EAAe,GAAuC,CAC1D,IAAM,EAAI,EAAG,IAAI,KAAK,EACtB,OAAO,EAAE,OAAS,EAAI,EAAI,IAAA,EAC5B,EAOa,EAAkB,GAA2B,CACxD,IAAI,EACJ,GAAI,CACF,EAAM,IAAI,EAAA,YAAY,CAAG,CAC3B,OAAS,EAAO,CACd,IAAM,EAAU,aAAiB,MAAQ,EAAM,QAAU,OAAO,CAAK,EACrE,MAAU,MAAM,GAAG,EAAoB,iBAAiB,IAAW,CACjE,OACF,CAAC,CACH,CAEA,IAAM,EAAiC,CAAC,EAexC,OAbA,EAAI,UAAW,GAAU,CAGvB,GAFI,EAAM,OAAS,WACf,EAAsB,IAAI,EAAM,IAAI,GACpC,EAAiB,CAAK,EAAG,OAE7B,IAAM,EAAO,EAAY,CAAK,EAC1B,IAAS,IAAA,IACT,EAAO,EAAM,QAAU,IAAA,KAE3B,EAAO,EAAM,MAAQ,EACvB,CAAC,EAGM,CAAE,KAAM,YAAa,GAAG,CAAO,CACxC,ECzGa,EAAgC,oCAWvC,EACJ,GAC0C,CAC1C,IAAM,EAAW,EAAI,WAAW,UAAU,EAC1C,GAAI,CAAC,EAAU,MAAO,CAAC,EAEvB,IAAK,IAAM,KAAU,EAAS,cAAc,QAAQ,EAC9C,KAAO,MAAM,OAAS,eAI1B,OAHI,EAAO,IAAI,KAAK,EAAE,YAAY,IAAM,OAC/B,CAAE,gBAAiB,eAAgB,EAErC,CAAC,EAGV,MAAO,CAAC,CACV,EAMa,EAAgB,GAA8B,CACzD,IAAM,EAAM,IAAI,EAAA,YAAY,CAAG,EAO/B,OANa,EAAI,MAAM,YAAY,IAEtB,kBACJ,CAAE,KAAM,OAAQ,GAAG,EAAgC,CAAG,CAAE,EAG1D,CAAE,KAAM,MAAO,CACxB,ECvCa,EACX,GAEI,IAAQ,IAAA,IAAa,EAAI,KAAK,EAAE,SAAW,EACtC,CAAC,EAGH,EACJ,KAAK,EACL,MAAM,KAAK,EACX,OAAQ,GAAM,EAAE,OAAS,CAAC,ECIlB,EACX,GAC0B,CAC1B,IAAM,EAAS,EAA8B,CAAU,EACvD,GAAI,EAAO,SAAW,EACpB,MAAO,CAAC,EAGV,IAAI,EACA,EAAO,SAAS,6BAA6B,IAC/C,EAAkB,cAEhB,EAAO,SAAS,gCAAgC,IAClD,EAAkB,iBAGpB,IAAI,EAcJ,OAbI,EAAO,SAAS,qBAAqB,IACvC,EAAgB,QAEd,EAAO,SAAS,0BAA0B,IAC5C,EAAgB,aAEd,EAAO,SAAS,6BAA6B,IAC/C,EAAgB,gBAEd,EAAO,SAAS,oCAAoC,IACtD,EAAgB,uBAGX,CACL,GAAI,IAAoB,IAAA,GAAkC,CAAC,EAAvB,CAAE,iBAAgB,EACtD,GAAI,IAAkB,IAAA,GAAgC,CAAC,EAArB,CAAE,eAAc,EAClD,GAAI,EAAO,SAAS,kBAAkB,EAClC,CAAE,eAAgB,EAAc,EAChC,CAAC,EACL,GAAI,EAAO,SAAS,mBAAmB,EACnC,CAAE,gBAAiB,EAAc,EACjC,CAAC,CACP,CACF,ECnBM,EAAoB,GACxB,EAAK,SAAS,GAAG,EAAI,EAAK,MAAM,EAAK,YAAY,GAAG,EAAI,CAAC,EAAI,EAEzD,GAAe,EAAqB,IACxC,EAAiB,CAAW,EAAE,YAAY,IAAM,EAAU,YAAY,EAElE,EAAgB,GACpB,EAAK,OAAS,UAEV,GACJ,EACA,IAC2B,CAC3B,IAAK,IAAM,KAAQ,EAAO,SACnB,KAAa,CAAI,GAClB,EAAY,EAAK,KAAM,CAAS,EAAG,OAAO,CAGlD,EAEM,GACJ,EACA,IACiB,CACjB,IAAM,EAAoB,CAAC,EAC3B,IAAK,IAAM,KAAQ,EAAO,SACnB,EAAa,CAAI,GAClB,EAAY,EAAK,KAAM,CAAS,GAAG,EAAI,KAAK,CAAI,EAEtD,OAAO,CACT,EAEM,EAA2B,GAA4C,CAC3E,IAAM,EAA+B,CAAC,EAoBtC,OAlBA,EAAW,UAAW,GAAU,CAC9B,GAAI,EAAiB,EAAM,IAAI,EAAE,YAAY,IAAM,aAAc,OAEjE,IAAM,EAAQ,EAAM,IAAI,KAAK,EAC7B,GAAI,EAAM,SAAW,EAAG,OAIxB,IAAM,GADJ,EAAM,KAAK,eAAiB,EAAM,KAAK,eAAiB,EAAM,KAAK,SACvC,KAAK,EAEnC,EAAY,KAAK,CACf,QACA,GAAI,IAAkB,IAAA,IAAa,EAAc,OAAS,EACtD,CAAE,OAAQ,CAAc,EACxB,CAAC,CACP,CAAC,CACH,CAAC,EAEM,CACT,EAEM,GACJ,EACA,IACuB,CACvB,IAAI,EAQJ,OAPA,EAAW,UAAW,GAAU,CAE9B,GADI,IAAU,IAAA,IACV,EAAiB,EAAM,IAAI,EAAE,YAAY,IAAM,EAAU,YAAY,EACvE,OACF,IAAM,EAAI,EAAM,IAAI,KAAK,EACrB,EAAE,OAAS,IAAG,EAAQ,EAC5B,CAAC,EACM,CACT,EAEM,GACJ,EACA,IACa,CACb,IAAM,EAAgB,CAAC,EAOvB,OANA,EAAW,UAAW,GAAU,CAC9B,GAAI,EAAiB,EAAM,IAAI,EAAE,YAAY,IAAM,EAAU,YAAY,EACvE,OACF,IAAM,EAAI,EAAM,IAAI,KAAK,EACrB,EAAE,OAAS,GAAG,EAAI,KAAK,CAAC,CAC9B,CAAC,EACM,CACT,EAEM,EACJ,GACuB,CACvB,IAAI,EAQJ,OAPA,EAAW,UAAW,GAAU,CAG9B,GAFI,IAAY,IAAA,IACZ,EAAiB,EAAM,IAAI,EAAE,YAAY,IAAM,QAC/C,EAAM,KAAK,MAAM,YAAY,IAAM,QAAS,OAChD,IAAM,EAAU,EAAM,KAAK,SAAS,KAAK,EACrC,IAAY,IAAA,IAAa,EAAQ,OAAS,IAAG,EAAU,EAC7D,CAAC,EACM,CACT,EAoBM,GAAoC,CACxC,gBACA,gBAIwB,CACxB,IAAM,EAAW,GACf,EAAK,WAAW,YAAY,EAAE,SAAS,QAAQ,IAAM,GAEjD,EAAuB,EAAc,KAAM,GAC1C,EAAQ,CAAI,EACV,EAA8B,EAAK,UAAU,EAAE,SACpD,aACF,EAH2B,EAI5B,EACD,GAAI,IAAyB,IAAA,GAAW,OAAO,EAAqB,KAEpE,GAAI,IAAe,IAAA,GAAW,CAC5B,IAAM,EAAiB,EAA2B,CAAU,EAC5D,GAAI,IAAmB,IAAA,GAAW,CAChC,IAAM,EAAQ,EAAc,KACzB,GAAS,EAAK,KAAO,GAAkB,EAAQ,CAAI,CACtD,EACA,GAAI,IAAU,IAAA,GAAW,OAAO,EAAM,IACxC,CACF,CAEA,OAAO,EAAc,KAClB,GAAS,EAAK,GAAG,YAAY,EAAE,SAAS,OAAO,GAAK,EAAQ,CAAI,CACnE,GAAG,IACL,EAEM,GACJ,EACA,IACuB,CAIvB,IAAM,EAHO,EAAmB,EAAY,MAAM,EAAE,KACjD,GAAM,EAAE,KAAK,WAAa,CAEjB,GAAM,IACd,SAAQ,IAAA,IAAa,EAAI,KAAK,EAAE,SAAW,GAC/C,OAAO,CACT,EAEM,EAAoB,GAAyC,CACjE,IAAM,EAAU,EAAgB,EAAK,OAAO,EAC5C,GAAI,IAAY,IAAA,GAAW,MAAO,CAAC,EAEnC,IAAM,EAA4B,CAAC,EAEnC,IAAK,IAAM,KAAO,EAAmB,EAAS,WAAW,EAAG,CAC1D,IAAM,EAAO,EAAI,KAAK,MAAM,KAAK,EAC7B,IAAS,IAAA,IAAa,EAAK,SAAW,GAC1C,EAAK,KAAK,CACR,OACA,MAAO,EAAI,KAAK,OAAO,KAAK,GAAK,GACjC,KAAM,EAAI,KAAK,MAAM,KAAK,GAAK,EACjC,CAAC,CACH,CAEA,OAAO,CACT,EAEM,EACJ,GACqC,CACrC,IAAM,EAAK,EAAK,KAAK,GACf,EAAO,EAAK,KAAK,KAEvB,GADI,IAAO,IAAA,IAAa,EAAG,SAAW,GAClC,IAAS,IAAA,IAAa,EAAK,SAAW,EAAG,OAE7C,IAAM,EAAY,EAAK,KAAK,cACtB,EAAa,EAAK,KAAK,YAAY,KAAK,EAC9C,MAAO,CACL,KACA,OACA,GAAI,IAAc,IAAA,IAAa,EAAU,OAAS,EAAI,CAAE,WAAU,EAAI,CAAC,EACvE,GAAI,IAAe,IAAA,IAAa,EAAW,OAAS,EAChD,CAAE,YAAW,EACb,CAAC,CACP,CACF,EAEM,EACJ,GAIG,CACH,IAAM,EAAgC,CAAC,EACjC,EAAO,IAAI,IAEjB,IAAK,IAAM,KAAM,EAAmB,EAAY,MAAM,EAAG,CACvD,IAAM,EAAS,EAA2B,CAAE,EACxC,IAAW,IAAA,KACf,EAAM,KAAK,CAAM,EACjB,EAAK,IAAI,EAAO,GAAI,CAAM,EAC5B,CAEA,MAAO,CAAE,QAAO,MAAK,CACvB,EAEM,GACJ,EACA,IACkB,CAClB,IAAM,EAAsB,CAAC,EAE7B,IAAK,IAAM,KAAW,EAAmB,EAAS,SAAS,EAAG,CAC5D,IAAM,EAAQ,EAAQ,KAAK,MAC3B,GAAI,IAAU,IAAA,IAAa,EAAM,KAAK,EAAE,SAAW,EAAG,SAEtD,IAAM,EAAe,EAAK,IAAI,CAAK,EACnC,GAAI,IAAiB,IAAA,GAAW,SAEhC,IAAM,EAAQ,EAAiC,EAAQ,KAAK,UAAU,EAEtE,EAAK,KAAK,CACR,QACA,GAAI,EAAa,GACjB,KAAM,EAAa,KACnB,GAAI,EAAa,YAAc,IAAA,GAE3B,CAAC,EADD,CAAE,UAAW,EAAa,SAAU,EAExC,GAAI,EAAa,aAAe,IAAA,GAE5B,CAAC,EADD,CAAE,WAAY,EAAa,UAAW,EAE1C,GAAI,EAAM,kBAAoB,IAAA,GAE1B,CAAC,EADD,CAAE,gBAAiB,EAAM,eAAgB,EAE7C,GAAI,EAAM,gBAAkB,IAAA,GAExB,CAAC,EADD,CAAE,cAAe,EAAM,aAAc,EAEzC,GAAI,EAAM,iBAAmB,IAAA,GAEzB,CAAC,EADD,CAAE,eAAgB,EAAM,cAAe,EAE3C,GAAI,EAAM,kBAAoB,IAAA,GAE1B,CAAC,EADD,CAAE,gBAAiB,EAAM,eAAgB,CAE/C,CAAC,CACH,CAEA,OAAO,CACT,EAoDa,EAAY,GAAgC,CACvD,IAAM,EAAM,IAAI,EAAA,YAAY,CAAM,EAC5B,EAAa,EAAgB,EAAK,UAAU,EAC5C,EAAU,EAAgB,EAAK,OAAO,EACtC,EAAa,EAAgB,EAAK,UAAU,EAE9C,EAAwC,CAAC,EACzC,EAA2B,CAAC,EAEhC,GAAI,IAAe,IAAA,GAAW,CAC5B,GAAM,CAAE,QAAO,QAAS,EAAqB,CAAU,EACvD,EAAgB,EACZ,IAAY,IAAA,KACd,EAAY,EAA0B,EAAM,CAAO,EAEvD,CAEA,IAAM,EACJ,GAAS,KAAK,8BACV,EACJ,IAAgC,IAAA,IAChC,EAA4B,KAAK,EAAE,OAAS,EACxC,EACA,IAAA,GAEA,EAAc,GAAS,KAAK,IAC5B,EACJ,IAAgB,IAAA,IAAa,EAAY,KAAK,EAAE,OAAS,EACrD,EAAY,KAAK,EACjB,IAAA,GAEF,EACA,EACA,EACA,EACA,EAAqB,CAAC,EACtB,EAAsB,CAAC,EACvB,EAAqB,CAAC,EACtB,EACA,EACA,EACE,EAA+B,CAAC,EAElC,IAAe,IAAA,KACjB,EAAQ,EAAqB,EAAY,OAAO,EAChD,EAAY,EAAqB,EAAY,WAAW,EACxD,EAAS,EAAqB,EAAY,QAAQ,EAClD,EAAO,EAAqB,EAAY,MAAM,EAC9C,EAAW,EAAiB,EAAY,SAAS,EACjD,EAAY,EAAiB,EAAY,UAAU,EACnD,EAAW,EAAiB,EAAY,SAAS,EACjD,EAAsB,EAAkB,EAAY,kBAAkB,EACtE,EAAoB,EAAkB,EAAY,gBAAgB,EAClE,EAAsB,EAAkB,EAAY,kBAAkB,EACtE,EAAY,KAAK,GAAG,EAAwB,CAAU,CAAC,GAGzD,IAAM,EAAY,EAAiC,CACjD,gBACA,YACF,CAAC,EAEK,EAAQ,EAAiB,CAAG,EAElC,MAAO,CACL,KAAM,MACN,gBACA,YACA,gBACA,cACA,QACA,WACA,YACA,SACA,YACA,WACA,OACA,YACA,sBACA,oBACA,sBACA,2BACA,OACF,CACF,ECjba,EAAgB,IAQpB,CACL,KAAM,IAAA,GACN,KAAM,IAAA,GACN,iBAAkB,IAAA,GAClB,iBAXkB,EAAM,gBAAgB,UAAU,SAAS,KAC1D,GAAM,EAAE,OAAS,cACpB,GAAG,QAGY,KAAK,EAAE,YAAY,IAAM,OAAS,gBAAkB,IAAA,GAOjE,MAAO,IAAA,GACP,QAAS,IAAA,GACT,UAAW,IAAA,GACX,OAAQ,IAAA,GACR,UAAW,IAAA,GACX,KAAM,IAAA,GACN,SAAU,IAAA,EACZ,GCvBI,EAAe,IAAI,IAAI,CAAC,EAAG,GAAI,GAAI,EAAE,CAAC,EAM/B,EACX,GACuB,CACvB,GAAI,GAA6B,KAAM,OAEvC,IAAM,EAAS,OAAO,CAAG,EAAE,QAAQ,MAAO,EAAE,EACxC,OAAO,SAAW,GAAK,CAAC,EAAa,IAAI,EAAO,MAAM,GAE1D,OAAO,CACT,ECfM,EAAyB,yBAYlB,EACX,GACuB,CACvB,GAAI,GAA6B,KAAM,OAEvC,IAAM,EAAW,OAAO,CAAG,EACxB,KAAK,EACL,QAAQ,cAAe,EAAE,EACzB,QAAQ,gBAAiB,EAAE,EAExB,EAAa,EAAS,QAAQ,YAAa,EAAE,EAEnD,GAAI,EAAW,SAAW,IAAM,EAAW,SAAW,GACpD,OAAO,EAAW,YAAY,EAGhC,IAAM,EAAQ,EAAS,MAAM,CAAsB,EACnD,GAAI,EAAO,OAAO,EAAM,GAAG,YAAY,CAGzC,EC3BM,EAAoB,GAA+C,CACvE,OAAQ,EAAK,MAAb,CACE,IAAK,oBACH,MAAO,MACT,IAAK,MACL,IAAK,KACH,MAAO,MACT,QACE,MACJ,CACF,EAEM,EAAmB,GAAgD,CACvE,GAAI,IAAQ,IAAA,GAAW,OACvB,IAAM,EAAU,EAAI,KAAK,EACzB,OAAO,EAAQ,OAAS,EAAI,EAAU,IAAA,EACxC,EAQM,EAAkB,GAClB,IAAQ,IAAA,GAAkB,CAAC,EACxB,EACJ,MAAM,GAAG,EACT,IAAK,GAAU,EAAM,KAAK,CAAC,EAC3B,OAAQ,GAAU,EAAM,OAAS,CAAC,EAGjC,EAAuB,GAAgD,CAC3E,IAAM,EAAU,EAAgB,CAAG,EAEnC,GADI,IAAY,IAAA,IACZ,CAAC,QAAQ,KAAK,CAAO,EAAG,OAC5B,IAAM,EAAI,OAAO,SAAS,EAAS,EAAE,EACrC,OAAO,OAAO,SAAS,CAAC,EAAI,EAAI,IAAA,EAClC,EAEM,EACJ,GAOe,CACf,IAAM,EAAO,EAAoB,EAAK,IAAI,EACpC,EAAQ,EAAoB,EAAK,KAAK,EACtC,EAAM,EAAoB,EAAK,GAAG,EAEpC,SAAS,IAAA,IAAa,IAAU,IAAA,IAAa,IAAQ,IAAA,IAIzD,MAAO,CACL,GAAI,IAAS,IAAA,GAAuB,CAAC,EAAZ,CAAE,MAAK,EAChC,GAAI,IAAU,IAAA,GAAwB,CAAC,EAAb,CAAE,OAAM,EAClC,GAAI,IAAQ,IAAA,GAAsB,CAAC,EAAX,CAAE,KAAI,CAChC,CACF,EAEa,EAAoB,GAA0C,CACzE,IAAM,EAAM,EAAK,KACX,EAAc,EAAgB,EAAK,WAAW,EAC9C,EAAU,EAAe,EAAK,MAAM,EACpC,EAAW,CAAC,GAAG,EAAe,EAAK,KAAK,EAAG,GAAG,EAAe,EAAK,IAAI,CAAC,EAE7E,MAAO,CACL,KAAM,EAAc,CAAG,EACvB,KAAM,EAAc,CAAG,EACvB,iBAAkB,EAAiB,CAAI,EACvC,gBAAiB,IAAA,GACjB,MAAO,EAAgB,EAAK,KAAK,EACjC,QAAS,EAAQ,OAAS,EAAI,EAAU,IAAA,GACxC,UAAW,EAAgB,EAAK,SAAS,EACzC,OAAQ,IAAA,GACR,UAAW,IAAgB,IAAA,GAA4B,IAAA,GAAhB,CAAC,CAAW,EACnD,KAAM,EAAqB,CAAI,EAC/B,SAAU,EAAS,OAAS,EAAI,EAAW,IAAA,EAC7C,CACF,ECrFa,EAAe,IAA+C,CACzE,KAAM,IAAA,GACN,KAAM,IAAA,GACN,iBAAkB,IAAA,GAClB,gBAAiB,EAAM,gBACvB,MAAO,IAAA,GACP,QAAS,IAAA,GACT,UAAW,IAAA,GACX,OAAQ,IAAA,GACR,UAAW,IAAA,GACX,KAAM,IAAA,GACN,SAAU,IAAA,EACZ,GCAa,EACX,GAOe,CACf,GAAI,IAAQ,IAAA,GAAW,OAEvB,IAAM,EAAQ,EAAI,KAAK,EAAE,MAAM,oCAAoC,EACnE,GAAI,IAAU,KAAM,OAEpB,GAAM,EAAG,EAAS,EAAU,GAAU,EAEtC,MAAO,CACL,GAAI,IAAY,IAAA,GAAqD,CAAC,EAA1C,CAAE,KAAM,OAAO,SAAS,EAAS,EAAE,CAAE,EACjE,GAAI,IAAa,IAAA,GAAuD,CAAC,EAA5C,CAAE,MAAO,OAAO,SAAS,EAAU,EAAE,CAAE,EACpE,GAAI,IAAW,IAAA,GAAmD,CAAC,EAAxC,CAAE,IAAK,OAAO,SAAS,EAAQ,EAAE,CAAE,CAChE,CACF,EC9BM,EACJ,GACuB,CACvB,IAAK,IAAM,KAAK,EACd,GAAI,EAAE,SAAW,IAAA,IAAa,EAAE,OAAO,KAAK,EAAE,YAAY,IAAM,QAC1D,EAAc,EAAE,KAAK,IAAM,IAAA,GAAW,OAAO,EAAE,MAIvD,IAAK,IAAM,KAAK,EACd,GAAI,EAAc,EAAE,KAAK,IAAM,IAAA,GAAW,OAAO,EAAE,KAIvD,EAEa,EAAc,GAA6C,CACtE,IAAM,EAAM,EAAM,0BAA0B,KAAK,EAAE,YAAY,EACzD,EAAmB,IAAQ,OAAS,IAAQ,MAAQ,EAAM,IAAA,GAE1D,EAAK,EAAM,qBAAqB,KAAK,EAAE,YAAY,EACnD,EACJ,IAAO,cAAgB,IAAO,gBAAkB,EAAK,IAAA,GAEjD,EAAM,EAA0B,EAAM,WAAW,EAEvD,MAAO,CACL,KAAM,EAAc,CAAG,EACvB,KAAM,EAAc,CAAG,EACvB,mBACA,kBACA,MAAO,EAAM,MACb,QAAS,EAAM,SAAS,OAAS,EAAI,CAAC,GAAG,EAAM,QAAQ,EAAI,IAAA,GAC3D,UAAW,EAAM,UACjB,OAAQ,EAAM,OACd,UAAW,EAAM,UAAU,OAAS,EAAI,CAAC,GAAG,EAAM,SAAS,EAAI,IAAA,GAC/D,KAAM,EAAgB,EAAM,IAAI,EAChC,SAAU,EAAM,SAAS,OAAS,EAAI,CAAC,GAAG,EAAM,QAAQ,EAAI,IAAA,EAC9D,CACF,yTC5BE,GAEI,EAAM,OAAS,YAAoB,EAAiB,CAAK,EACzD,EAAM,OAAS,OAAe,EAAY,CAAK,EAC/C,EAAM,OAAS,QAAgB,EAAa,CAAK,EAC9C,EAAW,CAAK"}
|
package/dist/kobo/parse.d.ts
CHANGED
|
@@ -1,14 +1,6 @@
|
|
|
1
1
|
export declare const KOBO_DISPLAY_OPTIONS_FILENAME = "com.kobobooks.display-options.xml";
|
|
2
|
-
/**
|
|
3
|
-
* Normalized fields from Kobo-specific XML sidecars. Grows as new Kobo
|
|
4
|
-
* document kinds are supported; absent keys mean not present or unknown.
|
|
5
|
-
*/
|
|
6
2
|
export type KoboMetadata = {
|
|
7
3
|
readonly kind: "kobo";
|
|
8
4
|
renditionLayout?: `reflowable` | `pre-paginated`;
|
|
9
5
|
};
|
|
10
|
-
/**
|
|
11
|
-
* Parse a Kobo-related XML document. Unsupported or unrecognized roots
|
|
12
|
-
* yield an empty object. Malformed XML throws from the underlying parser.
|
|
13
|
-
*/
|
|
14
6
|
export declare const parseKoboXml: (xml: string) => KoboMetadata;
|
package/dist/opf/parse.d.ts
CHANGED
|
@@ -31,30 +31,12 @@ export type OpfMetadata = {
|
|
|
31
31
|
readonly spineTocIdref: string | undefined;
|
|
32
32
|
readonly identifiers: ReadonlyArray<OpfIdentifier>;
|
|
33
33
|
readonly title: string | undefined;
|
|
34
|
-
/** `dc:creator` values, in document order, trimmed; empty when none. */
|
|
35
34
|
readonly creators: ReadonlyArray<string>;
|
|
36
|
-
/** First non-empty `dc:publisher`, trimmed. */
|
|
37
35
|
readonly publisher: string | undefined;
|
|
38
|
-
/** First non-empty `dc:rights`, trimmed. */
|
|
39
36
|
readonly rights: string | undefined;
|
|
40
|
-
/** `dc:language` values, in document order, trimmed; empty when none. */
|
|
41
37
|
readonly languages: ReadonlyArray<string>;
|
|
42
|
-
/** `dc:subject` values, in document order, trimmed; empty when none. */
|
|
43
38
|
readonly subjects: ReadonlyArray<string>;
|
|
44
|
-
/**
|
|
45
|
-
* Raw `dc:date` value as authored. EPUB 3 requires W3CDTF (a profile
|
|
46
|
-
* of ISO 8601), but real-world publishers also ship free text here,
|
|
47
|
-
* so the value is exposed verbatim and consumers normalize as needed.
|
|
48
|
-
*/
|
|
49
39
|
readonly date: string | undefined;
|
|
50
|
-
/**
|
|
51
|
-
* Manifest-relative `href` of the cover image, when one can be
|
|
52
|
-
* resolved from `cover-image` properties (EPUB 3), the EPUB 2
|
|
53
|
-
* `<meta name="cover">` convention, or an `id` that contains
|
|
54
|
-
* `cover` on an image manifest item. The href is returned exactly
|
|
55
|
-
* as it appears in the manifest — callers own folder-prefix
|
|
56
|
-
* resolution against the OPF's location in the archive.
|
|
57
|
-
*/
|
|
58
40
|
readonly coverHref: string | undefined;
|
|
59
41
|
readonly renditionLayoutMeta: string | undefined;
|
|
60
42
|
readonly renditionFlowMeta: string | undefined;
|
|
@@ -62,15 +44,4 @@ export type OpfMetadata = {
|
|
|
62
44
|
readonly pageProgressionDirection: string | undefined;
|
|
63
45
|
readonly guide: ReadonlyArray<OpfGuideReference>;
|
|
64
46
|
};
|
|
65
|
-
/**
|
|
66
|
-
* Parses an EPUB package document (OPF) into structured metadata.
|
|
67
|
-
*
|
|
68
|
-
* Direct children of `package` (`metadata`, `manifest`, `spine`, `guide`) and
|
|
69
|
-
* their structural children (`item`, `itemref`, `reference`, `meta`) are
|
|
70
|
-
* matched by **local name** (ASCII case-insensitive), so prefixed tags such as
|
|
71
|
-
* `opf:manifest` are supported the same as unprefixed `manifest`.
|
|
72
|
-
*
|
|
73
|
-
* Attribute names on `spine` / `itemref` are still read as emitted by xmldoc
|
|
74
|
-
* (no QName normalization).
|
|
75
|
-
*/
|
|
76
47
|
export declare const parseOpf: (opfXml: string) => OpfMetadata;
|