@prose-reader/streamer 1.300.0 → 1.301.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/index.d.ts +1 -0
- package/dist/index.js +325 -299
- package/dist/index.js.map +1 -1
- package/dist/index.umd.cjs +12 -12
- package/dist/index.umd.cjs.map +1 -1
- package/dist/utils/createXmlSafeId.d.ts +3 -0
- package/dist/utils/createXmlSafeId.test.d.ts +1 -0
- package/package.json +4 -4
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","sources":["../src/utils/sortByTitleComparator.ts","../src/utils/uri.ts","../src/archives/createArchiveFromArrayBufferList.ts","../src/report.ts","../src/archives/printTree.ts","../src/archives/createArchiveFromJszip.ts","../src/archives/createArchiveFromLibArchive.ts","../src/archives/createArchiveFromText.ts","../src/archives/createArchiveFromUrls.ts","../src/configure.ts","../src/epubs/getArchiveOpfInfo.ts","../src/epubs/readArchiveOpf.ts","../src/generators/manifest/hooks/apple.ts","../src/generators/manifest/hooks/comicInfo.ts","../src/generators/manifest/createManifestResourceHref.ts","../src/generators/manifest/hooks/default.ts","../src/epubs/getSpineItemFilesFromArchive.ts","../src/generators/manifest/hooks/epub/epub.ts","../src/generators/manifest/hooks/epubOptimizer.ts","../src/generators/manifest/hooks/kobo.ts","../src/epubs/isArchiveEpub.ts","../src/generators/manifest/hooks/nonEpub.ts","../src/parsers/xml.ts","../src/parsers/nav.ts","../src/generators/manifest/hooks/audiobookToc.ts","../src/generators/manifest/hooks/toc.ts","../src/generators/manifest/index.ts","../src/generators/resources/hooks/calibreFixHook.ts","../src/generators/resources/hooks/cssFixHook.ts","../src/generators/resources/hooks/defaultHook.ts","../src/generators/resources/hooks/selfClosingTagsFixHook.ts","../src/generators/resources/index.ts","../src/archives/archiveLoader.ts","../src/utils/createRangeResponse.ts","../src/Streamer.ts","../src/ServiceWorkerStreamer.ts"],"sourcesContent":["export const sortByTitleComparator = (a: string, b: string) => {\n const alist = a.split(/(\\d+)/)\n const blist = b.split(/(\\d+)/)\n\n for (let i = 0, len = alist.length; i < len; i++) {\n if (alist[i] !== blist[i]) {\n if (alist[i]?.match(/\\d/)) {\n return +(alist[i] || ``) - +(blist[i] || ``)\n }\n return (alist[i] || ``).localeCompare(blist[i] || ``)\n }\n }\n\n return 1\n}\n","export const getUriBasename = (uri: string) =>\n uri.substring(uri.lastIndexOf(`/`) + 1) || uri\n\nexport const removeTrailingSlash = (uri: string) =>\n uri.endsWith(\"/\") ? uri.slice(0, -1) : uri\n\nexport const getUriBasePath = (uri: string) => {\n const lastSlashIndex = uri.lastIndexOf(\"/\")\n\n return lastSlashIndex >= 0 ? uri.substring(0, lastSlashIndex) : \"\"\n}\n","import { detectMimeTypeFromName } from \"@prose-reader/shared\"\nimport { sortByTitleComparator } from \"../utils/sortByTitleComparator\"\nimport { getUriBasename } from \"../utils/uri\"\nimport type { Archive } from \"./types\"\n\nexport const createArchiveFromArrayBufferList = async (\n list: {\n isDir: boolean\n name: string\n size: number\n data: () => Promise<ArrayBuffer>\n }[],\n { orderByAlpha, name }: { orderByAlpha?: boolean; name?: string } = {},\n): Promise<Archive> => {\n let files = list\n\n if (orderByAlpha) {\n files = files.slice().sort((a, b) => sortByTitleComparator(a.name, b.name))\n }\n\n return {\n filename: name || ``,\n records: files.map((file) => {\n const size = file.size\n const basename = getUriBasename(file.name)\n\n if (file.isDir) {\n return {\n dir: true,\n basename,\n uri: file.name,\n size,\n }\n }\n\n return {\n dir: file.isDir,\n basename,\n encodingFormat: detectMimeTypeFromName(file.name),\n uri: file.name,\n blob: async () =>\n new Blob([await file.data()], {\n type: detectMimeTypeFromName(file.name) ?? ``,\n }),\n string: async () => {\n const data = await file.data()\n return String.fromCharCode.apply(\n null,\n Array.from(new Uint16Array(data)),\n )\n },\n size,\n }\n }),\n close: () => Promise.resolve(),\n }\n}\n","import { Report as SharedReport } from \"@prose-reader/shared\"\nimport { name } from \"../package.json\"\n\nexport const Report = SharedReport.namespace(name, false, {\n color: \"#ffae42\",\n})\n","interface TreeNode {\n [key: string]: TreeNode\n}\n\nexport const printTree = (paths: string[]): string => {\n // Split and collect all parts for tree reconstruction\n const tree: TreeNode = {}\n for (const path of paths) {\n const parts = path.split(\"/\")\n let node = tree\n for (let i = 0; i < parts.length; i++) {\n const part = parts[i]\n if (part === undefined) continue\n if (!node[part]) {\n node[part] = {}\n }\n node = node[part]\n }\n }\n\n // Recursively build the tree string\n const render = (node: TreeNode, indent = \"\"): string => {\n return Object.keys(node)\n .sort()\n .map((key, i, arr) => {\n const isLast = i === arr.length - 1\n const prefix = indent + (isLast ? \"└── \" : \"├── \")\n const nextIndent = indent + (isLast ? \" \" : \"│ \")\n const value = node[key]\n if (value && Object.keys(value).length > 0) {\n return `${prefix}${key}/\\n${render(value, nextIndent)}`\n }\n return `${prefix}${key}`\n })\n .join(\"\\n\")\n }\n\n return render(tree)\n}\n","import { detectMimeTypeFromName } from \"@prose-reader/shared\"\nimport { Report } from \"../report\"\nimport { sortByTitleComparator } from \"../utils/sortByTitleComparator\"\nimport { getUriBasename } from \"../utils/uri\"\nimport { printTree } from \"./printTree\"\nimport type { Archive, StreamResult } from \"./types\"\n\ninterface OutputByType {\n base64: string\n string: string\n text: string\n binarystring: string\n array: number[]\n uint8array: Uint8Array\n arraybuffer: ArrayBuffer\n blob: Blob\n nodebuffer: Buffer\n}\n\ntype OutputType = keyof OutputByType\n\ninterface JSZipObject {\n name: string\n dir: boolean\n date: Date\n comment: string\n unixPermissions: number | string | null\n dosPermissions: number | null\n async<T extends OutputType>(type: T): Promise<OutputByType[T]>\n // nodeStream(type?: `nodebuffer`): NodeJS.ReadableStream;\n internalStream?: (type?: `uint8array`) => StreamResult\n}\n\ninterface JSZip {\n files: { [key: string]: JSZipObject }\n}\n\nexport const createArchiveFromJszip = async (\n jszip: JSZip,\n { orderByAlpha, name }: { orderByAlpha?: boolean; name?: string } = {},\n): Promise<Archive> => {\n let files = Object.values(jszip.files)\n\n if (orderByAlpha) {\n files = files.slice().sort((a, b) => sortByTitleComparator(a.name, b.name))\n }\n\n const archive: Archive = {\n filename: name || ``,\n records: files.map((file) => {\n // this is private API\n // @ts-expect-error\n const size = file._data.uncompressedSize\n const basename = getUriBasename(file.name)\n\n if (file.dir) {\n return {\n dir: true,\n basename,\n uri: file.name,\n size,\n }\n }\n\n return {\n dir: false,\n basename: getUriBasename(file.name),\n uri: file.name,\n encodingFormat: detectMimeTypeFromName(file.name),\n blob: () => file.async(`blob`),\n string: () => file.async(\"string\"),\n ...(file.internalStream && {\n stream: file.internalStream,\n }),\n // this is private API\n // @ts-expect-error\n size: file._data.uncompressedSize,\n }\n }),\n close: () => Promise.resolve(),\n }\n\n Report.log(\"Generated archive\", archive)\n\n if (process.env.NODE_ENV === \"development\") {\n if (Report.isEnabled()) {\n const folderStructureStr = printTree(files.map((file) => file.name))\n Report.groupCollapsed(...Report.getGroupArgs(\"Archive folder structure\"))\n Report.log(`\\n${folderStructureStr}`)\n Report.groupEnd()\n }\n }\n\n return archive\n}\n","/**\n * @see https://github.com/nika-begiashvili/libarchivejs.\n *\n * Does not work in service worker due to usage of web worker.\n */\nimport { detectMimeTypeFromName } from \"@prose-reader/shared\"\nimport { Report } from \"../report\"\nimport type { Archive } from \"./types\"\n\ninterface ArchiveReader {\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n getFilesArray(): Promise<any[]>\n /**\n * Terminate worker to free up memory\n */\n close(): Promise<void>\n}\n\n/**\n * Represents compressed file before extraction\n */\ninterface CompressedFile {\n /**\n * File name\n */\n get name(): string\n /**\n * File size\n */\n get size(): number\n get lastModified(): number\n /**\n * Extract file from archive\n * @returns {Promise<File>} extracted file\n */\n\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n extract(): any\n}\n\nexport const createArchiveFromLibArchive = async (\n libArchive: ArchiveReader,\n { name }: { orderByAlpha?: boolean; name?: string } = {},\n): Promise<Archive> => {\n const objArray = await libArchive.getFilesArray()\n\n const archive: Archive = {\n close: () => libArchive.close(),\n filename: name ?? ``,\n records: objArray.map((item: { file: CompressedFile; path: string }) => ({\n dir: false,\n basename: item.file.name,\n encodingFormat: detectMimeTypeFromName(item.file.name),\n size: item.file.size,\n uri: `${item.path}${item.file.name}`,\n blob: async () => {\n const file = await (item.file.extract() as Promise<File>)\n\n return file\n },\n string: async () => {\n const file = await (item.file.extract() as Promise<File>)\n\n return file.text()\n },\n })),\n }\n\n Report.log(\"Generated archive\", archive)\n\n return archive\n}\n","import { getUriBasename } from \"../utils/uri\"\nimport type { Archive } from \"./types\"\n\n/**\n * Useful to create archive from txt content\n */\nexport const createArchiveFromText = async (\n content: string | Blob,\n {\n mimeType,\n direction,\n }: {\n direction?: `ltr` | `rtl`\n mimeType?: string\n } = { mimeType: \"text/plain\" },\n) => {\n const txtOpfContent = `\n <?xml version=\"1.0\" encoding=\"UTF-8\"?>\n <package xmlns=\"http://www.idpf.org/2007/opf\" version=\"3.0\" xml:lang=\"ja\" prefix=\"rendition: http://www.idpf.org/vocab/rendition/#\"\n unique-identifier=\"ootuya-id\">\n <metadata xmlns:opf=\"http://www.idpf.org/2007/opf\" xmlns:dc=\"http://purl.org/dc/elements/1.1/\"\n xmlns:dcterms=\"http://purl.org/dc/terms/\">\n <meta property=\"rendition:layout\">reflowable</meta>\n </metadata>\n <manifest>\n <item id=\"p01\" href=\"p01.txt\" media-type=\"text/plain\"/>\n </manifest>\n <spine page-progression-direction=\"${direction ?? `ltr`}\">\n <itemref idref=\"p01\" />\n </spine>\n </package>\n `\n\n const archive: Archive = {\n filename: `content.txt`,\n records: [\n {\n dir: false,\n basename: getUriBasename(`generated.opf`),\n uri: `generated.opf`,\n blob: async () => new Blob([txtOpfContent]),\n string: async () => txtOpfContent,\n size: 0,\n },\n {\n dir: false,\n basename: getUriBasename(`p01.txt`),\n uri: `p01.txt`,\n blob: async () => {\n if (typeof content === \"string\") return new Blob([content])\n return content\n },\n string: async () => {\n if (typeof content === \"string\") return content\n return content.text()\n },\n size: typeof content === \"string\" ? content.length : content.size,\n encodingFormat: mimeType,\n },\n ],\n close: () => Promise.resolve(),\n }\n\n return archive\n}\n","import { detectMimeTypeFromName } from \"@prose-reader/shared\"\nimport { getUriBasename } from \"../utils/uri\"\nimport type { Archive } from \"./types\"\n\n/**\n * @important\n * Make sure the urls are on the same origin or the cors header is set otherwise\n * the resource cannot be consumed as it is on the web.\n */\nexport const createArchiveFromUrls = async (\n urls: string[],\n options?: { useRenditionFlow: boolean },\n): Promise<Archive> => {\n const opfFileData = `\n <?xml version=\"1.0\" encoding=\"UTF-8\"?><package xmlns=\"http://www.idpf.org/2007/opf\" version=\"2.0\" unique-identifier=\"bookid\">\n <metadata>\n <meta property=\"rendition:layout\">${options?.useRenditionFlow ? `reflowable` : `pre-paginated`}</meta>\n ${options?.useRenditionFlow ? `<meta property=\"rendition:flow\">scrolled-continuous</meta>` : ``}\n </metadata>\n <manifest>\n ${urls\n .map(\n (url) =>\n `<item id=\"${getUriBasename(url)}\" href=\"${url}\" media-type=\"${detectMimeTypeFromName(url)}\"/>`,\n )\n .join(`\\n`)}\n </manifest>\n <spine>\n ${urls.map((url) => `<itemref idref=\"${getUriBasename(url)}\" />`).join(`\\n`)}\n </spine>\n </package>\n `\n\n const filesFromUrl: Archive[`records`] = urls.map((url) => ({\n dir: false,\n basename: getUriBasename(url),\n encodingFormat: detectMimeTypeFromName(url),\n uri: url,\n size: 100 / urls.length,\n blob: async () => {\n const response = await fetch(url)\n\n return response.blob()\n },\n string: async () => ``,\n }))\n\n const opfFile: Archive[`records`][number] = {\n dir: false,\n basename: `content.opf`,\n uri: `content.opf`,\n size: 0,\n blob: async () => new Blob(),\n string: async () => opfFileData,\n }\n\n return {\n filename: ``,\n records: [opfFile, ...filesFromUrl],\n close: () => Promise.resolve(),\n }\n}\n","import { Report } from \"./report\"\n\nexport const configure = ({\n enableReport,\n}: {\n enableReport?: boolean\n} = {}) => {\n Report.enable(!!enableReport)\n}\n","import type { Archive } from \"../archives/types\"\n\nexport const getArchiveOpfInfo = (archive: Archive) => {\n const filesAsArray = Object.values(archive.records).filter(\n (file) => !file.dir,\n )\n const file = filesAsArray.find((file) => file.uri.endsWith(`.opf`))\n\n return {\n data: file,\n basePath: file?.uri.substring(0, file.uri.lastIndexOf(`/`)) || ``,\n }\n}\n","import { type OpfMetadata, parseOpf } from \"@prose-reader/archive-parser\"\nimport type { Archive } from \"../archives/types\"\nimport { getArchiveOpfInfo } from \"./getArchiveOpfInfo\"\n\nexport type ArchiveOpfParsed = {\n opf: OpfMetadata\n basePath: string\n}\n\n/**\n * Loads and parses the package OPF from `archive`.\n *\n * **Attention — not memoized:** every call re-reads the OPF record and runs\n * `parseOpf`. Manifest generation already avoids redundant parses by awaiting\n * this once and passing `archiveOpf` into manifest hooks. The resource pipeline\n * (`generateResourceFromArchive` → `defaultHook`) invokes this again per\n * resource unless callers later thread parsed OPF in explicitly (optional\n * parameter, archive open hook, etc.).\n */\nexport async function readArchiveOpf(\n archive: Archive,\n): Promise<ArchiveOpfParsed | undefined> {\n const { data: opsFile, basePath } = getArchiveOpfInfo(archive) || {}\n\n if (!opsFile || opsFile.dir) {\n return undefined\n }\n\n const raw = await opsFile.string()\n\n return {\n opf: parseOpf(raw),\n basePath,\n }\n}\n","import {\n APPLE_IBOOKS_DISPLAY_OPTIONS_FILENAME,\n parseAppleDisplayOptionsXml,\n resolveArchiveMetadata,\n} from \"@prose-reader/archive-parser\"\nimport type { Manifest } from \"@prose-reader/shared\"\nimport type { Archive } from \"../../../archives/types\"\n\nconst appleDisplayOptionsBasename =\n APPLE_IBOOKS_DISPLAY_OPTIONS_FILENAME.toLowerCase()\n\nexport const apple =\n ({ archive }: { archive: Archive; baseUrl: string }) =>\n async (manifest: Manifest): Promise<Manifest> => {\n const infoFile = archive.records.find(\n (file) =>\n !file.dir &&\n file.basename.toLowerCase() === appleDisplayOptionsBasename,\n )\n\n if (!infoFile || infoFile.dir) {\n return manifest\n }\n\n const content = await infoFile.string()\n\n try {\n const parsed = parseAppleDisplayOptionsXml(content)\n const { renditionLayout } = resolveArchiveMetadata(parsed)\n\n return {\n ...manifest,\n renditionLayout: manifest.renditionLayout ?? renditionLayout,\n }\n } catch (e) {\n console.error(\n `Unable to parse ${APPLE_IBOOKS_DISPLAY_OPTIONS_FILENAME} for content\\n`,\n content,\n )\n console.error(e)\n\n return manifest\n }\n }\n","import {\n COMIC_INFO_FILENAME,\n parseComicInfo,\n resolveArchiveMetadata,\n} from \"@prose-reader/archive-parser\"\nimport type { Manifest } from \"@prose-reader/shared\"\nimport type { Archive } from \"../../../archives/types\"\n\nconst comicInfoFilenameLower = COMIC_INFO_FILENAME.toLowerCase()\n\nexport const comicInfo =\n ({ archive }: { archive: Archive; baseUrl: string }) =>\n async (manifest: Manifest): Promise<Manifest> => {\n const comicInfoFile = archive.records.find(\n (file) =>\n file.basename.toLowerCase() === comicInfoFilenameLower && !file.dir,\n )\n\n if (!comicInfoFile || comicInfoFile.dir) {\n return manifest\n }\n\n const manifestWithoutComicInfo = {\n ...manifest,\n spineItems: manifest.spineItems\n .filter(\n (item) => !item.id.toLowerCase().endsWith(comicInfoFilenameLower),\n )\n .map((item, _, items) => ({\n ...item,\n progressionWeight: 1 / items.length,\n })),\n }\n\n const content = await comicInfoFile.string()\n\n try {\n const parsed = parseComicInfo(content)\n const resolved = resolveArchiveMetadata(parsed)\n\n return {\n ...manifestWithoutComicInfo,\n readingDirection: resolved.readingDirection ?? `ltr`,\n }\n } catch (e) {\n console.error(\n `Unable to parse ${COMIC_INFO_FILENAME} for content\\n`,\n content,\n )\n console.error(e)\n\n return manifestWithoutComicInfo\n }\n }\n","export const createManifestResourceHref = ({\n baseUrl = ``,\n resourcePath,\n}: {\n baseUrl?: string\n resourcePath: string\n}) => {\n if (!baseUrl && /^https?:\\/\\//.test(resourcePath)) {\n return encodeURI(resourcePath)\n }\n\n const hrefBaseUrl = baseUrl\n ? `${baseUrl}${baseUrl.endsWith(`/`) ? `` : `/`}`\n : `file://`\n\n return encodeURI(`${hrefBaseUrl}${resourcePath}`)\n}\n","import type { Manifest } from \"@prose-reader/shared\"\nimport type { Archive } from \"../../../archives/types\"\nimport { createManifestResourceHref } from \"../createManifestResourceHref\"\n\nexport const defaultHook =\n ({ archive, baseUrl }: { archive: Archive; baseUrl: string }) =>\n async (): Promise<Manifest> => {\n const files = Object.values(archive.records).filter((file) => !file.dir)\n\n return {\n filename: archive.filename,\n title:\n archive.records.find(({ dir }) => dir)?.basename.replace(/\\/$/, ``) ||\n archive.filename,\n renditionLayout: undefined,\n renditionSpread: `auto`,\n readingDirection: `ltr`,\n spineItems: files\n .filter((file) => !file.basename.endsWith(`.db`))\n .map((file, index) => {\n return {\n // some books such as cbz can have same basename inside different sub folder\n // we need to make sure to have unique index\n // /chap01/01.png, /chap02/01.png, etc\n id: `${index}.${file.basename}`,\n index,\n href: createManifestResourceHref({\n baseUrl,\n resourcePath: file.uri,\n }),\n renditionLayout: undefined,\n progressionWeight: 1 / files.length,\n pageSpreadLeft: undefined,\n pageSpreadRight: undefined,\n mediaType: file.encodingFormat,\n }\n }),\n items: files.map((file, index) => ({\n id: `${index}.${file.basename}`,\n href: encodeURI(`${baseUrl}${file.uri}`),\n })),\n }\n }\n","import type { Archive } from \"../archives/types\"\nimport type { ArchiveOpfParsed } from \"./readArchiveOpf\"\n\nexport const getSpineItemFilesFromArchive = async ({\n archive,\n archiveOpf,\n}: {\n archive: Archive\n archiveOpf: ArchiveOpfParsed | undefined\n}) => {\n if (!archiveOpf) return []\n\n const { opf, basePath: opfBasePath } = archiveOpf\n const { spineRows } = opf\n\n const archiveSpineItems = archive.records.filter((file) => {\n return spineRows.find((item) => {\n if (!opfBasePath) return `${item.href}` === file.uri\n return `${opfBasePath}/${item.href}` === file.uri\n })\n })\n\n return archiveSpineItems\n}\n","import { resolveArchiveMetadata } from \"@prose-reader/archive-parser\"\nimport type { Manifest } from \"@prose-reader/shared\"\nimport type { Archive } from \"../../../../archives/types\"\nimport { getArchiveOpfInfo } from \"../../../../epubs/getArchiveOpfInfo\"\nimport { getSpineItemFilesFromArchive } from \"../../../../epubs/getSpineItemFilesFromArchive\"\nimport type { ArchiveOpfParsed } from \"../../../../epubs/readArchiveOpf\"\nimport { Report } from \"../../../../report\"\n\nexport const getItemsFromDoc = (\n manifestItems: ReadonlyArray<{\n id: string\n href: string\n mediaType?: string\n }>,\n archive: Archive,\n getBaseUrlForHref?: (href: string) => string,\n) => {\n const { basePath: opfBasePath } = getArchiveOpfInfo(archive) || {}\n\n return manifestItems.map((el) => {\n const href = el.href\n const baseUrl = getBaseUrlForHref?.(href) ?? ``\n\n return {\n href: opfBasePath\n ? `${baseUrl}${opfBasePath}/${href}`\n : `${baseUrl}${href}`,\n id: el.id,\n mediaType: el.mediaType,\n }\n })\n}\n\nconst manifestRenditionFlow = (\n raw: string | undefined,\n): NonNullable<Manifest[`renditionFlow`]> => {\n const v = raw?.trim()\n if (\n v === `scrolled-continuous` ||\n v === `scrolled-doc` ||\n v === `paginated` ||\n v === `auto`\n ) {\n return v\n }\n return `auto`\n}\n\nconst manifestRenditionSpread = (\n raw: string | undefined,\n): Manifest[`renditionSpread`] => {\n const v = raw?.trim()\n if (\n v === `none` ||\n v === `landscape` ||\n v === `portrait` ||\n v === `both` ||\n v === `auto`\n ) {\n return v\n }\n return undefined\n}\n\ntype ManifestGuideEntry = NonNullable<Manifest[`guide`]>[number]\n\nconst manifestGuideType = (\n raw: string | undefined,\n): ManifestGuideEntry[`type`] | undefined => {\n const v = raw?.trim()\n if (\n v === `cover` ||\n v === `title-page` ||\n v === `copyright-page` ||\n v === `text`\n ) {\n return v\n }\n return undefined\n}\n\nexport const epubHook =\n ({\n archive,\n baseUrl,\n archiveOpf,\n }: {\n archive: Archive\n baseUrl: string\n archiveOpf: ArchiveOpfParsed | undefined\n }) =>\n async (manifest: Manifest): Promise<Manifest> => {\n if (!archiveOpf) {\n return manifest\n }\n\n const { opf, basePath: opfBasePath } = archiveOpf\n const resolved = resolveArchiveMetadata(opf)\n\n Report.groupCollapsed(...Report.getGroupArgs(`OPF parsed`))\n Report.log(`opf`, opf)\n Report.groupEnd()\n\n const publisherRenditionLayout = opf.renditionLayoutMeta?.trim()\n const packageRenditionLayout =\n publisherRenditionLayout === `reflowable` ||\n publisherRenditionLayout === `pre-paginated`\n ? publisherRenditionLayout\n : resolved.renditionLayout\n\n const title =\n opf.title?.trim() ||\n archive.records.find(({ dir }) => dir)?.basename ||\n ``\n\n const readingDirection = resolved.readingDirection ?? `ltr`\n\n const archiveSpineItems = await getSpineItemFilesFromArchive({\n archive,\n archiveOpf,\n })\n\n const totalSize = archiveSpineItems.reduce(\n (size, file) => file.size + size,\n 0,\n )\n\n const guideRefs = opf.guide\n const guide: ManifestGuideEntry[] = []\n for (const elm of guideRefs) {\n const type = manifestGuideType(elm.type)\n if (type === undefined) continue\n guide.push({ href: elm.href, title: elm.title, type })\n }\n\n return {\n filename: archive.filename,\n renditionLayout: packageRenditionLayout,\n renditionFlow: manifestRenditionFlow(opf.renditionFlowMeta),\n renditionSpread: manifestRenditionSpread(opf.renditionSpreadMeta),\n title,\n readingDirection,\n /**\n * @see https://www.w3.org/TR/epub/#sec-itemref-elem\n */\n spineItems: opf.spineRows.map((row, index) => {\n const itemSize =\n archive.records.find((file) => file.uri.endsWith(row.href))?.size || 0\n\n const hrefBaseUri = baseUrl\n ? baseUrl\n : /^https?:\\/\\//.test(row.href)\n ? ``\n : `file://`\n\n return {\n id: row.id,\n index,\n href: row.href.startsWith(`https://`)\n ? row.href\n : opfBasePath\n ? `${hrefBaseUri}${opfBasePath}/${row.href}`\n : `${hrefBaseUri}${row.href}`,\n renditionLayout: row.renditionLayout ?? packageRenditionLayout,\n ...(row.renditionFlow !== undefined\n ? { renditionFlow: row.renditionFlow }\n : {}),\n progressionWeight: itemSize / totalSize,\n pageSpreadLeft: row.pageSpreadLeft,\n pageSpreadRight: row.pageSpreadRight,\n mediaType: row.mediaType,\n }\n }),\n items: getItemsFromDoc(opf.manifestItems, archive, (href) => {\n if (/^https?:\\/\\//.test(href)) {\n return ``\n }\n\n return baseUrl || `file://`\n }),\n guide: guide.length > 0 ? guide : undefined,\n }\n }\n","import { isXmlBasedMimeType, type Manifest } from \"@prose-reader/shared\"\nimport { XmlDocument } from \"xmldoc\"\nimport type { Archive } from \"../../../archives/types\"\nimport { getSpineItemFilesFromArchive } from \"../../../epubs/getSpineItemFilesFromArchive\"\nimport type { ArchiveOpfParsed } from \"../../../epubs/readArchiveOpf\"\n\nconst hasDocMetaViewport = (doc: XmlDocument) => {\n const metaElm = doc\n .descendantWithPath(\"head\")\n ?.childrenNamed(\"meta\")\n .find((node) => node.attr.name === \"viewport\")\n\n return !!(metaElm && metaElm.attr.name === \"viewport\")\n}\n\nconst allFilesHaveViewportMeta = (files: Archive[\"records\"]) =>\n files.reduce(async (result, current) => {\n const _result = await result\n\n if (!_result) return false\n\n if (\n !isXmlBasedMimeType({\n mimeType: current.encodingFormat,\n uri: current.uri,\n })\n ) {\n return false\n }\n\n const file = current.dir ? null : await current.string()\n\n if (!file) return false\n\n return hasDocMetaViewport(new XmlDocument(file))\n }, Promise.resolve(true))\n\nexport const epubOptimizerHook =\n ({\n archive,\n archiveOpf,\n }: {\n archive: Archive\n baseUrl: string\n archiveOpf: ArchiveOpfParsed | undefined\n }) =>\n async (manifest: Manifest): Promise<Manifest> => {\n const bookIsFullReflowable =\n manifest.renditionLayout === \"reflowable\" &&\n manifest.spineItems.every((item) => item.renditionLayout === \"reflowable\")\n\n if (bookIsFullReflowable) {\n const files = await getSpineItemFilesFromArchive({\n archive,\n archiveOpf,\n })\n\n const hasAllViewport = await allFilesHaveViewportMeta(files)\n\n if (hasAllViewport) {\n return {\n ...manifest,\n spineItems: manifest.spineItems.map((item) => ({\n ...item,\n renditionLayout: \"pre-paginated\",\n })),\n renditionLayout: \"pre-paginated\",\n }\n }\n }\n\n return manifest\n }\n","import {\n KOBO_DISPLAY_OPTIONS_FILENAME,\n type KoboMetadata,\n parseKoboXml,\n resolveArchiveMetadata,\n} from \"@prose-reader/archive-parser\"\nimport type { Manifest } from \"@prose-reader/shared\"\nimport type { Archive } from \"../../../archives/types\"\n\nconst extractKoboInformationFromArchive = async (\n archive: Archive,\n): Promise<KoboMetadata> => {\n let renditionLayout: KoboMetadata[\"renditionLayout\"]\n\n await Promise.all(\n archive.records.map(async (file) => {\n if (file.dir || !file.uri.endsWith(KOBO_DISPLAY_OPTIONS_FILENAME)) return\n\n const content = await file.string()\n\n try {\n const { renditionLayout: layout } = parseKoboXml(content)\n if (layout) renditionLayout = layout\n } catch (e) {\n console.error(\n `Unable to parse ${KOBO_DISPLAY_OPTIONS_FILENAME} for content\\n`,\n content,\n )\n console.error(e)\n }\n }),\n )\n\n return {\n kind: `kobo`,\n ...(renditionLayout !== undefined ? { renditionLayout } : {}),\n }\n}\n\nexport const kobo =\n ({ archive }: { archive: Archive; baseUrl: string }) =>\n async (manifest: Manifest): Promise<Manifest> => {\n const koboMeta = await extractKoboInformationFromArchive(archive)\n const { renditionLayout } = resolveArchiveMetadata(koboMeta)\n\n return {\n ...manifest,\n renditionLayout: manifest.renditionLayout ?? renditionLayout,\n }\n }\n","import type { Archive } from \"../archives/types\"\n\nconst hasOpfExtension = (path: string) => path.toLowerCase().endsWith(`.opf`)\n\nexport const isArchiveEpub = (archive: Archive) => {\n return archive.records.some(\n (file) =>\n !file.dir &&\n (hasOpfExtension(file.basename) || hasOpfExtension(file.uri)),\n )\n}\n","import {\n detectMimeTypeFromName,\n isMediaContentMimeType,\n type Manifest,\n parseContentType,\n} from \"@prose-reader/shared\"\nimport type { Archive } from \"../../../archives/types\"\nimport { isArchiveEpub } from \"../../../epubs/isArchiveEpub\"\n\n/**\n * If we don't have a regular epub, we have an archive that could be many things.\n * We try to refine the manifest based on the type of content as much as possible.\n */\nexport const nonEpub =\n ({ archive }: { archive: Archive; baseUrl: string }) =>\n async (manifest: Manifest): Promise<Manifest> => {\n const isEpub = isArchiveEpub(archive)\n\n if (isEpub) return manifest\n\n return {\n ...manifest,\n spineItems: manifest.spineItems.map((spineItem) => {\n const archiveItem = archive.records.find((item) =>\n decodeURI(spineItem.href).endsWith(item.uri),\n )\n\n const mimeType =\n parseContentType(archiveItem?.encodingFormat ?? \"\") ||\n detectMimeTypeFromName(archiveItem?.basename ?? \"\")\n\n return {\n ...spineItem,\n renditionLayout:\n mimeType && isMediaContentMimeType(mimeType)\n ? `pre-paginated`\n : spineItem.renditionLayout,\n }\n }),\n }\n }\n","import { XmlElement, XmlTextNode } from \"xmldoc\"\n\nexport const getXmlElementInnerText = (\n node: XmlElement | undefined,\n): string => {\n if (!node) return \"\"\n\n return node.children\n .map((child) => {\n if (child instanceof XmlTextNode) return child.text\n if (child instanceof XmlElement) return getXmlElementInnerText(child)\n return \"\"\n })\n .join(\"\")\n .trim()\n}\n","import {\n type OpfMetadata,\n tokenizeXmlSpaceSeparatedList,\n} from \"@prose-reader/archive-parser\"\nimport type { Manifest } from \"@prose-reader/shared\"\nimport { urlJoin } from \"@prose-reader/shared\"\nimport { XmlDocument, type XmlElement, type XmlNodeBase } from \"xmldoc\"\nimport { type Archive, getArchiveOpfInfo } from \"..\"\nimport { getUriBasePath } from \"../utils/uri\"\nimport { getXmlElementInnerText } from \"./xml\"\n\ntype Toc = NonNullable<Manifest[`nav`]>[`toc`]\ntype TocItem = NonNullable<Manifest[`nav`]>[`toc`][number]\n\nconst manifestItemIsNavDocument = (item: {\n readonly properties?: string\n}): boolean => tokenizeXmlSpaceSeparatedList(item.properties).includes(`nav`)\n\n/**\n * @see https://www.w3.org/TR/epub-33/#sec-nav-def-model\n */\nconst extractNavChapter = (\n li: XmlElement,\n { basePath, baseUrl }: { basePath: string; baseUrl: string },\n) => {\n const chp: TocItem = {\n contents: [],\n path: ``,\n href: ``,\n title: ``,\n }\n\n let contentNode = li.childNamed(`span`) || li.childNamed(`a`)\n\n chp.title =\n (contentNode?.attr.title ||\n contentNode?.val.trim() ||\n getXmlElementInnerText(contentNode)) ??\n ``\n\n let node = contentNode?.name\n\n if (node !== `a`) {\n contentNode = li.descendantWithPath(`${node}.a`)\n if (contentNode) {\n node = contentNode.name.toLowerCase()\n }\n }\n\n if (node === `a` && contentNode?.attr.href) {\n chp.path = urlJoin(basePath, contentNode.attr.href)\n chp.href = urlJoin(baseUrl, basePath, contentNode.attr.href)\n }\n const sublistNode = li.childNamed(`ol`)\n if (sublistNode) {\n const children = sublistNode.childrenNamed(`li`)\n if (children && children.length > 0) {\n chp.contents = children.map((child) =>\n extractNavChapter(child, { basePath, baseUrl }),\n )\n }\n }\n\n return chp\n}\n\nconst buildTOCFromNav = (\n doc: XmlDocument,\n { basePath, baseUrl }: { basePath: string; baseUrl: string },\n) => {\n const toc: Toc = []\n\n let navDataChildren: XmlNodeBase[] | undefined\n\n if (doc.descendantWithPath(`body.nav.ol`)) {\n navDataChildren = doc.descendantWithPath(`body.nav.ol`)?.children\n } else if (doc.descendantWithPath(`body.section.nav.ol`)) {\n navDataChildren = doc.descendantWithPath(`body.section.nav.ol`)?.children\n }\n\n if (navDataChildren && navDataChildren.length > 0) {\n navDataChildren\n .filter((li) => (li as XmlElement).name === `li`)\n .forEach((li) => {\n toc.push(extractNavChapter(li as XmlElement, { basePath, baseUrl }))\n })\n }\n\n return toc\n}\n\nconst parseTocFromNavPath = async (\n opf: OpfMetadata,\n archive: Archive,\n { baseUrl }: { baseUrl: string },\n) => {\n const navItem = opf.manifestItems.find(manifestItemIsNavDocument)\n\n if (navItem?.href) {\n const tocFile = Object.values(archive.records).find((item) =>\n item.uri.endsWith(navItem.href),\n )\n\n if (tocFile && !tocFile.dir) {\n const doc = new XmlDocument(await tocFile.string())\n\n const tocFileBasePath = getUriBasePath(tocFile.uri)\n\n /**\n * links inside toc.xhtml are relative to the toc.xhtml file,\n * not the opf file anymore\n */\n return buildTOCFromNav(doc, { basePath: tocFileBasePath, baseUrl })\n }\n }\n}\n\nconst mapNcxChapter = (\n point: XmlElement,\n {\n opfBasePath,\n baseUrl,\n prefix,\n }: { opfBasePath: string; baseUrl: string; prefix: string },\n) => {\n const src = point?.childNamed(`${prefix}content`)?.attr.src || ``\n\n const out: TocItem = {\n title:\n point?.descendantWithPath(`${prefix}navLabel.${prefix}text`)?.val || ``,\n path: urlJoin(opfBasePath, src),\n href: urlJoin(baseUrl, opfBasePath, src),\n contents: [],\n }\n const children = point.childrenNamed(`${prefix}navPoint`)\n if (children && children.length > 0) {\n out.contents = children.map((pt) =>\n mapNcxChapter(pt, { opfBasePath, baseUrl, prefix }),\n )\n }\n\n return out\n}\n\nconst buildTOCFromNCX = (\n ncxData: XmlDocument,\n { opfBasePath, baseUrl }: { opfBasePath: string; baseUrl: string },\n) => {\n const toc: NonNullable<Manifest[`nav`]>[`toc`] = []\n\n const rootTagName = ncxData.name\n let prefix = ``\n if (rootTagName.indexOf(`:`) !== -1) {\n prefix = `${rootTagName.split(`:`)[0]}:`\n }\n\n ncxData\n .childNamed(`${prefix}navMap`)\n ?.childrenNamed(`${prefix}navPoint`)\n .forEach((point) => {\n toc.push(mapNcxChapter(point, { opfBasePath, baseUrl, prefix }))\n })\n\n return toc\n}\n\nconst parseTocFromNcx = async ({\n opf,\n opfBasePath,\n baseUrl,\n archive,\n}: {\n opf: OpfMetadata\n opfBasePath: string\n archive: Archive\n baseUrl: string\n}) => {\n const ncxId = opf.spineTocIdref\n\n if (ncxId) {\n const ncxItem = opf.manifestItems.find((item) => item.id === ncxId)\n\n if (ncxItem) {\n const ncxPath = `${opfBasePath}${opfBasePath === `` ? `` : `/`}${ncxItem.href}`\n\n const file = Object.values(archive.records).find((item) =>\n item.uri.endsWith(ncxPath),\n )\n\n if (file && !file.dir) {\n const ncxData = new XmlDocument(await file.string())\n\n return buildTOCFromNCX(ncxData, { opfBasePath, baseUrl })\n }\n }\n }\n}\n\nexport const parseToc = async (\n opf: OpfMetadata,\n archive: Archive,\n { baseUrl }: { baseUrl: string },\n) => {\n const { basePath: opfBasePath } = getArchiveOpfInfo(archive) || {}\n\n const tocFromNav = await parseTocFromNavPath(opf, archive, {\n baseUrl,\n })\n\n if (tocFromNav) {\n return tocFromNav\n }\n\n const tocFromNcx = await parseTocFromNcx({\n opf,\n opfBasePath: opfBasePath ?? ``,\n archive,\n baseUrl,\n })\n\n if (tocFromNcx) {\n return tocFromNcx\n }\n}\n","import {\n detectMimeTypeFromName,\n type Manifest,\n parseContentType,\n} from \"@prose-reader/shared\"\nimport type { Archive } from \"../../../archives/types\"\n\ntype Toc = NonNullable<Manifest[\"nav\"]>[\"toc\"]\n\nexport const normalizeFilenameAsTitle = (filename: string) =>\n filename\n .replace(/\\.[^.]+$/, \"\")\n .replace(/[_-]/g, \" \")\n .replace(/\\s+/g, \" \")\n .trim()\n\nexport const buildAudiobookToc = (\n manifest: Manifest,\n archive: Archive,\n): Toc | undefined => {\n if (manifest.spineItems.length === 0) return undefined\n\n const allAudio = manifest.spineItems.every((item) => {\n const mimeType =\n parseContentType(item.mediaType ?? \"\") ||\n detectMimeTypeFromName(item.href)\n\n return mimeType?.startsWith(\"audio/\")\n })\n\n if (!allAudio) return undefined\n\n return manifest.spineItems.map((item) => {\n const record = archive.records.find(\n (r) => !r.dir && decodeURI(item.href).endsWith(r.uri),\n )\n\n return {\n title: normalizeFilenameAsTitle(record?.basename ?? item.href),\n href: item.href,\n path: record?.uri ?? item.href,\n contents: [],\n }\n })\n}\n","import { type Manifest, urlJoin } from \"@prose-reader/shared\"\nimport type { Archive } from \"../../../archives/types\"\nimport type { ArchiveOpfParsed } from \"../../../epubs/readArchiveOpf\"\nimport { parseToc } from \"../../../parsers/nav\"\nimport { sortByTitleComparator } from \"../../../utils/sortByTitleComparator\"\nimport { buildAudiobookToc } from \"./audiobookToc\"\n\ntype Toc = NonNullable<Manifest[\"nav\"]>[\"toc\"]\ntype TocItem = NonNullable<Manifest[\"nav\"]>[\"toc\"][number]\n\nconst buildFolderFallbackToc = (\n archive: Archive,\n { baseUrl }: { baseUrl: string },\n): Toc => {\n const filesSortedByAlpha = [...archive.records].sort((a, b) =>\n sortByTitleComparator(a.uri, b.uri),\n )\n\n const combineWith = (\n toc: TocItem[],\n folder: string,\n subFolders: string[],\n href: string,\n path: string,\n ): TocItem[] => {\n const foundEntry = toc.find((entry) => entry.title === folder)\n const [nextFolderCursor, ...nextSubFolders] = subFolders\n\n if (foundEntry) {\n if (nextFolderCursor) {\n return [\n ...toc.filter((entry) => entry !== foundEntry),\n {\n ...foundEntry,\n contents: [\n ...foundEntry.contents,\n ...combineWith(\n foundEntry.contents,\n nextFolderCursor,\n nextSubFolders,\n href,\n path,\n ),\n ],\n } satisfies TocItem,\n ]\n }\n\n const previousRegisteredPathWasLonger =\n foundEntry.path.split(\"/\").length > path.split(\"/\").length\n\n if (previousRegisteredPathWasLonger) {\n return [\n ...toc.filter((entry) => entry !== foundEntry),\n {\n ...foundEntry,\n path,\n href,\n } satisfies TocItem,\n ]\n }\n\n return toc\n }\n\n if (nextFolderCursor) {\n return [\n ...toc,\n {\n contents: combineWith(\n [],\n nextFolderCursor,\n nextSubFolders,\n href,\n path,\n ),\n href,\n path,\n title: folder,\n },\n ]\n }\n\n return [\n ...toc,\n {\n contents: [],\n href,\n path,\n title: folder,\n },\n ]\n }\n\n return filesSortedByAlpha.reduce((acc, file) => {\n if (file.dir) return acc\n\n const folders = file.uri.split(\"/\").slice(0, -1)\n const [firstFolder, ...restFolders] = folders\n\n if (!firstFolder) return acc\n\n const href = urlJoin(baseUrl, encodeURI(file.uri)).replace(/\\/$/, \"\")\n const path = file.uri.replace(/\\/$/, \"\")\n\n return combineWith(acc, firstFolder, restFolders, href, path)\n }, [] as Toc)\n}\n\nconst resolveTocFromArchive = async (\n archive: Archive,\n manifest: Manifest,\n {\n baseUrl,\n archiveOpf,\n }: { baseUrl: string; archiveOpf: ArchiveOpfParsed | undefined },\n): Promise<Toc | undefined> => {\n if (archiveOpf) {\n const toc = await parseToc(archiveOpf.opf, archive, { baseUrl })\n\n // Keep explicit empty TOC for EPUB-like inputs even when there is no nav file.\n return toc || []\n }\n\n const audiobookToc = buildAudiobookToc(manifest, archive)\n if (audiobookToc) return audiobookToc\n\n const toc = buildFolderFallbackToc(archive, { baseUrl })\n\n if (toc.length === 0) {\n return undefined\n }\n\n return toc\n}\n\n/**\n * Resolve the table of contents from a single entry point.\n * Internally handles EPUB nav, NCX, and folder fallback.\n */\nexport const tocHook =\n ({\n archive,\n baseUrl,\n archiveOpf,\n }: {\n archive: Archive\n baseUrl: string\n archiveOpf: ArchiveOpfParsed | undefined\n }) =>\n async (manifest: Manifest): Promise<Manifest> => {\n if (manifest.nav) return manifest\n\n const toc = await resolveTocFromArchive(archive, manifest, {\n baseUrl,\n archiveOpf,\n })\n if (!toc) return manifest\n\n return {\n ...manifest,\n nav: {\n toc,\n },\n }\n }\n","import type { Archive } from \"../../archives/types\"\nimport { readArchiveOpf } from \"../../epubs/readArchiveOpf\"\nimport type { StreamerManifestHooks } from \"../../hooks\"\nimport { Report } from \"../../report\"\nimport { apple } from \"./hooks/apple\"\nimport { comicInfo } from \"./hooks/comicInfo\"\nimport { defaultHook } from \"./hooks/default\"\nimport { epubHook } from \"./hooks/epub/epub\"\nimport { epubOptimizerHook } from \"./hooks/epubOptimizer\"\nimport { kobo } from \"./hooks/kobo\"\nimport { nonEpub } from \"./hooks/nonEpub\"\nimport { tocHook } from \"./hooks/toc\"\n\nconst normalizeBaseUrl = (baseUrl: string | undefined) => {\n if (!baseUrl) return ``\n\n return baseUrl.endsWith(`/`) ? baseUrl : `${baseUrl}/`\n}\n\nexport const generateManifestFromArchive = async (\n archive: Archive,\n {\n baseUrl = ``,\n hooks = {},\n }: { baseUrl?: string; hooks?: StreamerManifestHooks } = {},\n) => {\n const archiveOpf = await readArchiveOpf(archive)\n const normalizedBaseUrl = normalizeBaseUrl(baseUrl)\n const createExternalHooks = (\n hookFactories: StreamerManifestHooks[keyof StreamerManifestHooks],\n ) =>\n (hookFactories ?? []).map((hook) =>\n hook({ archive, baseUrl: normalizedBaseUrl }),\n )\n\n const contentHooks = [\n epubHook({ archive, baseUrl: normalizedBaseUrl, archiveOpf }),\n comicInfo({ archive, baseUrl: normalizedBaseUrl }),\n apple({ archive, baseUrl: normalizedBaseUrl }),\n nonEpub({ archive, baseUrl: normalizedBaseUrl }),\n ...createExternalHooks(hooks.content),\n ]\n const spineHooks = createExternalHooks(hooks.spine)\n const presentationHooks = [\n epubOptimizerHook({ archive, baseUrl: normalizedBaseUrl, archiveOpf }),\n kobo({ archive, baseUrl: normalizedBaseUrl }),\n ...createExternalHooks(hooks.presentation),\n ]\n const navigationHooks = [\n tocHook({ archive, baseUrl: normalizedBaseUrl, archiveOpf }),\n ...createExternalHooks(hooks.navigation),\n ]\n const manifestHooks = [\n ...contentHooks,\n ...spineHooks,\n ...presentationHooks,\n ...navigationHooks,\n ]\n\n try {\n const baseManifestPromise = defaultHook({\n archive,\n baseUrl: normalizedBaseUrl,\n })()\n\n const manifest = await manifestHooks.reduce(async (manifest, gen) => {\n return await gen(await manifest)\n }, baseManifestPromise)\n\n Report.log(\"Generated manifest\", manifest)\n\n if (process.env.NODE_ENV === \"development\") {\n if (Report.isEnabled()) {\n const manifestStr = JSON.stringify(manifest, null, 2)\n Report.groupCollapsed(...Report.getGroupArgs(\"Generated manifest\"))\n Report.log(`\\n${manifestStr}`)\n Report.groupEnd()\n }\n }\n\n return manifest\n } catch (e) {\n Report.error(e)\n\n throw e\n }\n}\n","import { XmlDocument } from \"xmldoc\"\nimport type { Archive } from \"../../../archives/types\"\nimport type { HookResource } from \"./types\"\n\nconst hasCalibreCoverMeta = (doc: XmlDocument) => {\n const metaElm = doc\n .descendantWithPath(\"head\")\n ?.childrenNamed(\"meta\")\n .find((node) => node.attr.name === \"calibre:cover\")\n\n return !!(metaElm && metaElm.attr.name === \"calibre:cover\")\n}\n\nconst getBuggyCoverSvg = (doc: XmlDocument) => {\n return doc\n .descendantWithPath(\"body\")\n ?.descendantWithPath(\"div\")\n ?.childrenNamed(\"svg\")\n ?.find(\n (node) =>\n node.attr.width === \"100%\" && node.attr.preserveAspectRatio === \"none\",\n )\n}\n\nconst fixBuggyCover =\n ({ archive, resourcePath }: { archive: Archive; resourcePath: string }) =>\n async (resource: HookResource): Promise<HookResource> => {\n const file = Object.values(archive.records).find(\n (file) => file.uri === resourcePath && !file.dir,\n )\n\n if (file && !file.dir && file.basename.endsWith(`.xhtml`)) {\n const bodyToParse =\n typeof resource.body === `string` ? resource.body : await file.string()\n\n const opfXmlDoc = new XmlDocument(bodyToParse)\n\n if (hasCalibreCoverMeta(opfXmlDoc)) {\n const buggySvg = getBuggyCoverSvg(opfXmlDoc)\n\n if (buggySvg) {\n delete buggySvg.attr.preserveAspectRatio\n }\n\n return {\n ...resource,\n body: opfXmlDoc?.toString(),\n }\n }\n }\n\n return resource\n }\n\nexport const calibreFixHook =\n ({ archive, resourcePath }: { archive: Archive; resourcePath: string }) =>\n async (resource: HookResource): Promise<HookResource> => {\n return fixBuggyCover({ archive, resourcePath })(resource)\n }\n","import type { Archive } from \"../../../archives/types\"\nimport type { HookResource } from \"./types\"\n\nexport const cssFixHook =\n ({ archive, resourcePath }: { archive: Archive; resourcePath: string }) =>\n async (resource: HookResource): Promise<HookResource> => {\n const file = Object.values(archive.records).find(\n (file) => file.uri === resourcePath && !file.dir,\n )\n\n if (file && !file.dir && file.basename.endsWith(`.css`)) {\n const bodyToParse =\n typeof resource.body === `string` ? resource.body : await file.string()\n\n /**\n * Fix the potentially invalid writing mode present on some vertical book.\n * This has the benefit of making it compatible with firefox as well.\n */\n const newBody = bodyToParse.replaceAll(\n `-webkit-writing-mode`,\n `writing-mode`,\n )\n\n return {\n ...resource,\n body: newBody,\n }\n }\n\n return resource\n }\n","import type { Archive } from \"../../../archives/types\"\nimport { readArchiveOpf } from \"../../../epubs/readArchiveOpf\"\nimport { getItemsFromDoc } from \"../../manifest/hooks/epub/epub\"\nimport type { HookResource } from \"./types\"\n\n/**\n * Resolves `contentType` from the package OPF when possible, else from the URI.\n * Uses `readArchiveOpf` each time — see `generateResourceFromArchive` JSDoc for\n * why that can mean repeated OPF work on the resource path.\n */\nconst getMetadata = async (archive: Archive, resourcePath: string) => {\n const parsed = await readArchiveOpf(archive)\n\n if (parsed) {\n const { opf } = parsed\n const items = getItemsFromDoc(opf.manifestItems, archive, () => \"\")\n\n // we are comparing opf items relative absolute path in epub archive\n // against resourcePatch (which are absolute path in archive).\n // They should in theory match.\n const foundMediaType = items.find((item) =>\n resourcePath.endsWith(item.href),\n )?.mediaType\n\n if (foundMediaType) {\n return {\n mediaType: items.find((item) => resourcePath.endsWith(item.href))\n ?.mediaType,\n }\n }\n }\n\n return {\n mediaType: getContentTypeFromExtension(resourcePath),\n }\n}\n\nconst getContentTypeFromExtension = (uri: string) => {\n if (uri.endsWith(`.css`)) {\n return `text/css; charset=UTF-8`\n }\n if (uri.endsWith(`.jpg`)) {\n return `image/jpg`\n }\n if (uri.endsWith(`.xhtml`)) {\n return `application/xhtml+xml`\n }\n if (uri.endsWith(`.mp4`)) {\n return `video/mp4`\n }\n if (uri.endsWith(`.svg`)) {\n return `image/svg+xml`\n }\n}\n\nexport const defaultHook =\n ({ archive, resourcePath }: { archive: Archive; resourcePath: string }) =>\n async (resource: HookResource): Promise<HookResource> => {\n const file = Object.values(archive.records).find(\n (file) => file.uri === resourcePath && !file.dir,\n )\n\n if (!file || file.dir) return resource\n\n const metadata = await getMetadata(archive, resourcePath)\n\n return {\n ...resource,\n params: {\n ...resource.params,\n ...(file?.encodingFormat && {\n contentType: file.encodingFormat,\n }),\n ...(metadata.mediaType && {\n contentType: metadata.mediaType,\n }),\n },\n }\n }\n","import type { Archive } from \"../../../archives/types\"\nimport type { HookResource } from \"./types\"\n\nconst invalidSelfClosingTags = [\n \"div\",\n \"span\",\n \"p\",\n \"a\",\n \"li\",\n \"ul\",\n \"ol\",\n \"h1\",\n \"h2\",\n \"h3\",\n \"h4\",\n \"h5\",\n \"h6\",\n \"table\",\n \"tr\",\n \"td\",\n \"th\",\n \"thead\",\n \"tbody\",\n \"tfoot\",\n \"section\",\n \"article\",\n \"header\",\n \"footer\",\n \"nav\",\n \"aside\",\n \"main\",\n \"figure\",\n \"figcaption\",\n \"blockquote\",\n \"pre\",\n \"code\",\n \"form\",\n \"textarea\",\n \"select\",\n \"option\",\n \"button\",\n \"label\",\n \"fieldset\",\n \"legend\",\n \"caption\",\n \"dl\",\n \"dt\",\n \"dd\",\n \"iframe\",\n \"video\",\n \"audio\",\n \"canvas\",\n \"script\",\n \"style\",\n]\n\n/**\n * Some books uses xhtml files but also includes wrong self closin tags. This happens\n * a lot with kobo epub which have a lot of self closing <script ... />. This breaks on some\n * browser such as webkit. We use a regex to replace and fix them. There\n * is a first lighter regex which check if any such tag exist in the first place before running\n * the full replace regex empty.\n */\nexport const selfClosingTagsFixHook =\n ({ archive, resourcePath }: { archive: Archive; resourcePath: string }) =>\n async (resource: HookResource): Promise<HookResource> => {\n const file = Object.values(archive.records).find(\n (file) => file.uri === resourcePath && !file.dir,\n )\n\n if (file && !file.dir && file.basename.endsWith(`.xhtml`)) {\n const bodyToParse =\n typeof resource.body === `string` ? resource.body : await file.string()\n\n const tagCheckPattern = new RegExp(\n `<(${invalidSelfClosingTags.join(\"|\")})[\\\\s/>]`,\n \"i\",\n )\n if (!tagCheckPattern.test(bodyToParse)) {\n return resource\n }\n\n const tagPattern = new RegExp(\n `<(${invalidSelfClosingTags.join(\"|\")})(\\\\s[^>]*)?\\\\s*/>`,\n \"gi\",\n )\n\n const fixedBody = bodyToParse.replace(\n tagPattern,\n (_, tagName, attributes = \"\") => {\n // Convert to an opening and closing tag\n return `<${tagName} ${attributes.trim()}></${tagName}>`\n },\n )\n\n return {\n ...resource,\n body: fixedBody,\n }\n }\n\n return resource\n }\n","import type { Archive } from \"../..\"\nimport type { StreamerResourceHookFactory } from \"../../hooks\"\nimport { Report } from \"../../report\"\nimport { calibreFixHook } from \"./hooks/calibreFixHook\"\nimport { cssFixHook } from \"./hooks/cssFixHook\"\nimport { defaultHook } from \"./hooks/defaultHook\"\nimport { selfClosingTagsFixHook } from \"./hooks/selfClosingTagsFixHook\"\nimport type { HookResource } from \"./hooks/types\"\n\n/**\n * Builds one streamed asset from `archive` (hot path: typically once per\n * fetched file in the reader).\n *\n * **OPF limitation:** Unlike manifest generation, this API does not take\n * `ArchiveOpfParsed`. `defaultHook` calls `readArchiveOpf(archive)` to resolve\n * `media-type` from the package document, so EPUBs repeat OPF I/O + parse per\n * resource. Usually fine (small OPF); if profiling shows regressions, address\n * explicitly—e.g. optional `archiveOpf` (or equivalent) threaded from\n * `Streamer.fetchResource` / manifest, or an opt-in cache on `Archive` wired at\n * construction time—rather than hidden process-wide memoization.\n */\nexport const generateResourceFromArchive = async (\n archive: Archive,\n resourcePath: string,\n { hooks = [] }: { hooks?: StreamerResourceHookFactory[] } = {},\n) => {\n const defaultResource: HookResource = {\n params: {},\n }\n const externalHooks = hooks.map((hook) => hook({ archive, resourcePath }))\n\n /**\n * EPUB: `defaultHook` calls `readArchiveOpf` → repeated OPF I/O + parse per\n * resource until we thread `ArchiveOpfParsed` here (see export JSDoc).\n */\n const resourceHooks = [\n ...externalHooks,\n defaultHook({ archive, resourcePath }),\n selfClosingTagsFixHook({ archive, resourcePath }),\n cssFixHook({ archive, resourcePath }),\n calibreFixHook({ archive, resourcePath }),\n ]\n\n try {\n const resource = await resourceHooks.reduce(async (manifest, gen) => {\n return await gen(await manifest)\n }, Promise.resolve(defaultResource))\n\n Report.log(\"Generated resource\", resourcePath, resource)\n\n if (resource.body !== undefined) {\n return resource\n }\n\n const file = Object.values(archive.records).find(\n (file) => file.uri === resourcePath && !file.dir,\n )\n\n if (!file || file.dir) {\n throw new Error(`no file found for resourcePath:${resourcePath}`)\n }\n\n return {\n ...resource,\n body: await file.blob(),\n }\n } catch (e) {\n Report.error(e)\n\n throw e\n }\n}\n","import {\n BehaviorSubject,\n catchError,\n distinctUntilChanged,\n EMPTY,\n filter,\n first,\n from,\n ignoreElements,\n map,\n merge,\n mergeMap,\n NEVER,\n type ObservedValueOf,\n pairwise,\n Subject,\n shareReplay,\n startWith,\n switchMap,\n takeUntil,\n tap,\n timer,\n} from \"rxjs\"\nimport { Report } from \"../report\"\nimport type { Archive } from \"./types\"\n\nclass ArchiveEntry {\n state$ = new BehaviorSubject<{\n status: \"idle\" | \"loading\" | \"success\" | \"error\"\n error?: unknown | undefined\n archive?: undefined | Archive\n locks: number\n }>({\n status: `idle`,\n locks: 0,\n })\n\n constructor(private cleanArchiveAfter: number) {}\n\n update(update: Partial<ObservedValueOf<typeof this.state$>>) {\n this.state$.next({ ...this.state$.getValue(), ...update })\n }\n\n get locks$() {\n return this.state$.pipe(map(({ locks }) => locks))\n }\n\n get state() {\n return this.state$.getValue()\n }\n\n get isUnlocked$() {\n return this.locks$.pipe(\n map((locks) => locks <= 0),\n distinctUntilChanged(),\n shareReplay(),\n )\n }\n\n get overTTL$() {\n return this.isUnlocked$.pipe(\n switchMap((isUnlocked) =>\n !isUnlocked\n ? NEVER\n : this.cleanArchiveAfter === Infinity\n ? NEVER\n : timer(this.cleanArchiveAfter),\n ),\n )\n }\n}\n\nexport const createArchiveLoader = ({\n getArchive,\n cleanArchiveAfter = 5 * 60 * 1000, // 5mn\n}: {\n getArchive: (key: string) => Promise<Archive>\n cleanArchiveAfter?: number\n}) => {\n const loadSubject = new Subject<string>()\n const destroySubject = new Subject<void>()\n const purgeSubject = new Subject<void>()\n const archives: Record<string, ArchiveEntry> = {}\n\n const loadArchive$ = loadSubject.pipe(\n mergeMap((key) => {\n const archiveEntry = archives[key]\n\n if (!archiveEntry || archiveEntry.state.status !== \"idle\") return EMPTY\n\n let isClosed = false\n\n const cleanupArchive = (key: string) => {\n Report.debug(`Cleaning up archive with key: ${key}`)\n\n const entry = archives[key]\n\n delete archives[key]\n\n if (!isClosed) {\n entry?.state.archive?.close()\n isClosed = true\n }\n }\n\n archiveEntry.update({\n status: \"loading\",\n })\n\n const locks$ = archiveEntry.locks$\n const isUnlocked$ = archiveEntry.isUnlocked$\n\n const newAccess$ = locks$.pipe(\n pairwise(),\n filter(([prev, curent]) => curent > prev),\n startWith(true),\n )\n\n const archive$ = from(getArchive(key))\n\n return archive$.pipe(\n tap((archive) => {\n archiveEntry.update({\n archive,\n status: \"success\",\n })\n }),\n catchError((error) => {\n cleanupArchive(key)\n\n archiveEntry.update({\n status: \"error\",\n error,\n })\n\n return EMPTY\n }),\n switchMap(() => {\n const readyForPurge$ = newAccess$.pipe(\n switchMap(() => purgeSubject),\n switchMap(() => isUnlocked$),\n filter((isUnlocked) => isUnlocked),\n )\n\n const toCleanup$ = merge(readyForPurge$, archiveEntry.overTTL$)\n\n return toCleanup$.pipe(\n first(),\n tap(() => {\n cleanupArchive(key)\n }),\n )\n }),\n )\n }),\n takeUntil(destroySubject),\n )\n\n const access = (key: string) => {\n let releaseCalled = false\n\n const archiveEntry = archives[key] ?? new ArchiveEntry(cleanArchiveAfter)\n\n archives[key] = archiveEntry\n\n archiveEntry.update({\n locks: archiveEntry.state.locks + 1,\n })\n\n const release = () => {\n if (releaseCalled) return\n\n releaseCalled = true\n\n archiveEntry.update({\n locks: archiveEntry.state.locks - 1,\n })\n }\n\n loadSubject.next(key)\n\n const archive$ = archiveEntry.state$.pipe(\n map(({ archive }) => archive),\n filter((archive) => !!archive),\n )\n\n const error$ = archiveEntry.state$.pipe(\n tap(({ error }) => {\n if (error) {\n throw error\n }\n }),\n ignoreElements(),\n )\n\n return merge(archive$, error$).pipe(\n first(),\n map((archive) => ({ archive, release })),\n catchError((error) => {\n release()\n\n throw error\n }),\n )\n }\n\n /**\n * Will purge immediatly archives as soon as they are released\n */\n const purge = () => {\n purgeSubject.next()\n }\n\n loadArchive$.subscribe()\n\n return {\n access,\n purge,\n _archives: archives,\n }\n}\n","const parseByteRangePart = (value: string) => {\n if (!value) {\n return {\n valid: true,\n value: undefined,\n }\n }\n\n if (!/^\\d+$/.test(value)) {\n return {\n valid: false,\n value: undefined,\n }\n }\n\n return {\n valid: true,\n value: Number.parseInt(value, 10),\n }\n}\n\nconst parseSingleByteRange = (rangeHeader: string) => {\n if (!rangeHeader.toLowerCase().startsWith(\"bytes=\")) {\n return {\n kind: \"missing\" as const,\n }\n }\n\n const rangeValue = rangeHeader.slice(\"bytes=\".length).trim()\n\n if (!rangeValue) {\n return {\n kind: \"invalid\" as const,\n }\n }\n\n if (rangeValue.includes(\",\")) {\n return {\n kind: \"multi\" as const,\n }\n }\n\n const matches = /^(\\d*)-(\\d*)$/.exec(rangeValue)\n\n if (!matches) {\n return {\n kind: \"invalid\" as const,\n }\n }\n\n const [, rawStart = \"\", rawEnd = \"\"] = matches\n const parsedStart = parseByteRangePart(rawStart.trim())\n const parsedEnd = parseByteRangePart(rawEnd.trim())\n\n if (!parsedStart.valid || !parsedEnd.valid) {\n return {\n kind: \"invalid\" as const,\n }\n }\n\n return {\n kind: \"single\" as const,\n start: parsedStart.value,\n end: parsedEnd.value,\n }\n}\n\n/**\n * Normalize the body into a sliceable form that avoids passing a\n * `Blob.slice()` result directly to `new Response()`. Some Node.js\n * versions do not recognise sliced Blobs as valid `BodyInit`,\n * producing `\"[object Blob]\"` instead of the actual content.\n */\nconst materializeForSlicing = (body: Blob | string) => {\n if (body instanceof Blob) {\n return {\n size: body.size,\n slice: (start: number, endExclusive: number) => {\n const part = body.slice(start, endExclusive)\n return { content: part, length: part.size }\n },\n }\n }\n\n const bytes = new TextEncoder().encode(body)\n\n return {\n size: bytes.byteLength,\n slice: (start: number, endExclusive: number) => {\n const part = bytes.slice(start, endExclusive)\n return { content: part, length: part.byteLength }\n },\n }\n}\n\nexport const createRangeResponse = ({\n body,\n contentType,\n rangeHeader,\n}: {\n body: Blob | string\n contentType?: string\n rangeHeader?: string | null\n}) => {\n const headers = new Headers()\n\n if (contentType) {\n headers.set(\"Content-Type\", contentType)\n }\n\n headers.set(\"Accept-Ranges\", \"bytes\")\n\n if (!rangeHeader) {\n if (body instanceof Blob) {\n headers.set(\"Content-Length\", String(body.size))\n }\n\n return new Response(body, {\n status: 200,\n headers,\n })\n }\n\n const parsedRange = parseSingleByteRange(rangeHeader)\n\n if (parsedRange.kind === \"missing\" || parsedRange.kind === \"multi\") {\n if (body instanceof Blob) {\n headers.set(\"Content-Length\", String(body.size))\n }\n\n return new Response(body, {\n status: 200,\n headers,\n })\n }\n\n const rangeBody = materializeForSlicing(body)\n const size = rangeBody.size\n\n if (parsedRange.kind === \"invalid\") {\n return new Response(null, {\n status: 416,\n headers: {\n \"Content-Range\": `bytes */${size}`,\n },\n })\n }\n\n let start = parsedRange.start\n let end = parsedRange.end\n\n if (start === undefined && end === undefined) {\n return new Response(null, {\n status: 416,\n headers: {\n \"Content-Range\": `bytes */${size}`,\n },\n })\n }\n\n if (start === undefined) {\n const suffixLength = Math.min(end ?? 0, size)\n start = Math.max(0, size - suffixLength)\n end = size - 1\n } else if (end === undefined || end >= size) {\n end = size - 1\n }\n\n if (start < 0 || end < 0 || start >= size || end >= size || start > end) {\n return new Response(null, {\n status: 416,\n headers: {\n \"Content-Range\": `bytes */${size}`,\n },\n })\n }\n\n const partial = rangeBody.slice(start, end + 1)\n\n headers.set(\"Content-Length\", String(partial.length))\n headers.set(\"Content-Range\", `bytes ${start}-${end}/${size}`)\n\n return new Response(partial.content, {\n status: 206,\n headers,\n })\n}\n","import type { Manifest } from \"@prose-reader/shared\"\nimport {\n catchError,\n finalize,\n from,\n lastValueFrom,\n map,\n mergeMap,\n type Observable,\n of,\n switchMap,\n} from \"rxjs\"\nimport { createArchiveLoader } from \"./archives/archiveLoader\"\nimport type { Archive } from \"./archives/types\"\nimport { generateManifestFromArchive } from \"./generators/manifest\"\nimport { generateResourceFromArchive } from \"./generators/resources\"\nimport type { StreamerHooks } from \"./hooks\"\nimport { createRangeResponse } from \"./utils/createRangeResponse\"\n\ntype OnError = (error: unknown) => Response\ntype OnManifestSuccess = (params: {\n manifest: Manifest\n archive: Archive\n}) => Observable<Manifest> | Promise<Manifest>\ntype WithArchiveResponse = (\n archive: Archive,\n) => Observable<Response> | Promise<Response>\n\nconst fileProtocol = `file://`\nconst httpProtocolPattern = /^https?:\\/\\//\n\nconst decodeResourcePath = (resourcePath: string) => {\n try {\n return decodeURIComponent(resourcePath)\n } catch {\n return resourcePath\n }\n}\n\nconst stripFileProtocol = (resourcePath: string) =>\n resourcePath.startsWith(fileProtocol)\n ? resourcePath.slice(fileProtocol.length)\n : resourcePath\n\nconst normalizeResourcePath = (resourcePath: string) => {\n const resourcePathWithoutFileProtocol = stripFileProtocol(resourcePath)\n\n if (httpProtocolPattern.test(resourcePathWithoutFileProtocol)) {\n return resourcePathWithoutFileProtocol\n }\n\n return stripFileProtocol(decodeResourcePath(resourcePathWithoutFileProtocol))\n}\n\nexport class Streamer {\n protected archiveLoader: ReturnType<typeof createArchiveLoader>\n protected onError: OnError = (error) => {\n console.error(error)\n\n return new Response(String(error), { status: 500 })\n }\n protected hooks: StreamerHooks\n protected onManifestSuccess: OnManifestSuccess\n protected lastAccessedKey: string | undefined\n\n constructor({\n hooks,\n onError,\n onManifestSuccess,\n ...rest\n }: Parameters<typeof createArchiveLoader>[0] & {\n hooks?: StreamerHooks\n onError?: OnError\n onManifestSuccess?: OnManifestSuccess\n }) {\n this.archiveLoader = createArchiveLoader(rest)\n this.hooks = hooks ?? {}\n\n this.onManifestSuccess =\n onManifestSuccess ?? (({ manifest }) => Promise.resolve(manifest))\n this.onError = onError ?? this.onError\n }\n\n public prune() {\n this.archiveLoader.purge()\n }\n\n public accessArchive(key: string) {\n if (this.lastAccessedKey !== undefined && this.lastAccessedKey !== key) {\n this.archiveLoader.purge()\n }\n\n this.lastAccessedKey = key\n\n return this.archiveLoader.access(key)\n }\n\n public accessArchiveWithoutLock(key: string) {\n return this.accessArchive(key).pipe(\n map(({ archive, release }) => {\n release()\n\n return archive\n }),\n )\n }\n\n protected withArchiveResponse({\n key,\n getResponse,\n }: {\n key: string\n getResponse: WithArchiveResponse\n }) {\n const response$ = this.accessArchive(key).pipe(\n mergeMap(({ archive, release }) =>\n from(getResponse(archive)).pipe(\n finalize(() => {\n release()\n }),\n ),\n ),\n catchError((error) => {\n return of(this.onError(error))\n }),\n )\n\n return lastValueFrom(response$)\n }\n\n public fetchManifest({ key, baseUrl }: { key: string; baseUrl?: string }) {\n return this.withArchiveResponse({\n key,\n getResponse: (archive) => {\n const manifest$ = from(\n generateManifestFromArchive(archive, {\n baseUrl,\n hooks: this.hooks.manifest,\n }),\n )\n\n return manifest$.pipe(\n switchMap((manifest) =>\n from(this.onManifestSuccess({ manifest, archive })),\n ),\n map(\n (manifest) =>\n new Response(JSON.stringify(manifest satisfies Manifest), {\n status: 200,\n }),\n ),\n )\n },\n })\n }\n\n public fetchResource({\n key,\n resourcePath,\n request,\n }: {\n key: string\n resourcePath: string\n request?: Request\n }) {\n return this.withArchiveResponse({\n key,\n getResponse: (archive) => {\n /**\n * We commonly use file:// for manifest without baseUrl. This ensure we\n * have valid URL for the reader while flagging them as local.\n *\n * However, we obviously don't have the file:// prefix on the archive.\n */\n const cleanedResourcePath = normalizeResourcePath(resourcePath)\n\n /**\n * Reader hot path: one `generateResourceFromArchive` per request. For\n * EPUBs, `defaultHook` re-reads and re-parses the package OPF each time\n * (no `archiveOpf` threading yet). See `generateResourceFromArchive` JSDoc.\n */\n const resource$ = from(\n generateResourceFromArchive(archive, cleanedResourcePath, {\n hooks: this.hooks.resource,\n }),\n )\n\n return resource$.pipe(\n map((resource) =>\n createRangeResponse({\n body: resource.body ?? \"\",\n contentType: resource.params.contentType,\n rangeHeader: request?.headers.get(\"range\"),\n }),\n ),\n )\n },\n })\n }\n}\n","import { Streamer } from \"./Streamer\"\nimport { removeTrailingSlash } from \"./utils/uri\"\n\ntype ConflictFreeWebWorkerFetchEvent = {\n readonly request: Request\n /** [MDN Reference](https://developer.mozilla.org/docs/Web/API/FetchEvent/respondWith) */\n respondWith(r: Response | PromiseLike<Response>): void\n}\n\nexport class ServiceWorkerStreamer extends Streamer {\n protected getUriInfo: (event: ConflictFreeWebWorkerFetchEvent) =>\n | {\n baseUrl: string\n }\n | undefined\n\n constructor({\n getUriInfo,\n ...rest\n }: ConstructorParameters<typeof Streamer>[0] & {\n getUriInfo: (event: ConflictFreeWebWorkerFetchEvent) =>\n | {\n baseUrl: string\n }\n | undefined\n }) {\n super(rest)\n\n this.getUriInfo = getUriInfo\n this.fetchEventListener = this.fetchEventListener.bind(this)\n }\n\n fetchEventListener(event: ConflictFreeWebWorkerFetchEvent) {\n try {\n const uriInfo = this.getUriInfo(event)\n\n if (!uriInfo) return\n\n const baseUrl = removeTrailingSlash(uriInfo.baseUrl)\n const streamerPath = event.request.url.substring(\n baseUrl.length + `/`.length,\n )\n const [key = ``] = streamerPath.split(\"/\")\n const resourcePath = removeTrailingSlash(\n streamerPath.substring(key.length + `/`.length),\n )\n\n if (streamerPath.endsWith(`/manifest`)) {\n event.respondWith(\n this.fetchManifest({ key, baseUrl: `${baseUrl}/${key}/` }),\n )\n } else {\n event.respondWith(\n this.fetchResource({\n key,\n resourcePath,\n request: event.request,\n }),\n )\n }\n } catch (e) {\n event.respondWith(new Response(String(e), { status: 500 }))\n }\n }\n}\n"],"names":["sortByTitleComparator","a","b","alist","blist","i","len","getUriBasename","uri","removeTrailingSlash","getUriBasePath","lastSlashIndex","createArchiveFromArrayBufferList","list","orderByAlpha","name","files","file","size","basename","detectMimeTypeFromName","data","Report","SharedReport","printTree","paths","tree","path","parts","node","part","render","indent","key","arr","isLast","prefix","nextIndent","value","createArchiveFromJszip","jszip","archive","folderStructureStr","createArchiveFromLibArchive","libArchive","objArray","item","createArchiveFromText","content","mimeType","direction","txtOpfContent","createArchiveFromUrls","urls","options","opfFileData","url","filesFromUrl","configure","enableReport","getArchiveOpfInfo","readArchiveOpf","opsFile","basePath","raw","parseOpf","appleDisplayOptionsBasename","APPLE_IBOOKS_DISPLAY_OPTIONS_FILENAME","apple","manifest","infoFile","parsed","parseAppleDisplayOptionsXml","renditionLayout","resolveArchiveMetadata","e","comicInfoFilenameLower","COMIC_INFO_FILENAME","comicInfo","comicInfoFile","manifestWithoutComicInfo","_","items","parseComicInfo","resolved","createManifestResourceHref","baseUrl","resourcePath","hrefBaseUrl","defaultHook","dir","index","getSpineItemFilesFromArchive","archiveOpf","opf","opfBasePath","spineRows","getItemsFromDoc","manifestItems","getBaseUrlForHref","el","href","manifestRenditionFlow","v","manifestRenditionSpread","manifestGuideType","epubHook","publisherRenditionLayout","packageRenditionLayout","title","readingDirection","totalSize","guideRefs","guide","elm","type","row","itemSize","hrefBaseUri","hasDocMetaViewport","doc","metaElm","allFilesHaveViewportMeta","result","current","isXmlBasedMimeType","XmlDocument","epubOptimizerHook","extractKoboInformationFromArchive","KOBO_DISPLAY_OPTIONS_FILENAME","layout","parseKoboXml","kobo","koboMeta","hasOpfExtension","isArchiveEpub","nonEpub","spineItem","archiveItem","parseContentType","isMediaContentMimeType","getXmlElementInnerText","child","XmlTextNode","XmlElement","manifestItemIsNavDocument","tokenizeXmlSpaceSeparatedList","extractNavChapter","li","chp","contentNode","urlJoin","sublistNode","children","buildTOCFromNav","toc","navDataChildren","parseTocFromNavPath","navItem","tocFile","tocFileBasePath","mapNcxChapter","point","src","out","pt","buildTOCFromNCX","ncxData","rootTagName","parseTocFromNcx","ncxId","ncxItem","ncxPath","parseToc","tocFromNav","tocFromNcx","normalizeFilenameAsTitle","filename","buildAudiobookToc","record","r","buildFolderFallbackToc","filesSortedByAlpha","combineWith","folder","subFolders","foundEntry","entry","nextFolderCursor","nextSubFolders","acc","folders","firstFolder","restFolders","resolveTocFromArchive","audiobookToc","tocHook","normalizeBaseUrl","generateManifestFromArchive","hooks","normalizedBaseUrl","createExternalHooks","hookFactories","hook","contentHooks","spineHooks","presentationHooks","navigationHooks","manifestHooks","baseManifestPromise","gen","manifestStr","hasCalibreCoverMeta","getBuggyCoverSvg","fixBuggyCover","resource","bodyToParse","opfXmlDoc","buggySvg","calibreFixHook","cssFixHook","newBody","getMetadata","getContentTypeFromExtension","metadata","invalidSelfClosingTags","selfClosingTagsFixHook","tagPattern","fixedBody","tagName","attributes","generateResourceFromArchive","defaultResource","resourceHooks","ArchiveEntry","cleanArchiveAfter","BehaviorSubject","update","map","locks","distinctUntilChanged","shareReplay","switchMap","isUnlocked","NEVER","timer","createArchiveLoader","getArchive","loadSubject","Subject","destroySubject","purgeSubject","archives","loadArchive$","mergeMap","archiveEntry","EMPTY","isClosed","cleanupArchive","locks$","isUnlocked$","newAccess$","pairwise","filter","prev","curent","startWith","from","tap","catchError","error","readyForPurge$","merge","first","takeUntil","access","releaseCalled","release","archive$","error$","ignoreElements","purge","parseByteRangePart","parseSingleByteRange","rangeHeader","rangeValue","matches","rawStart","rawEnd","parsedStart","parsedEnd","materializeForSlicing","body","start","endExclusive","bytes","createRangeResponse","contentType","headers","parsedRange","rangeBody","end","suffixLength","partial","fileProtocol","httpProtocolPattern","decodeResourcePath","stripFileProtocol","normalizeResourcePath","resourcePathWithoutFileProtocol","Streamer","onError","onManifestSuccess","rest","getResponse","response$","finalize","of","lastValueFrom","request","cleanedResourcePath","ServiceWorkerStreamer","getUriInfo","event","uriInfo","streamerPath"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"index.js","sources":["../src/utils/sortByTitleComparator.ts","../src/utils/uri.ts","../src/archives/createArchiveFromArrayBufferList.ts","../src/report.ts","../src/archives/printTree.ts","../src/archives/createArchiveFromJszip.ts","../src/archives/createArchiveFromLibArchive.ts","../src/archives/createArchiveFromText.ts","../src/utils/createXmlSafeId.ts","../src/archives/createArchiveFromUrls.ts","../src/configure.ts","../src/epubs/getArchiveOpfInfo.ts","../src/epubs/readArchiveOpf.ts","../src/generators/manifest/hooks/apple.ts","../src/generators/manifest/hooks/comicInfo.ts","../src/generators/manifest/createManifestResourceHref.ts","../src/generators/manifest/hooks/default.ts","../src/epubs/getSpineItemFilesFromArchive.ts","../src/generators/manifest/hooks/epub/epub.ts","../src/generators/manifest/hooks/epubOptimizer.ts","../src/generators/manifest/hooks/kobo.ts","../src/epubs/isArchiveEpub.ts","../src/generators/manifest/hooks/nonEpub.ts","../src/parsers/xml.ts","../src/parsers/nav.ts","../src/generators/manifest/hooks/audiobookToc.ts","../src/generators/manifest/hooks/toc.ts","../src/generators/manifest/index.ts","../src/generators/resources/hooks/calibreFixHook.ts","../src/generators/resources/hooks/cssFixHook.ts","../src/generators/resources/hooks/defaultHook.ts","../src/generators/resources/hooks/selfClosingTagsFixHook.ts","../src/generators/resources/index.ts","../src/archives/archiveLoader.ts","../src/utils/createRangeResponse.ts","../src/Streamer.ts","../src/ServiceWorkerStreamer.ts"],"sourcesContent":["export const sortByTitleComparator = (a: string, b: string) => {\n const alist = a.split(/(\\d+)/)\n const blist = b.split(/(\\d+)/)\n\n for (let i = 0, len = alist.length; i < len; i++) {\n if (alist[i] !== blist[i]) {\n if (alist[i]?.match(/\\d/)) {\n return +(alist[i] || ``) - +(blist[i] || ``)\n }\n return (alist[i] || ``).localeCompare(blist[i] || ``)\n }\n }\n\n return 1\n}\n","export const getUriBasename = (uri: string) =>\n uri.substring(uri.lastIndexOf(`/`) + 1) || uri\n\nexport const removeTrailingSlash = (uri: string) =>\n uri.endsWith(\"/\") ? uri.slice(0, -1) : uri\n\nexport const getUriBasePath = (uri: string) => {\n const lastSlashIndex = uri.lastIndexOf(\"/\")\n\n return lastSlashIndex >= 0 ? uri.substring(0, lastSlashIndex) : \"\"\n}\n","import { detectMimeTypeFromName } from \"@prose-reader/shared\"\nimport { sortByTitleComparator } from \"../utils/sortByTitleComparator\"\nimport { getUriBasename } from \"../utils/uri\"\nimport type { Archive } from \"./types\"\n\nexport const createArchiveFromArrayBufferList = async (\n list: {\n isDir: boolean\n name: string\n size: number\n data: () => Promise<ArrayBuffer>\n }[],\n { orderByAlpha, name }: { orderByAlpha?: boolean; name?: string } = {},\n): Promise<Archive> => {\n let files = list\n\n if (orderByAlpha) {\n files = files.slice().sort((a, b) => sortByTitleComparator(a.name, b.name))\n }\n\n return {\n filename: name || ``,\n records: files.map((file) => {\n const size = file.size\n const basename = getUriBasename(file.name)\n\n if (file.isDir) {\n return {\n dir: true,\n basename,\n uri: file.name,\n size,\n }\n }\n\n return {\n dir: file.isDir,\n basename,\n encodingFormat: detectMimeTypeFromName(file.name),\n uri: file.name,\n blob: async () =>\n new Blob([await file.data()], {\n type: detectMimeTypeFromName(file.name) ?? ``,\n }),\n string: async () => {\n const data = await file.data()\n return String.fromCharCode.apply(\n null,\n Array.from(new Uint16Array(data)),\n )\n },\n size,\n }\n }),\n close: () => Promise.resolve(),\n }\n}\n","import { Report as SharedReport } from \"@prose-reader/shared\"\nimport { name } from \"../package.json\"\n\nexport const Report = SharedReport.namespace(name, false, {\n color: \"#ffae42\",\n})\n","interface TreeNode {\n [key: string]: TreeNode\n}\n\nexport const printTree = (paths: string[]): string => {\n // Split and collect all parts for tree reconstruction\n const tree: TreeNode = {}\n for (const path of paths) {\n const parts = path.split(\"/\")\n let node = tree\n for (let i = 0; i < parts.length; i++) {\n const part = parts[i]\n if (part === undefined) continue\n if (!node[part]) {\n node[part] = {}\n }\n node = node[part]\n }\n }\n\n // Recursively build the tree string\n const render = (node: TreeNode, indent = \"\"): string => {\n return Object.keys(node)\n .sort()\n .map((key, i, arr) => {\n const isLast = i === arr.length - 1\n const prefix = indent + (isLast ? \"└── \" : \"├── \")\n const nextIndent = indent + (isLast ? \" \" : \"│ \")\n const value = node[key]\n if (value && Object.keys(value).length > 0) {\n return `${prefix}${key}/\\n${render(value, nextIndent)}`\n }\n return `${prefix}${key}`\n })\n .join(\"\\n\")\n }\n\n return render(tree)\n}\n","import { detectMimeTypeFromName } from \"@prose-reader/shared\"\nimport { Report } from \"../report\"\nimport { sortByTitleComparator } from \"../utils/sortByTitleComparator\"\nimport { getUriBasename } from \"../utils/uri\"\nimport { printTree } from \"./printTree\"\nimport type { Archive, StreamResult } from \"./types\"\n\ninterface OutputByType {\n base64: string\n string: string\n text: string\n binarystring: string\n array: number[]\n uint8array: Uint8Array\n arraybuffer: ArrayBuffer\n blob: Blob\n nodebuffer: Buffer\n}\n\ntype OutputType = keyof OutputByType\n\ninterface JSZipObject {\n name: string\n dir: boolean\n date: Date\n comment: string\n unixPermissions: number | string | null\n dosPermissions: number | null\n async<T extends OutputType>(type: T): Promise<OutputByType[T]>\n // nodeStream(type?: `nodebuffer`): NodeJS.ReadableStream;\n internalStream?: (type?: `uint8array`) => StreamResult\n}\n\ninterface JSZip {\n files: { [key: string]: JSZipObject }\n}\n\nexport const createArchiveFromJszip = async (\n jszip: JSZip,\n { orderByAlpha, name }: { orderByAlpha?: boolean; name?: string } = {},\n): Promise<Archive> => {\n let files = Object.values(jszip.files)\n\n if (orderByAlpha) {\n files = files.slice().sort((a, b) => sortByTitleComparator(a.name, b.name))\n }\n\n const archive: Archive = {\n filename: name || ``,\n records: files.map((file) => {\n // this is private API\n // @ts-expect-error\n const size = file._data.uncompressedSize\n const basename = getUriBasename(file.name)\n\n if (file.dir) {\n return {\n dir: true,\n basename,\n uri: file.name,\n size,\n }\n }\n\n return {\n dir: false,\n basename: getUriBasename(file.name),\n uri: file.name,\n encodingFormat: detectMimeTypeFromName(file.name),\n blob: () => file.async(`blob`),\n string: () => file.async(\"string\"),\n ...(file.internalStream && {\n stream: file.internalStream,\n }),\n // this is private API\n // @ts-expect-error\n size: file._data.uncompressedSize,\n }\n }),\n close: () => Promise.resolve(),\n }\n\n Report.log(\"Generated archive\", archive)\n\n if (process.env.NODE_ENV === \"development\") {\n if (Report.isEnabled()) {\n const folderStructureStr = printTree(files.map((file) => file.name))\n Report.groupCollapsed(...Report.getGroupArgs(\"Archive folder structure\"))\n Report.log(`\\n${folderStructureStr}`)\n Report.groupEnd()\n }\n }\n\n return archive\n}\n","/**\n * @see https://github.com/nika-begiashvili/libarchivejs.\n *\n * Does not work in service worker due to usage of web worker.\n */\nimport { detectMimeTypeFromName } from \"@prose-reader/shared\"\nimport { Report } from \"../report\"\nimport type { Archive } from \"./types\"\n\ninterface ArchiveReader {\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n getFilesArray(): Promise<any[]>\n /**\n * Terminate worker to free up memory\n */\n close(): Promise<void>\n}\n\n/**\n * Represents compressed file before extraction\n */\ninterface CompressedFile {\n /**\n * File name\n */\n get name(): string\n /**\n * File size\n */\n get size(): number\n get lastModified(): number\n /**\n * Extract file from archive\n * @returns {Promise<File>} extracted file\n */\n\n // biome-ignore lint/suspicious/noExplicitAny: TODO\n extract(): any\n}\n\nexport const createArchiveFromLibArchive = async (\n libArchive: ArchiveReader,\n { name }: { orderByAlpha?: boolean; name?: string } = {},\n): Promise<Archive> => {\n const objArray = await libArchive.getFilesArray()\n\n const archive: Archive = {\n close: () => libArchive.close(),\n filename: name ?? ``,\n records: objArray.map((item: { file: CompressedFile; path: string }) => ({\n dir: false,\n basename: item.file.name,\n encodingFormat: detectMimeTypeFromName(item.file.name),\n size: item.file.size,\n uri: `${item.path}${item.file.name}`,\n blob: async () => {\n const file = await (item.file.extract() as Promise<File>)\n\n return file\n },\n string: async () => {\n const file = await (item.file.extract() as Promise<File>)\n\n return file.text()\n },\n })),\n }\n\n Report.log(\"Generated archive\", archive)\n\n return archive\n}\n","import { getUriBasename } from \"../utils/uri\"\nimport type { Archive } from \"./types\"\n\n/**\n * Useful to create archive from txt content\n */\nexport const createArchiveFromText = async (\n content: string | Blob,\n {\n mimeType,\n direction,\n }: {\n direction?: `ltr` | `rtl`\n mimeType?: string\n } = { mimeType: \"text/plain\" },\n) => {\n const txtOpfContent = `\n <?xml version=\"1.0\" encoding=\"UTF-8\"?>\n <package xmlns=\"http://www.idpf.org/2007/opf\" version=\"3.0\" xml:lang=\"ja\" prefix=\"rendition: http://www.idpf.org/vocab/rendition/#\"\n unique-identifier=\"ootuya-id\">\n <metadata xmlns:opf=\"http://www.idpf.org/2007/opf\" xmlns:dc=\"http://purl.org/dc/elements/1.1/\"\n xmlns:dcterms=\"http://purl.org/dc/terms/\">\n <meta property=\"rendition:layout\">reflowable</meta>\n </metadata>\n <manifest>\n <item id=\"p01\" href=\"p01.txt\" media-type=\"text/plain\"/>\n </manifest>\n <spine page-progression-direction=\"${direction ?? `ltr`}\">\n <itemref idref=\"p01\" />\n </spine>\n </package>\n `\n\n const archive: Archive = {\n filename: `content.txt`,\n records: [\n {\n dir: false,\n basename: getUriBasename(`generated.opf`),\n uri: `generated.opf`,\n blob: async () => new Blob([txtOpfContent]),\n string: async () => txtOpfContent,\n size: 0,\n },\n {\n dir: false,\n basename: getUriBasename(`p01.txt`),\n uri: `p01.txt`,\n blob: async () => {\n if (typeof content === \"string\") return new Blob([content])\n return content\n },\n string: async () => {\n if (typeof content === \"string\") return content\n return content.text()\n },\n size: typeof content === \"string\" ? content.length : content.size,\n encodingFormat: mimeType,\n },\n ],\n close: () => Promise.resolve(),\n }\n\n return archive\n}\n","// Generated non-EPUB IDs preserve a leading digit to stay close to common\n// reader-generated identifiers, even though strict XML IDs disallow it.\nconst XML_SAFE_ID_PATTERN = /^[A-Za-z0-9_][A-Za-z0-9_.-]*$/\nconst XML_SAFE_ID_START_PATTERN = /^[A-Za-z0-9_]/\nconst XML_RESERVED_ID_START_PATTERN = /^xml/i\nconst INVALID_XML_SAFE_ID_CHARACTER_PATTERN = /[^A-Za-z0-9_.-]+/g\nconst XML_SAFE_ID_START_FALLBACK = `_`\n\nconst trimIdSeparators = (value: string) => value.replace(/^_+|_+$/g, ``)\n\nconst normalizeXmlSafeIdValue = (value: string) =>\n trimIdSeparators(\n value\n .trim()\n .replaceAll(`/`, `_`)\n .replace(INVALID_XML_SAFE_ID_CHARACTER_PATTERN, `_`),\n )\n\nexport const createXmlSafeId = (value: string) => {\n if (\n XML_SAFE_ID_PATTERN.test(value) &&\n !XML_RESERVED_ID_START_PATTERN.test(value)\n ) {\n return value\n }\n\n const normalized = normalizeXmlSafeIdValue(value)\n const valueWithValidStart =\n normalized && !XML_RESERVED_ID_START_PATTERN.test(normalized)\n ? normalized\n : `${XML_SAFE_ID_START_FALLBACK}${normalized}`\n\n if (XML_SAFE_ID_START_PATTERN.test(valueWithValidStart)) {\n return valueWithValidStart\n }\n\n return `${XML_SAFE_ID_START_FALLBACK}${valueWithValidStart}`\n}\n\nexport const createUniqueXmlSafeId = (value: string, usedIds: Set<string>) => {\n const id = createXmlSafeId(value)\n\n if (!usedIds.has(id)) {\n usedIds.add(id)\n\n return id\n }\n\n let index = 2\n let nextId = `${id}-${index}`\n\n while (usedIds.has(nextId)) {\n index += 1\n nextId = `${id}-${index}`\n }\n\n usedIds.add(nextId)\n\n return nextId\n}\n\nexport const createXmlSafeIdFactory = () => {\n const usedIds = new Set<string>()\n\n return (value: string) => createUniqueXmlSafeId(value, usedIds)\n}\n","import {\n detectMimeTypeFromName,\n escapeXmlAttributeValue,\n} from \"@prose-reader/shared\"\nimport { createXmlSafeIdFactory } from \"../utils/createXmlSafeId\"\nimport { getUriBasename } from \"../utils/uri\"\nimport type { Archive } from \"./types\"\n\n/**\n * @important\n * Make sure the urls are on the same origin or the cors header is set otherwise\n * the resource cannot be consumed as it is on the web.\n */\nexport const createArchiveFromUrls = async (\n urls: string[],\n options?: { useRenditionFlow: boolean },\n): Promise<Archive> => {\n const createSafeId = createXmlSafeIdFactory()\n const resources = urls.map((url) => ({\n id: createSafeId(url),\n url,\n }))\n const opfFileData = `\n <?xml version=\"1.0\" encoding=\"UTF-8\"?><package xmlns=\"http://www.idpf.org/2007/opf\" version=\"2.0\" unique-identifier=\"bookid\">\n <metadata>\n <meta property=\"rendition:layout\">${options?.useRenditionFlow ? `reflowable` : `pre-paginated`}</meta>\n ${options?.useRenditionFlow ? `<meta property=\"rendition:flow\">scrolled-continuous</meta>` : ``}\n </metadata>\n <manifest>\n ${resources\n .map(({ id, url }) => {\n const mediaType = detectMimeTypeFromName(url)\n\n return `<item id=\"${id}\" href=\"${escapeXmlAttributeValue(url)}\" media-type=\"${escapeXmlAttributeValue(mediaType ?? ``)}\"/>`\n })\n .join(`\\n`)}\n </manifest>\n <spine>\n ${resources.map(({ id }) => `<itemref idref=\"${id}\" />`).join(`\\n`)}\n </spine>\n </package>\n `\n\n const filesFromUrl: Archive[`records`] = urls.map((url) => ({\n dir: false,\n basename: getUriBasename(url),\n encodingFormat: detectMimeTypeFromName(url),\n uri: url,\n size: 100 / urls.length,\n blob: async () => {\n const response = await fetch(url)\n\n return response.blob()\n },\n string: async () => ``,\n }))\n\n const opfFile: Archive[`records`][number] = {\n dir: false,\n basename: `content.opf`,\n uri: `content.opf`,\n size: 0,\n blob: async () => new Blob(),\n string: async () => opfFileData,\n }\n\n return {\n filename: ``,\n records: [opfFile, ...filesFromUrl],\n close: () => Promise.resolve(),\n }\n}\n","import { Report } from \"./report\"\n\nexport const configure = ({\n enableReport,\n}: {\n enableReport?: boolean\n} = {}) => {\n Report.enable(!!enableReport)\n}\n","import type { Archive } from \"../archives/types\"\n\nexport const getArchiveOpfInfo = (archive: Archive) => {\n const filesAsArray = Object.values(archive.records).filter(\n (file) => !file.dir,\n )\n const file = filesAsArray.find((file) => file.uri.endsWith(`.opf`))\n\n return {\n data: file,\n basePath: file?.uri.substring(0, file.uri.lastIndexOf(`/`)) || ``,\n }\n}\n","import { type OpfMetadata, parseOpf } from \"@prose-reader/archive-parser\"\nimport type { Archive } from \"../archives/types\"\nimport { getArchiveOpfInfo } from \"./getArchiveOpfInfo\"\n\nexport type ArchiveOpfParsed = {\n opf: OpfMetadata\n basePath: string\n}\n\n/**\n * Loads and parses the package OPF from `archive`.\n *\n * **Attention — not memoized:** every call re-reads the OPF record and runs\n * `parseOpf`. Manifest generation already avoids redundant parses by awaiting\n * this once and passing `archiveOpf` into manifest hooks. The resource pipeline\n * (`generateResourceFromArchive` → `defaultHook`) invokes this again per\n * resource unless callers later thread parsed OPF in explicitly (optional\n * parameter, archive open hook, etc.).\n */\nexport async function readArchiveOpf(\n archive: Archive,\n): Promise<ArchiveOpfParsed | undefined> {\n const { data: opsFile, basePath } = getArchiveOpfInfo(archive) || {}\n\n if (!opsFile || opsFile.dir) {\n return undefined\n }\n\n const raw = await opsFile.string()\n\n return {\n opf: parseOpf(raw),\n basePath,\n }\n}\n","import {\n APPLE_IBOOKS_DISPLAY_OPTIONS_FILENAME,\n parseAppleDisplayOptionsXml,\n resolveArchiveMetadata,\n} from \"@prose-reader/archive-parser\"\nimport type { Manifest } from \"@prose-reader/shared\"\nimport type { Archive } from \"../../../archives/types\"\n\nconst appleDisplayOptionsBasename =\n APPLE_IBOOKS_DISPLAY_OPTIONS_FILENAME.toLowerCase()\n\nexport const apple =\n ({ archive }: { archive: Archive; baseUrl: string }) =>\n async (manifest: Manifest): Promise<Manifest> => {\n const infoFile = archive.records.find(\n (file) =>\n !file.dir &&\n file.basename.toLowerCase() === appleDisplayOptionsBasename,\n )\n\n if (!infoFile || infoFile.dir) {\n return manifest\n }\n\n const content = await infoFile.string()\n\n try {\n const parsed = parseAppleDisplayOptionsXml(content)\n const { renditionLayout } = resolveArchiveMetadata(parsed)\n\n return {\n ...manifest,\n renditionLayout: manifest.renditionLayout ?? renditionLayout,\n }\n } catch (e) {\n console.error(\n `Unable to parse ${APPLE_IBOOKS_DISPLAY_OPTIONS_FILENAME} for content\\n`,\n content,\n )\n console.error(e)\n\n return manifest\n }\n }\n","import {\n COMIC_INFO_FILENAME,\n parseComicInfo,\n resolveArchiveMetadata,\n} from \"@prose-reader/archive-parser\"\nimport type { Manifest } from \"@prose-reader/shared\"\nimport type { Archive } from \"../../../archives/types\"\n\nconst comicInfoFilenameLower = COMIC_INFO_FILENAME.toLowerCase()\n\nexport const comicInfo =\n ({ archive }: { archive: Archive; baseUrl: string }) =>\n async (manifest: Manifest): Promise<Manifest> => {\n const comicInfoFile = archive.records.find(\n (file) =>\n file.basename.toLowerCase() === comicInfoFilenameLower && !file.dir,\n )\n\n if (!comicInfoFile || comicInfoFile.dir) {\n return manifest\n }\n\n const manifestWithoutComicInfo = {\n ...manifest,\n spineItems: manifest.spineItems\n .filter(\n (item) => !item.id.toLowerCase().endsWith(comicInfoFilenameLower),\n )\n .map((item, _, items) => ({\n ...item,\n progressionWeight: 1 / items.length,\n })),\n }\n\n const content = await comicInfoFile.string()\n\n try {\n const parsed = parseComicInfo(content)\n const resolved = resolveArchiveMetadata(parsed)\n\n return {\n ...manifestWithoutComicInfo,\n readingDirection: resolved.readingDirection ?? `ltr`,\n }\n } catch (e) {\n console.error(\n `Unable to parse ${COMIC_INFO_FILENAME} for content\\n`,\n content,\n )\n console.error(e)\n\n return manifestWithoutComicInfo\n }\n }\n","export const createManifestResourceHref = ({\n baseUrl = ``,\n resourcePath,\n}: {\n baseUrl?: string\n resourcePath: string\n}) => {\n if (!baseUrl && /^https?:\\/\\//.test(resourcePath)) {\n return encodeURI(resourcePath)\n }\n\n const hrefBaseUrl = baseUrl\n ? `${baseUrl}${baseUrl.endsWith(`/`) ? `` : `/`}`\n : `file://`\n\n return encodeURI(`${hrefBaseUrl}${resourcePath}`)\n}\n","import type { Manifest } from \"@prose-reader/shared\"\nimport type { Archive } from \"../../../archives/types\"\nimport { createXmlSafeIdFactory } from \"../../../utils/createXmlSafeId\"\nimport { createManifestResourceHref } from \"../createManifestResourceHref\"\n\nexport const defaultHook =\n ({ archive, baseUrl }: { archive: Archive; baseUrl: string }) =>\n async (): Promise<Manifest> => {\n const files = Object.values(archive.records).filter((file) => !file.dir)\n const createSafeId = createXmlSafeIdFactory()\n const filesWithIds = files.map((file) => ({\n file,\n id: createSafeId(file.uri),\n }))\n\n return {\n filename: archive.filename,\n title:\n archive.records.find(({ dir }) => dir)?.basename.replace(/\\/$/, ``) ||\n archive.filename,\n renditionLayout: undefined,\n renditionSpread: `auto`,\n readingDirection: `ltr`,\n spineItems: filesWithIds\n .filter(({ file }) => !file.basename.endsWith(`.db`))\n .map(({ file, id }, index) => {\n return {\n id,\n index,\n href: createManifestResourceHref({\n baseUrl,\n resourcePath: file.uri,\n }),\n renditionLayout: undefined,\n progressionWeight: 1 / files.length,\n pageSpreadLeft: undefined,\n pageSpreadRight: undefined,\n mediaType: file.encodingFormat,\n }\n }),\n items: filesWithIds.map(({ file, id }) => ({\n id,\n href: encodeURI(`${baseUrl}${file.uri}`),\n })),\n }\n }\n","import type { Archive } from \"../archives/types\"\nimport type { ArchiveOpfParsed } from \"./readArchiveOpf\"\n\nexport const getSpineItemFilesFromArchive = async ({\n archive,\n archiveOpf,\n}: {\n archive: Archive\n archiveOpf: ArchiveOpfParsed | undefined\n}) => {\n if (!archiveOpf) return []\n\n const { opf, basePath: opfBasePath } = archiveOpf\n const { spineRows } = opf\n\n const archiveSpineItems = archive.records.filter((file) => {\n return spineRows.find((item) => {\n if (!opfBasePath) return `${item.href}` === file.uri\n return `${opfBasePath}/${item.href}` === file.uri\n })\n })\n\n return archiveSpineItems\n}\n","import { resolveArchiveMetadata } from \"@prose-reader/archive-parser\"\nimport type { Manifest } from \"@prose-reader/shared\"\nimport type { Archive } from \"../../../../archives/types\"\nimport { getArchiveOpfInfo } from \"../../../../epubs/getArchiveOpfInfo\"\nimport { getSpineItemFilesFromArchive } from \"../../../../epubs/getSpineItemFilesFromArchive\"\nimport type { ArchiveOpfParsed } from \"../../../../epubs/readArchiveOpf\"\nimport { Report } from \"../../../../report\"\n\nexport const getItemsFromDoc = (\n manifestItems: ReadonlyArray<{\n id: string\n href: string\n mediaType?: string\n }>,\n archive: Archive,\n getBaseUrlForHref?: (href: string) => string,\n) => {\n const { basePath: opfBasePath } = getArchiveOpfInfo(archive) || {}\n\n return manifestItems.map((el) => {\n const href = el.href\n const baseUrl = getBaseUrlForHref?.(href) ?? ``\n\n return {\n href: opfBasePath\n ? `${baseUrl}${opfBasePath}/${href}`\n : `${baseUrl}${href}`,\n id: el.id,\n mediaType: el.mediaType,\n }\n })\n}\n\nconst manifestRenditionFlow = (\n raw: string | undefined,\n): NonNullable<Manifest[`renditionFlow`]> => {\n const v = raw?.trim()\n if (\n v === `scrolled-continuous` ||\n v === `scrolled-doc` ||\n v === `paginated` ||\n v === `auto`\n ) {\n return v\n }\n return `auto`\n}\n\nconst manifestRenditionSpread = (\n raw: string | undefined,\n): Manifest[`renditionSpread`] => {\n const v = raw?.trim()\n if (\n v === `none` ||\n v === `landscape` ||\n v === `portrait` ||\n v === `both` ||\n v === `auto`\n ) {\n return v\n }\n return undefined\n}\n\ntype ManifestGuideEntry = NonNullable<Manifest[`guide`]>[number]\n\nconst manifestGuideType = (\n raw: string | undefined,\n): ManifestGuideEntry[`type`] | undefined => {\n const v = raw?.trim()\n if (\n v === `cover` ||\n v === `title-page` ||\n v === `copyright-page` ||\n v === `text`\n ) {\n return v\n }\n return undefined\n}\n\nexport const epubHook =\n ({\n archive,\n baseUrl,\n archiveOpf,\n }: {\n archive: Archive\n baseUrl: string\n archiveOpf: ArchiveOpfParsed | undefined\n }) =>\n async (manifest: Manifest): Promise<Manifest> => {\n if (!archiveOpf) {\n return manifest\n }\n\n const { opf, basePath: opfBasePath } = archiveOpf\n const resolved = resolveArchiveMetadata(opf)\n\n Report.groupCollapsed(...Report.getGroupArgs(`OPF parsed`))\n Report.log(`opf`, opf)\n Report.groupEnd()\n\n const publisherRenditionLayout = opf.renditionLayoutMeta?.trim()\n const packageRenditionLayout =\n publisherRenditionLayout === `reflowable` ||\n publisherRenditionLayout === `pre-paginated`\n ? publisherRenditionLayout\n : resolved.renditionLayout\n\n const title =\n opf.title?.trim() ||\n archive.records.find(({ dir }) => dir)?.basename ||\n ``\n\n const readingDirection = resolved.readingDirection ?? `ltr`\n\n const archiveSpineItems = await getSpineItemFilesFromArchive({\n archive,\n archiveOpf,\n })\n\n const totalSize = archiveSpineItems.reduce(\n (size, file) => file.size + size,\n 0,\n )\n\n const guideRefs = opf.guide\n const guide: ManifestGuideEntry[] = []\n for (const elm of guideRefs) {\n const type = manifestGuideType(elm.type)\n if (type === undefined) continue\n guide.push({ href: elm.href, title: elm.title, type })\n }\n\n return {\n filename: archive.filename,\n renditionLayout: packageRenditionLayout,\n renditionFlow: manifestRenditionFlow(opf.renditionFlowMeta),\n renditionSpread: manifestRenditionSpread(opf.renditionSpreadMeta),\n title,\n readingDirection,\n /**\n * @see https://www.w3.org/TR/epub/#sec-itemref-elem\n */\n spineItems: opf.spineRows.map((row, index) => {\n const itemSize =\n archive.records.find((file) => file.uri.endsWith(row.href))?.size || 0\n\n const hrefBaseUri = baseUrl\n ? baseUrl\n : /^https?:\\/\\//.test(row.href)\n ? ``\n : `file://`\n\n return {\n id: row.id,\n index,\n href: row.href.startsWith(`https://`)\n ? row.href\n : opfBasePath\n ? `${hrefBaseUri}${opfBasePath}/${row.href}`\n : `${hrefBaseUri}${row.href}`,\n renditionLayout: row.renditionLayout ?? packageRenditionLayout,\n ...(row.renditionFlow !== undefined\n ? { renditionFlow: row.renditionFlow }\n : {}),\n progressionWeight: itemSize / totalSize,\n pageSpreadLeft: row.pageSpreadLeft,\n pageSpreadRight: row.pageSpreadRight,\n mediaType: row.mediaType,\n }\n }),\n items: getItemsFromDoc(opf.manifestItems, archive, (href) => {\n if (/^https?:\\/\\//.test(href)) {\n return ``\n }\n\n return baseUrl || `file://`\n }),\n guide: guide.length > 0 ? guide : undefined,\n }\n }\n","import { isXmlBasedMimeType, type Manifest } from \"@prose-reader/shared\"\nimport { XmlDocument } from \"xmldoc\"\nimport type { Archive } from \"../../../archives/types\"\nimport { getSpineItemFilesFromArchive } from \"../../../epubs/getSpineItemFilesFromArchive\"\nimport type { ArchiveOpfParsed } from \"../../../epubs/readArchiveOpf\"\n\nconst hasDocMetaViewport = (doc: XmlDocument) => {\n const metaElm = doc\n .descendantWithPath(\"head\")\n ?.childrenNamed(\"meta\")\n .find((node) => node.attr.name === \"viewport\")\n\n return !!(metaElm && metaElm.attr.name === \"viewport\")\n}\n\nconst allFilesHaveViewportMeta = (files: Archive[\"records\"]) =>\n files.reduce(async (result, current) => {\n const _result = await result\n\n if (!_result) return false\n\n if (\n !isXmlBasedMimeType({\n mimeType: current.encodingFormat,\n uri: current.uri,\n })\n ) {\n return false\n }\n\n const file = current.dir ? null : await current.string()\n\n if (!file) return false\n\n return hasDocMetaViewport(new XmlDocument(file))\n }, Promise.resolve(true))\n\nexport const epubOptimizerHook =\n ({\n archive,\n archiveOpf,\n }: {\n archive: Archive\n baseUrl: string\n archiveOpf: ArchiveOpfParsed | undefined\n }) =>\n async (manifest: Manifest): Promise<Manifest> => {\n const bookIsFullReflowable =\n manifest.renditionLayout === \"reflowable\" &&\n manifest.spineItems.every((item) => item.renditionLayout === \"reflowable\")\n\n if (bookIsFullReflowable) {\n const files = await getSpineItemFilesFromArchive({\n archive,\n archiveOpf,\n })\n\n const hasAllViewport = await allFilesHaveViewportMeta(files)\n\n if (hasAllViewport) {\n return {\n ...manifest,\n spineItems: manifest.spineItems.map((item) => ({\n ...item,\n renditionLayout: \"pre-paginated\",\n })),\n renditionLayout: \"pre-paginated\",\n }\n }\n }\n\n return manifest\n }\n","import {\n KOBO_DISPLAY_OPTIONS_FILENAME,\n type KoboMetadata,\n parseKoboXml,\n resolveArchiveMetadata,\n} from \"@prose-reader/archive-parser\"\nimport type { Manifest } from \"@prose-reader/shared\"\nimport type { Archive } from \"../../../archives/types\"\n\nconst extractKoboInformationFromArchive = async (\n archive: Archive,\n): Promise<KoboMetadata> => {\n let renditionLayout: KoboMetadata[\"renditionLayout\"]\n\n await Promise.all(\n archive.records.map(async (file) => {\n if (file.dir || !file.uri.endsWith(KOBO_DISPLAY_OPTIONS_FILENAME)) return\n\n const content = await file.string()\n\n try {\n const { renditionLayout: layout } = parseKoboXml(content)\n if (layout) renditionLayout = layout\n } catch (e) {\n console.error(\n `Unable to parse ${KOBO_DISPLAY_OPTIONS_FILENAME} for content\\n`,\n content,\n )\n console.error(e)\n }\n }),\n )\n\n return {\n kind: `kobo`,\n ...(renditionLayout !== undefined ? { renditionLayout } : {}),\n }\n}\n\nexport const kobo =\n ({ archive }: { archive: Archive; baseUrl: string }) =>\n async (manifest: Manifest): Promise<Manifest> => {\n const koboMeta = await extractKoboInformationFromArchive(archive)\n const { renditionLayout } = resolveArchiveMetadata(koboMeta)\n\n return {\n ...manifest,\n renditionLayout: manifest.renditionLayout ?? renditionLayout,\n }\n }\n","import type { Archive } from \"../archives/types\"\n\nconst hasOpfExtension = (path: string) => path.toLowerCase().endsWith(`.opf`)\n\nexport const isArchiveEpub = (archive: Archive) => {\n return archive.records.some(\n (file) =>\n !file.dir &&\n (hasOpfExtension(file.basename) || hasOpfExtension(file.uri)),\n )\n}\n","import {\n detectMimeTypeFromName,\n isMediaContentMimeType,\n type Manifest,\n parseContentType,\n} from \"@prose-reader/shared\"\nimport type { Archive } from \"../../../archives/types\"\nimport { isArchiveEpub } from \"../../../epubs/isArchiveEpub\"\n\n/**\n * If we don't have a regular epub, we have an archive that could be many things.\n * We try to refine the manifest based on the type of content as much as possible.\n */\nexport const nonEpub =\n ({ archive }: { archive: Archive; baseUrl: string }) =>\n async (manifest: Manifest): Promise<Manifest> => {\n const isEpub = isArchiveEpub(archive)\n\n if (isEpub) return manifest\n\n return {\n ...manifest,\n spineItems: manifest.spineItems.map((spineItem) => {\n const archiveItem = archive.records.find((item) =>\n decodeURI(spineItem.href).endsWith(item.uri),\n )\n\n const mimeType =\n parseContentType(archiveItem?.encodingFormat ?? \"\") ||\n detectMimeTypeFromName(archiveItem?.basename ?? \"\")\n\n return {\n ...spineItem,\n renditionLayout:\n mimeType && isMediaContentMimeType(mimeType)\n ? `pre-paginated`\n : spineItem.renditionLayout,\n }\n }),\n }\n }\n","import { XmlElement, XmlTextNode } from \"xmldoc\"\n\nexport const getXmlElementInnerText = (\n node: XmlElement | undefined,\n): string => {\n if (!node) return \"\"\n\n return node.children\n .map((child) => {\n if (child instanceof XmlTextNode) return child.text\n if (child instanceof XmlElement) return getXmlElementInnerText(child)\n return \"\"\n })\n .join(\"\")\n .trim()\n}\n","import {\n type OpfMetadata,\n tokenizeXmlSpaceSeparatedList,\n} from \"@prose-reader/archive-parser\"\nimport type { Manifest } from \"@prose-reader/shared\"\nimport { urlJoin } from \"@prose-reader/shared\"\nimport { XmlDocument, type XmlElement, type XmlNodeBase } from \"xmldoc\"\nimport { type Archive, getArchiveOpfInfo } from \"..\"\nimport { getUriBasePath } from \"../utils/uri\"\nimport { getXmlElementInnerText } from \"./xml\"\n\ntype Toc = NonNullable<Manifest[`nav`]>[`toc`]\ntype TocItem = NonNullable<Manifest[`nav`]>[`toc`][number]\n\nconst manifestItemIsNavDocument = (item: {\n readonly properties?: string\n}): boolean => tokenizeXmlSpaceSeparatedList(item.properties).includes(`nav`)\n\n/**\n * @see https://www.w3.org/TR/epub-33/#sec-nav-def-model\n */\nconst extractNavChapter = (\n li: XmlElement,\n { basePath, baseUrl }: { basePath: string; baseUrl: string },\n) => {\n const chp: TocItem = {\n contents: [],\n path: ``,\n href: ``,\n title: ``,\n }\n\n let contentNode = li.childNamed(`span`) || li.childNamed(`a`)\n\n chp.title =\n (contentNode?.attr.title ||\n contentNode?.val.trim() ||\n getXmlElementInnerText(contentNode)) ??\n ``\n\n let node = contentNode?.name\n\n if (node !== `a`) {\n contentNode = li.descendantWithPath(`${node}.a`)\n if (contentNode) {\n node = contentNode.name.toLowerCase()\n }\n }\n\n if (node === `a` && contentNode?.attr.href) {\n chp.path = urlJoin(basePath, contentNode.attr.href)\n chp.href = urlJoin(baseUrl, basePath, contentNode.attr.href)\n }\n const sublistNode = li.childNamed(`ol`)\n if (sublistNode) {\n const children = sublistNode.childrenNamed(`li`)\n if (children && children.length > 0) {\n chp.contents = children.map((child) =>\n extractNavChapter(child, { basePath, baseUrl }),\n )\n }\n }\n\n return chp\n}\n\nconst buildTOCFromNav = (\n doc: XmlDocument,\n { basePath, baseUrl }: { basePath: string; baseUrl: string },\n) => {\n const toc: Toc = []\n\n let navDataChildren: XmlNodeBase[] | undefined\n\n if (doc.descendantWithPath(`body.nav.ol`)) {\n navDataChildren = doc.descendantWithPath(`body.nav.ol`)?.children\n } else if (doc.descendantWithPath(`body.section.nav.ol`)) {\n navDataChildren = doc.descendantWithPath(`body.section.nav.ol`)?.children\n }\n\n if (navDataChildren && navDataChildren.length > 0) {\n navDataChildren\n .filter((li) => (li as XmlElement).name === `li`)\n .forEach((li) => {\n toc.push(extractNavChapter(li as XmlElement, { basePath, baseUrl }))\n })\n }\n\n return toc\n}\n\nconst parseTocFromNavPath = async (\n opf: OpfMetadata,\n archive: Archive,\n { baseUrl }: { baseUrl: string },\n) => {\n const navItem = opf.manifestItems.find(manifestItemIsNavDocument)\n\n if (navItem?.href) {\n const tocFile = Object.values(archive.records).find((item) =>\n item.uri.endsWith(navItem.href),\n )\n\n if (tocFile && !tocFile.dir) {\n const doc = new XmlDocument(await tocFile.string())\n\n const tocFileBasePath = getUriBasePath(tocFile.uri)\n\n /**\n * links inside toc.xhtml are relative to the toc.xhtml file,\n * not the opf file anymore\n */\n return buildTOCFromNav(doc, { basePath: tocFileBasePath, baseUrl })\n }\n }\n}\n\nconst mapNcxChapter = (\n point: XmlElement,\n {\n opfBasePath,\n baseUrl,\n prefix,\n }: { opfBasePath: string; baseUrl: string; prefix: string },\n) => {\n const src = point?.childNamed(`${prefix}content`)?.attr.src || ``\n\n const out: TocItem = {\n title:\n point?.descendantWithPath(`${prefix}navLabel.${prefix}text`)?.val || ``,\n path: urlJoin(opfBasePath, src),\n href: urlJoin(baseUrl, opfBasePath, src),\n contents: [],\n }\n const children = point.childrenNamed(`${prefix}navPoint`)\n if (children && children.length > 0) {\n out.contents = children.map((pt) =>\n mapNcxChapter(pt, { opfBasePath, baseUrl, prefix }),\n )\n }\n\n return out\n}\n\nconst buildTOCFromNCX = (\n ncxData: XmlDocument,\n { opfBasePath, baseUrl }: { opfBasePath: string; baseUrl: string },\n) => {\n const toc: NonNullable<Manifest[`nav`]>[`toc`] = []\n\n const rootTagName = ncxData.name\n let prefix = ``\n if (rootTagName.indexOf(`:`) !== -1) {\n prefix = `${rootTagName.split(`:`)[0]}:`\n }\n\n ncxData\n .childNamed(`${prefix}navMap`)\n ?.childrenNamed(`${prefix}navPoint`)\n .forEach((point) => {\n toc.push(mapNcxChapter(point, { opfBasePath, baseUrl, prefix }))\n })\n\n return toc\n}\n\nconst parseTocFromNcx = async ({\n opf,\n opfBasePath,\n baseUrl,\n archive,\n}: {\n opf: OpfMetadata\n opfBasePath: string\n archive: Archive\n baseUrl: string\n}) => {\n const ncxId = opf.spineTocIdref\n\n if (ncxId) {\n const ncxItem = opf.manifestItems.find((item) => item.id === ncxId)\n\n if (ncxItem) {\n const ncxPath = `${opfBasePath}${opfBasePath === `` ? `` : `/`}${ncxItem.href}`\n\n const file = Object.values(archive.records).find((item) =>\n item.uri.endsWith(ncxPath),\n )\n\n if (file && !file.dir) {\n const ncxData = new XmlDocument(await file.string())\n\n return buildTOCFromNCX(ncxData, { opfBasePath, baseUrl })\n }\n }\n }\n}\n\nexport const parseToc = async (\n opf: OpfMetadata,\n archive: Archive,\n { baseUrl }: { baseUrl: string },\n) => {\n const { basePath: opfBasePath } = getArchiveOpfInfo(archive) || {}\n\n const tocFromNav = await parseTocFromNavPath(opf, archive, {\n baseUrl,\n })\n\n if (tocFromNav) {\n return tocFromNav\n }\n\n const tocFromNcx = await parseTocFromNcx({\n opf,\n opfBasePath: opfBasePath ?? ``,\n archive,\n baseUrl,\n })\n\n if (tocFromNcx) {\n return tocFromNcx\n }\n}\n","import {\n detectMimeTypeFromName,\n type Manifest,\n parseContentType,\n} from \"@prose-reader/shared\"\nimport type { Archive } from \"../../../archives/types\"\n\ntype Toc = NonNullable<Manifest[\"nav\"]>[\"toc\"]\n\nexport const normalizeFilenameAsTitle = (filename: string) =>\n filename\n .replace(/\\.[^.]+$/, \"\")\n .replace(/[_-]/g, \" \")\n .replace(/\\s+/g, \" \")\n .trim()\n\nexport const buildAudiobookToc = (\n manifest: Manifest,\n archive: Archive,\n): Toc | undefined => {\n if (manifest.spineItems.length === 0) return undefined\n\n const allAudio = manifest.spineItems.every((item) => {\n const mimeType =\n parseContentType(item.mediaType ?? \"\") ||\n detectMimeTypeFromName(item.href)\n\n return mimeType?.startsWith(\"audio/\")\n })\n\n if (!allAudio) return undefined\n\n return manifest.spineItems.map((item) => {\n const record = archive.records.find(\n (r) => !r.dir && decodeURI(item.href).endsWith(r.uri),\n )\n\n return {\n title: normalizeFilenameAsTitle(record?.basename ?? item.href),\n href: item.href,\n path: record?.uri ?? item.href,\n contents: [],\n }\n })\n}\n","import { type Manifest, urlJoin } from \"@prose-reader/shared\"\nimport type { Archive } from \"../../../archives/types\"\nimport type { ArchiveOpfParsed } from \"../../../epubs/readArchiveOpf\"\nimport { parseToc } from \"../../../parsers/nav\"\nimport { sortByTitleComparator } from \"../../../utils/sortByTitleComparator\"\nimport { buildAudiobookToc } from \"./audiobookToc\"\n\ntype Toc = NonNullable<Manifest[\"nav\"]>[\"toc\"]\ntype TocItem = NonNullable<Manifest[\"nav\"]>[\"toc\"][number]\n\nconst buildFolderFallbackToc = (\n archive: Archive,\n { baseUrl }: { baseUrl: string },\n): Toc => {\n const filesSortedByAlpha = [...archive.records].sort((a, b) =>\n sortByTitleComparator(a.uri, b.uri),\n )\n\n const combineWith = (\n toc: TocItem[],\n folder: string,\n subFolders: string[],\n href: string,\n path: string,\n ): TocItem[] => {\n const foundEntry = toc.find((entry) => entry.title === folder)\n const [nextFolderCursor, ...nextSubFolders] = subFolders\n\n if (foundEntry) {\n if (nextFolderCursor) {\n return [\n ...toc.filter((entry) => entry !== foundEntry),\n {\n ...foundEntry,\n contents: [\n ...foundEntry.contents,\n ...combineWith(\n foundEntry.contents,\n nextFolderCursor,\n nextSubFolders,\n href,\n path,\n ),\n ],\n } satisfies TocItem,\n ]\n }\n\n const previousRegisteredPathWasLonger =\n foundEntry.path.split(\"/\").length > path.split(\"/\").length\n\n if (previousRegisteredPathWasLonger) {\n return [\n ...toc.filter((entry) => entry !== foundEntry),\n {\n ...foundEntry,\n path,\n href,\n } satisfies TocItem,\n ]\n }\n\n return toc\n }\n\n if (nextFolderCursor) {\n return [\n ...toc,\n {\n contents: combineWith(\n [],\n nextFolderCursor,\n nextSubFolders,\n href,\n path,\n ),\n href,\n path,\n title: folder,\n },\n ]\n }\n\n return [\n ...toc,\n {\n contents: [],\n href,\n path,\n title: folder,\n },\n ]\n }\n\n return filesSortedByAlpha.reduce((acc, file) => {\n if (file.dir) return acc\n\n const folders = file.uri.split(\"/\").slice(0, -1)\n const [firstFolder, ...restFolders] = folders\n\n if (!firstFolder) return acc\n\n const href = urlJoin(baseUrl, encodeURI(file.uri)).replace(/\\/$/, \"\")\n const path = file.uri.replace(/\\/$/, \"\")\n\n return combineWith(acc, firstFolder, restFolders, href, path)\n }, [] as Toc)\n}\n\nconst resolveTocFromArchive = async (\n archive: Archive,\n manifest: Manifest,\n {\n baseUrl,\n archiveOpf,\n }: { baseUrl: string; archiveOpf: ArchiveOpfParsed | undefined },\n): Promise<Toc | undefined> => {\n if (archiveOpf) {\n const toc = await parseToc(archiveOpf.opf, archive, { baseUrl })\n\n // Keep explicit empty TOC for EPUB-like inputs even when there is no nav file.\n return toc || []\n }\n\n const audiobookToc = buildAudiobookToc(manifest, archive)\n if (audiobookToc) return audiobookToc\n\n const toc = buildFolderFallbackToc(archive, { baseUrl })\n\n if (toc.length === 0) {\n return undefined\n }\n\n return toc\n}\n\n/**\n * Resolve the table of contents from a single entry point.\n * Internally handles EPUB nav, NCX, and folder fallback.\n */\nexport const tocHook =\n ({\n archive,\n baseUrl,\n archiveOpf,\n }: {\n archive: Archive\n baseUrl: string\n archiveOpf: ArchiveOpfParsed | undefined\n }) =>\n async (manifest: Manifest): Promise<Manifest> => {\n if (manifest.nav) return manifest\n\n const toc = await resolveTocFromArchive(archive, manifest, {\n baseUrl,\n archiveOpf,\n })\n if (!toc) return manifest\n\n return {\n ...manifest,\n nav: {\n toc,\n },\n }\n }\n","import type { Archive } from \"../../archives/types\"\nimport { readArchiveOpf } from \"../../epubs/readArchiveOpf\"\nimport type { StreamerManifestHooks } from \"../../hooks\"\nimport { Report } from \"../../report\"\nimport { apple } from \"./hooks/apple\"\nimport { comicInfo } from \"./hooks/comicInfo\"\nimport { defaultHook } from \"./hooks/default\"\nimport { epubHook } from \"./hooks/epub/epub\"\nimport { epubOptimizerHook } from \"./hooks/epubOptimizer\"\nimport { kobo } from \"./hooks/kobo\"\nimport { nonEpub } from \"./hooks/nonEpub\"\nimport { tocHook } from \"./hooks/toc\"\n\nconst normalizeBaseUrl = (baseUrl: string | undefined) => {\n if (!baseUrl) return ``\n\n return baseUrl.endsWith(`/`) ? baseUrl : `${baseUrl}/`\n}\n\nexport const generateManifestFromArchive = async (\n archive: Archive,\n {\n baseUrl = ``,\n hooks = {},\n }: { baseUrl?: string; hooks?: StreamerManifestHooks } = {},\n) => {\n Report.log(\"Generating manifest from archive\", archive)\n\n const archiveOpf = await readArchiveOpf(archive)\n const normalizedBaseUrl = normalizeBaseUrl(baseUrl)\n const createExternalHooks = (\n hookFactories: StreamerManifestHooks[keyof StreamerManifestHooks],\n ) =>\n (hookFactories ?? []).map((hook) =>\n hook({ archive, baseUrl: normalizedBaseUrl }),\n )\n\n const contentHooks = [\n epubHook({ archive, baseUrl: normalizedBaseUrl, archiveOpf }),\n comicInfo({ archive, baseUrl: normalizedBaseUrl }),\n apple({ archive, baseUrl: normalizedBaseUrl }),\n nonEpub({ archive, baseUrl: normalizedBaseUrl }),\n ...createExternalHooks(hooks.content),\n ]\n const spineHooks = createExternalHooks(hooks.spine)\n const presentationHooks = [\n epubOptimizerHook({ archive, baseUrl: normalizedBaseUrl, archiveOpf }),\n kobo({ archive, baseUrl: normalizedBaseUrl }),\n ...createExternalHooks(hooks.presentation),\n ]\n const navigationHooks = [\n tocHook({ archive, baseUrl: normalizedBaseUrl, archiveOpf }),\n ...createExternalHooks(hooks.navigation),\n ]\n const manifestHooks = [\n ...contentHooks,\n ...spineHooks,\n ...presentationHooks,\n ...navigationHooks,\n ]\n\n try {\n const baseManifestPromise = defaultHook({\n archive,\n baseUrl: normalizedBaseUrl,\n })()\n\n const manifest = await manifestHooks.reduce(async (manifest, gen) => {\n return await gen(await manifest)\n }, baseManifestPromise)\n\n Report.log(\"Generated manifest\", manifest)\n\n if (process.env.NODE_ENV === \"development\") {\n if (Report.isEnabled()) {\n const manifestStr = JSON.stringify(manifest, null, 2)\n Report.groupCollapsed(...Report.getGroupArgs(\"Generated manifest\"))\n Report.log(`\\n${manifestStr}`)\n Report.groupEnd()\n }\n }\n\n return manifest\n } catch (e) {\n Report.error(e)\n\n throw e\n }\n}\n","import { XmlDocument } from \"xmldoc\"\nimport type { Archive } from \"../../../archives/types\"\nimport type { HookResource } from \"./types\"\n\nconst hasCalibreCoverMeta = (doc: XmlDocument) => {\n const metaElm = doc\n .descendantWithPath(\"head\")\n ?.childrenNamed(\"meta\")\n .find((node) => node.attr.name === \"calibre:cover\")\n\n return !!(metaElm && metaElm.attr.name === \"calibre:cover\")\n}\n\nconst getBuggyCoverSvg = (doc: XmlDocument) => {\n return doc\n .descendantWithPath(\"body\")\n ?.descendantWithPath(\"div\")\n ?.childrenNamed(\"svg\")\n ?.find(\n (node) =>\n node.attr.width === \"100%\" && node.attr.preserveAspectRatio === \"none\",\n )\n}\n\nconst fixBuggyCover =\n ({ archive, resourcePath }: { archive: Archive; resourcePath: string }) =>\n async (resource: HookResource): Promise<HookResource> => {\n const file = Object.values(archive.records).find(\n (file) => file.uri === resourcePath && !file.dir,\n )\n\n if (file && !file.dir && file.basename.endsWith(`.xhtml`)) {\n const bodyToParse =\n typeof resource.body === `string` ? resource.body : await file.string()\n\n const opfXmlDoc = new XmlDocument(bodyToParse)\n\n if (hasCalibreCoverMeta(opfXmlDoc)) {\n const buggySvg = getBuggyCoverSvg(opfXmlDoc)\n\n if (buggySvg) {\n delete buggySvg.attr.preserveAspectRatio\n }\n\n return {\n ...resource,\n body: opfXmlDoc?.toString(),\n }\n }\n }\n\n return resource\n }\n\nexport const calibreFixHook =\n ({ archive, resourcePath }: { archive: Archive; resourcePath: string }) =>\n async (resource: HookResource): Promise<HookResource> => {\n return fixBuggyCover({ archive, resourcePath })(resource)\n }\n","import type { Archive } from \"../../../archives/types\"\nimport type { HookResource } from \"./types\"\n\nexport const cssFixHook =\n ({ archive, resourcePath }: { archive: Archive; resourcePath: string }) =>\n async (resource: HookResource): Promise<HookResource> => {\n const file = Object.values(archive.records).find(\n (file) => file.uri === resourcePath && !file.dir,\n )\n\n if (file && !file.dir && file.basename.endsWith(`.css`)) {\n const bodyToParse =\n typeof resource.body === `string` ? resource.body : await file.string()\n\n /**\n * Fix the potentially invalid writing mode present on some vertical book.\n * This has the benefit of making it compatible with firefox as well.\n */\n const newBody = bodyToParse.replaceAll(\n `-webkit-writing-mode`,\n `writing-mode`,\n )\n\n return {\n ...resource,\n body: newBody,\n }\n }\n\n return resource\n }\n","import type { Archive } from \"../../../archives/types\"\nimport { readArchiveOpf } from \"../../../epubs/readArchiveOpf\"\nimport { getItemsFromDoc } from \"../../manifest/hooks/epub/epub\"\nimport type { HookResource } from \"./types\"\n\n/**\n * Resolves `contentType` from the package OPF when possible, else from the URI.\n * Uses `readArchiveOpf` each time — see `generateResourceFromArchive` JSDoc for\n * why that can mean repeated OPF work on the resource path.\n */\nconst getMetadata = async (archive: Archive, resourcePath: string) => {\n const parsed = await readArchiveOpf(archive)\n\n if (parsed) {\n const { opf } = parsed\n const items = getItemsFromDoc(opf.manifestItems, archive, () => \"\")\n\n // we are comparing opf items relative absolute path in epub archive\n // against resourcePatch (which are absolute path in archive).\n // They should in theory match.\n const foundMediaType = items.find((item) =>\n resourcePath.endsWith(item.href),\n )?.mediaType\n\n if (foundMediaType) {\n return {\n mediaType: items.find((item) => resourcePath.endsWith(item.href))\n ?.mediaType,\n }\n }\n }\n\n return {\n mediaType: getContentTypeFromExtension(resourcePath),\n }\n}\n\nconst getContentTypeFromExtension = (uri: string) => {\n if (uri.endsWith(`.css`)) {\n return `text/css; charset=UTF-8`\n }\n if (uri.endsWith(`.jpg`)) {\n return `image/jpg`\n }\n if (uri.endsWith(`.xhtml`)) {\n return `application/xhtml+xml`\n }\n if (uri.endsWith(`.mp4`)) {\n return `video/mp4`\n }\n if (uri.endsWith(`.svg`)) {\n return `image/svg+xml`\n }\n}\n\nexport const defaultHook =\n ({ archive, resourcePath }: { archive: Archive; resourcePath: string }) =>\n async (resource: HookResource): Promise<HookResource> => {\n const file = Object.values(archive.records).find(\n (file) => file.uri === resourcePath && !file.dir,\n )\n\n if (!file || file.dir) return resource\n\n const metadata = await getMetadata(archive, resourcePath)\n\n return {\n ...resource,\n params: {\n ...resource.params,\n ...(file?.encodingFormat && {\n contentType: file.encodingFormat,\n }),\n ...(metadata.mediaType && {\n contentType: metadata.mediaType,\n }),\n },\n }\n }\n","import type { Archive } from \"../../../archives/types\"\nimport type { HookResource } from \"./types\"\n\nconst invalidSelfClosingTags = [\n \"div\",\n \"span\",\n \"p\",\n \"a\",\n \"li\",\n \"ul\",\n \"ol\",\n \"h1\",\n \"h2\",\n \"h3\",\n \"h4\",\n \"h5\",\n \"h6\",\n \"table\",\n \"tr\",\n \"td\",\n \"th\",\n \"thead\",\n \"tbody\",\n \"tfoot\",\n \"section\",\n \"article\",\n \"header\",\n \"footer\",\n \"nav\",\n \"aside\",\n \"main\",\n \"figure\",\n \"figcaption\",\n \"blockquote\",\n \"pre\",\n \"code\",\n \"form\",\n \"textarea\",\n \"select\",\n \"option\",\n \"button\",\n \"label\",\n \"fieldset\",\n \"legend\",\n \"caption\",\n \"dl\",\n \"dt\",\n \"dd\",\n \"iframe\",\n \"video\",\n \"audio\",\n \"canvas\",\n \"script\",\n \"style\",\n]\n\n/**\n * Some books uses xhtml files but also includes wrong self closin tags. This happens\n * a lot with kobo epub which have a lot of self closing <script ... />. This breaks on some\n * browser such as webkit. We use a regex to replace and fix them. There\n * is a first lighter regex which check if any such tag exist in the first place before running\n * the full replace regex empty.\n */\nexport const selfClosingTagsFixHook =\n ({ archive, resourcePath }: { archive: Archive; resourcePath: string }) =>\n async (resource: HookResource): Promise<HookResource> => {\n const file = Object.values(archive.records).find(\n (file) => file.uri === resourcePath && !file.dir,\n )\n\n if (file && !file.dir && file.basename.endsWith(`.xhtml`)) {\n const bodyToParse =\n typeof resource.body === `string` ? resource.body : await file.string()\n\n const tagCheckPattern = new RegExp(\n `<(${invalidSelfClosingTags.join(\"|\")})[\\\\s/>]`,\n \"i\",\n )\n if (!tagCheckPattern.test(bodyToParse)) {\n return resource\n }\n\n const tagPattern = new RegExp(\n `<(${invalidSelfClosingTags.join(\"|\")})(\\\\s[^>]*)?\\\\s*/>`,\n \"gi\",\n )\n\n const fixedBody = bodyToParse.replace(\n tagPattern,\n (_, tagName, attributes = \"\") => {\n // Convert to an opening and closing tag\n return `<${tagName} ${attributes.trim()}></${tagName}>`\n },\n )\n\n return {\n ...resource,\n body: fixedBody,\n }\n }\n\n return resource\n }\n","import type { Archive } from \"../..\"\nimport type { StreamerResourceHookFactory } from \"../../hooks\"\nimport { Report } from \"../../report\"\nimport { calibreFixHook } from \"./hooks/calibreFixHook\"\nimport { cssFixHook } from \"./hooks/cssFixHook\"\nimport { defaultHook } from \"./hooks/defaultHook\"\nimport { selfClosingTagsFixHook } from \"./hooks/selfClosingTagsFixHook\"\nimport type { HookResource } from \"./hooks/types\"\n\n/**\n * Builds one streamed asset from `archive` (hot path: typically once per\n * fetched file in the reader).\n *\n * **OPF limitation:** Unlike manifest generation, this API does not take\n * `ArchiveOpfParsed`. `defaultHook` calls `readArchiveOpf(archive)` to resolve\n * `media-type` from the package document, so EPUBs repeat OPF I/O + parse per\n * resource. Usually fine (small OPF); if profiling shows regressions, address\n * explicitly—e.g. optional `archiveOpf` (or equivalent) threaded from\n * `Streamer.fetchResource` / manifest, or an opt-in cache on `Archive` wired at\n * construction time—rather than hidden process-wide memoization.\n */\nexport const generateResourceFromArchive = async (\n archive: Archive,\n resourcePath: string,\n { hooks = [] }: { hooks?: StreamerResourceHookFactory[] } = {},\n) => {\n const defaultResource: HookResource = {\n params: {},\n }\n const externalHooks = hooks.map((hook) => hook({ archive, resourcePath }))\n\n /**\n * EPUB: `defaultHook` calls `readArchiveOpf` → repeated OPF I/O + parse per\n * resource until we thread `ArchiveOpfParsed` here (see export JSDoc).\n */\n const resourceHooks = [\n ...externalHooks,\n defaultHook({ archive, resourcePath }),\n selfClosingTagsFixHook({ archive, resourcePath }),\n cssFixHook({ archive, resourcePath }),\n calibreFixHook({ archive, resourcePath }),\n ]\n\n try {\n const resource = await resourceHooks.reduce(async (manifest, gen) => {\n return await gen(await manifest)\n }, Promise.resolve(defaultResource))\n\n Report.log(\"Generated resource\", resourcePath, resource)\n\n if (resource.body !== undefined) {\n return resource\n }\n\n const file = Object.values(archive.records).find(\n (file) => file.uri === resourcePath && !file.dir,\n )\n\n if (!file || file.dir) {\n throw new Error(`no file found for resourcePath:${resourcePath}`)\n }\n\n return {\n ...resource,\n body: await file.blob(),\n }\n } catch (e) {\n Report.error(e)\n\n throw e\n }\n}\n","import {\n BehaviorSubject,\n catchError,\n distinctUntilChanged,\n EMPTY,\n filter,\n first,\n from,\n ignoreElements,\n map,\n merge,\n mergeMap,\n NEVER,\n type ObservedValueOf,\n pairwise,\n Subject,\n shareReplay,\n startWith,\n switchMap,\n takeUntil,\n tap,\n timer,\n} from \"rxjs\"\nimport { Report } from \"../report\"\nimport type { Archive } from \"./types\"\n\nclass ArchiveEntry {\n state$ = new BehaviorSubject<{\n status: \"idle\" | \"loading\" | \"success\" | \"error\"\n error?: unknown | undefined\n archive?: undefined | Archive\n locks: number\n }>({\n status: `idle`,\n locks: 0,\n })\n\n constructor(private cleanArchiveAfter: number) {}\n\n update(update: Partial<ObservedValueOf<typeof this.state$>>) {\n this.state$.next({ ...this.state$.getValue(), ...update })\n }\n\n get locks$() {\n return this.state$.pipe(map(({ locks }) => locks))\n }\n\n get state() {\n return this.state$.getValue()\n }\n\n get isUnlocked$() {\n return this.locks$.pipe(\n map((locks) => locks <= 0),\n distinctUntilChanged(),\n shareReplay(),\n )\n }\n\n get overTTL$() {\n return this.isUnlocked$.pipe(\n switchMap((isUnlocked) =>\n !isUnlocked\n ? NEVER\n : this.cleanArchiveAfter === Infinity\n ? NEVER\n : timer(this.cleanArchiveAfter),\n ),\n )\n }\n}\n\nexport const createArchiveLoader = ({\n getArchive,\n cleanArchiveAfter = 5 * 60 * 1000, // 5mn\n}: {\n getArchive: (key: string) => Promise<Archive>\n cleanArchiveAfter?: number\n}) => {\n const loadSubject = new Subject<string>()\n const destroySubject = new Subject<void>()\n const purgeSubject = new Subject<void>()\n const archives: Record<string, ArchiveEntry> = {}\n\n const loadArchive$ = loadSubject.pipe(\n mergeMap((key) => {\n const archiveEntry = archives[key]\n\n if (!archiveEntry || archiveEntry.state.status !== \"idle\") return EMPTY\n\n let isClosed = false\n\n const cleanupArchive = (key: string) => {\n Report.debug(`Cleaning up archive with key: ${key}`)\n\n const entry = archives[key]\n\n delete archives[key]\n\n if (!isClosed) {\n entry?.state.archive?.close()\n isClosed = true\n }\n }\n\n archiveEntry.update({\n status: \"loading\",\n })\n\n const locks$ = archiveEntry.locks$\n const isUnlocked$ = archiveEntry.isUnlocked$\n\n const newAccess$ = locks$.pipe(\n pairwise(),\n filter(([prev, curent]) => curent > prev),\n startWith(true),\n )\n\n const archive$ = from(getArchive(key))\n\n return archive$.pipe(\n tap((archive) => {\n archiveEntry.update({\n archive,\n status: \"success\",\n })\n }),\n catchError((error) => {\n cleanupArchive(key)\n\n archiveEntry.update({\n status: \"error\",\n error,\n })\n\n return EMPTY\n }),\n switchMap(() => {\n const readyForPurge$ = newAccess$.pipe(\n switchMap(() => purgeSubject),\n switchMap(() => isUnlocked$),\n filter((isUnlocked) => isUnlocked),\n )\n\n const toCleanup$ = merge(readyForPurge$, archiveEntry.overTTL$)\n\n return toCleanup$.pipe(\n first(),\n tap(() => {\n cleanupArchive(key)\n }),\n )\n }),\n )\n }),\n takeUntil(destroySubject),\n )\n\n const access = (key: string) => {\n let releaseCalled = false\n\n const archiveEntry = archives[key] ?? new ArchiveEntry(cleanArchiveAfter)\n\n archives[key] = archiveEntry\n\n archiveEntry.update({\n locks: archiveEntry.state.locks + 1,\n })\n\n const release = () => {\n if (releaseCalled) return\n\n releaseCalled = true\n\n archiveEntry.update({\n locks: archiveEntry.state.locks - 1,\n })\n }\n\n loadSubject.next(key)\n\n const archive$ = archiveEntry.state$.pipe(\n map(({ archive }) => archive),\n filter((archive) => !!archive),\n )\n\n const error$ = archiveEntry.state$.pipe(\n tap(({ error }) => {\n if (error) {\n throw error\n }\n }),\n ignoreElements(),\n )\n\n return merge(archive$, error$).pipe(\n first(),\n map((archive) => ({ archive, release })),\n catchError((error) => {\n release()\n\n throw error\n }),\n )\n }\n\n /**\n * Will purge immediatly archives as soon as they are released\n */\n const purge = () => {\n purgeSubject.next()\n }\n\n loadArchive$.subscribe()\n\n return {\n access,\n purge,\n _archives: archives,\n }\n}\n","const parseByteRangePart = (value: string) => {\n if (!value) {\n return {\n valid: true,\n value: undefined,\n }\n }\n\n if (!/^\\d+$/.test(value)) {\n return {\n valid: false,\n value: undefined,\n }\n }\n\n return {\n valid: true,\n value: Number.parseInt(value, 10),\n }\n}\n\nconst parseSingleByteRange = (rangeHeader: string) => {\n if (!rangeHeader.toLowerCase().startsWith(\"bytes=\")) {\n return {\n kind: \"missing\" as const,\n }\n }\n\n const rangeValue = rangeHeader.slice(\"bytes=\".length).trim()\n\n if (!rangeValue) {\n return {\n kind: \"invalid\" as const,\n }\n }\n\n if (rangeValue.includes(\",\")) {\n return {\n kind: \"multi\" as const,\n }\n }\n\n const matches = /^(\\d*)-(\\d*)$/.exec(rangeValue)\n\n if (!matches) {\n return {\n kind: \"invalid\" as const,\n }\n }\n\n const [, rawStart = \"\", rawEnd = \"\"] = matches\n const parsedStart = parseByteRangePart(rawStart.trim())\n const parsedEnd = parseByteRangePart(rawEnd.trim())\n\n if (!parsedStart.valid || !parsedEnd.valid) {\n return {\n kind: \"invalid\" as const,\n }\n }\n\n return {\n kind: \"single\" as const,\n start: parsedStart.value,\n end: parsedEnd.value,\n }\n}\n\n/**\n * Normalize the body into a sliceable form that avoids passing a\n * `Blob.slice()` result directly to `new Response()`. Some Node.js\n * versions do not recognise sliced Blobs as valid `BodyInit`,\n * producing `\"[object Blob]\"` instead of the actual content.\n */\nconst materializeForSlicing = (body: Blob | string) => {\n if (body instanceof Blob) {\n return {\n size: body.size,\n slice: (start: number, endExclusive: number) => {\n const part = body.slice(start, endExclusive)\n return { content: part, length: part.size }\n },\n }\n }\n\n const bytes = new TextEncoder().encode(body)\n\n return {\n size: bytes.byteLength,\n slice: (start: number, endExclusive: number) => {\n const part = bytes.slice(start, endExclusive)\n return { content: part, length: part.byteLength }\n },\n }\n}\n\nexport const createRangeResponse = ({\n body,\n contentType,\n rangeHeader,\n}: {\n body: Blob | string\n contentType?: string\n rangeHeader?: string | null\n}) => {\n const headers = new Headers()\n\n if (contentType) {\n headers.set(\"Content-Type\", contentType)\n }\n\n headers.set(\"Accept-Ranges\", \"bytes\")\n\n if (!rangeHeader) {\n if (body instanceof Blob) {\n headers.set(\"Content-Length\", String(body.size))\n }\n\n return new Response(body, {\n status: 200,\n headers,\n })\n }\n\n const parsedRange = parseSingleByteRange(rangeHeader)\n\n if (parsedRange.kind === \"missing\" || parsedRange.kind === \"multi\") {\n if (body instanceof Blob) {\n headers.set(\"Content-Length\", String(body.size))\n }\n\n return new Response(body, {\n status: 200,\n headers,\n })\n }\n\n const rangeBody = materializeForSlicing(body)\n const size = rangeBody.size\n\n if (parsedRange.kind === \"invalid\") {\n return new Response(null, {\n status: 416,\n headers: {\n \"Content-Range\": `bytes */${size}`,\n },\n })\n }\n\n let start = parsedRange.start\n let end = parsedRange.end\n\n if (start === undefined && end === undefined) {\n return new Response(null, {\n status: 416,\n headers: {\n \"Content-Range\": `bytes */${size}`,\n },\n })\n }\n\n if (start === undefined) {\n const suffixLength = Math.min(end ?? 0, size)\n start = Math.max(0, size - suffixLength)\n end = size - 1\n } else if (end === undefined || end >= size) {\n end = size - 1\n }\n\n if (start < 0 || end < 0 || start >= size || end >= size || start > end) {\n return new Response(null, {\n status: 416,\n headers: {\n \"Content-Range\": `bytes */${size}`,\n },\n })\n }\n\n const partial = rangeBody.slice(start, end + 1)\n\n headers.set(\"Content-Length\", String(partial.length))\n headers.set(\"Content-Range\", `bytes ${start}-${end}/${size}`)\n\n return new Response(partial.content, {\n status: 206,\n headers,\n })\n}\n","import type { Manifest } from \"@prose-reader/shared\"\nimport {\n catchError,\n finalize,\n from,\n lastValueFrom,\n map,\n mergeMap,\n type Observable,\n of,\n switchMap,\n} from \"rxjs\"\nimport { createArchiveLoader } from \"./archives/archiveLoader\"\nimport type { Archive } from \"./archives/types\"\nimport { generateManifestFromArchive } from \"./generators/manifest\"\nimport { generateResourceFromArchive } from \"./generators/resources\"\nimport type { StreamerHooks } from \"./hooks\"\nimport { createRangeResponse } from \"./utils/createRangeResponse\"\n\ntype OnError = (error: unknown) => Response\ntype OnManifestSuccess = (params: {\n manifest: Manifest\n archive: Archive\n}) => Observable<Manifest> | Promise<Manifest>\ntype WithArchiveResponse = (\n archive: Archive,\n) => Observable<Response> | Promise<Response>\n\nconst fileProtocol = `file://`\nconst httpProtocolPattern = /^https?:\\/\\//\n\nconst decodeResourcePath = (resourcePath: string) => {\n try {\n return decodeURIComponent(resourcePath)\n } catch {\n return resourcePath\n }\n}\n\nconst stripFileProtocol = (resourcePath: string) =>\n resourcePath.startsWith(fileProtocol)\n ? resourcePath.slice(fileProtocol.length)\n : resourcePath\n\nconst normalizeResourcePath = (resourcePath: string) => {\n const resourcePathWithoutFileProtocol = stripFileProtocol(resourcePath)\n\n if (httpProtocolPattern.test(resourcePathWithoutFileProtocol)) {\n return resourcePathWithoutFileProtocol\n }\n\n return stripFileProtocol(decodeResourcePath(resourcePathWithoutFileProtocol))\n}\n\nexport class Streamer {\n protected archiveLoader: ReturnType<typeof createArchiveLoader>\n protected onError: OnError = (error) => {\n console.error(error)\n\n return new Response(String(error), { status: 500 })\n }\n protected hooks: StreamerHooks\n protected onManifestSuccess: OnManifestSuccess\n protected lastAccessedKey: string | undefined\n\n constructor({\n hooks,\n onError,\n onManifestSuccess,\n ...rest\n }: Parameters<typeof createArchiveLoader>[0] & {\n hooks?: StreamerHooks\n onError?: OnError\n onManifestSuccess?: OnManifestSuccess\n }) {\n this.archiveLoader = createArchiveLoader(rest)\n this.hooks = hooks ?? {}\n\n this.onManifestSuccess =\n onManifestSuccess ?? (({ manifest }) => Promise.resolve(manifest))\n this.onError = onError ?? this.onError\n }\n\n public prune() {\n this.archiveLoader.purge()\n }\n\n public accessArchive(key: string) {\n if (this.lastAccessedKey !== undefined && this.lastAccessedKey !== key) {\n this.archiveLoader.purge()\n }\n\n this.lastAccessedKey = key\n\n return this.archiveLoader.access(key)\n }\n\n public accessArchiveWithoutLock(key: string) {\n return this.accessArchive(key).pipe(\n map(({ archive, release }) => {\n release()\n\n return archive\n }),\n )\n }\n\n protected withArchiveResponse({\n key,\n getResponse,\n }: {\n key: string\n getResponse: WithArchiveResponse\n }) {\n const response$ = this.accessArchive(key).pipe(\n mergeMap(({ archive, release }) =>\n from(getResponse(archive)).pipe(\n finalize(() => {\n release()\n }),\n ),\n ),\n catchError((error) => {\n return of(this.onError(error))\n }),\n )\n\n return lastValueFrom(response$)\n }\n\n public fetchManifest({ key, baseUrl }: { key: string; baseUrl?: string }) {\n return this.withArchiveResponse({\n key,\n getResponse: (archive) => {\n const manifest$ = from(\n generateManifestFromArchive(archive, {\n baseUrl,\n hooks: this.hooks.manifest,\n }),\n )\n\n return manifest$.pipe(\n switchMap((manifest) =>\n from(this.onManifestSuccess({ manifest, archive })),\n ),\n map(\n (manifest) =>\n new Response(JSON.stringify(manifest satisfies Manifest), {\n status: 200,\n }),\n ),\n )\n },\n })\n }\n\n public fetchResource({\n key,\n resourcePath,\n request,\n }: {\n key: string\n resourcePath: string\n request?: Request\n }) {\n return this.withArchiveResponse({\n key,\n getResponse: (archive) => {\n /**\n * We commonly use file:// for manifest without baseUrl. This ensure we\n * have valid URL for the reader while flagging them as local.\n *\n * However, we obviously don't have the file:// prefix on the archive.\n */\n const cleanedResourcePath = normalizeResourcePath(resourcePath)\n\n /**\n * Reader hot path: one `generateResourceFromArchive` per request. For\n * EPUBs, `defaultHook` re-reads and re-parses the package OPF each time\n * (no `archiveOpf` threading yet). See `generateResourceFromArchive` JSDoc.\n */\n const resource$ = from(\n generateResourceFromArchive(archive, cleanedResourcePath, {\n hooks: this.hooks.resource,\n }),\n )\n\n return resource$.pipe(\n map((resource) =>\n createRangeResponse({\n body: resource.body ?? \"\",\n contentType: resource.params.contentType,\n rangeHeader: request?.headers.get(\"range\"),\n }),\n ),\n )\n },\n })\n }\n}\n","import { Streamer } from \"./Streamer\"\nimport { removeTrailingSlash } from \"./utils/uri\"\n\ntype ConflictFreeWebWorkerFetchEvent = {\n readonly request: Request\n /** [MDN Reference](https://developer.mozilla.org/docs/Web/API/FetchEvent/respondWith) */\n respondWith(r: Response | PromiseLike<Response>): void\n}\n\nexport class ServiceWorkerStreamer extends Streamer {\n protected getUriInfo: (event: ConflictFreeWebWorkerFetchEvent) =>\n | {\n baseUrl: string\n }\n | undefined\n\n constructor({\n getUriInfo,\n ...rest\n }: ConstructorParameters<typeof Streamer>[0] & {\n getUriInfo: (event: ConflictFreeWebWorkerFetchEvent) =>\n | {\n baseUrl: string\n }\n | undefined\n }) {\n super(rest)\n\n this.getUriInfo = getUriInfo\n this.fetchEventListener = this.fetchEventListener.bind(this)\n }\n\n fetchEventListener(event: ConflictFreeWebWorkerFetchEvent) {\n try {\n const uriInfo = this.getUriInfo(event)\n\n if (!uriInfo) return\n\n const baseUrl = removeTrailingSlash(uriInfo.baseUrl)\n const streamerPath = event.request.url.substring(\n baseUrl.length + `/`.length,\n )\n const [key = ``] = streamerPath.split(\"/\")\n const resourcePath = removeTrailingSlash(\n streamerPath.substring(key.length + `/`.length),\n )\n\n if (streamerPath.endsWith(`/manifest`)) {\n event.respondWith(\n this.fetchManifest({ key, baseUrl: `${baseUrl}/${key}/` }),\n )\n } else {\n event.respondWith(\n this.fetchResource({\n key,\n resourcePath,\n request: event.request,\n }),\n )\n }\n } catch (e) {\n event.respondWith(new Response(String(e), { status: 500 }))\n }\n }\n}\n"],"names":["sortByTitleComparator","a","b","alist","blist","i","len","getUriBasename","uri","removeTrailingSlash","getUriBasePath","lastSlashIndex","createArchiveFromArrayBufferList","list","orderByAlpha","name","files","file","size","basename","detectMimeTypeFromName","data","Report","SharedReport","printTree","paths","tree","path","parts","node","part","render","indent","key","arr","isLast","prefix","nextIndent","value","createArchiveFromJszip","jszip","archive","folderStructureStr","createArchiveFromLibArchive","libArchive","objArray","item","createArchiveFromText","content","mimeType","direction","txtOpfContent","XML_SAFE_ID_PATTERN","XML_SAFE_ID_START_PATTERN","XML_RESERVED_ID_START_PATTERN","INVALID_XML_SAFE_ID_CHARACTER_PATTERN","XML_SAFE_ID_START_FALLBACK","trimIdSeparators","normalizeXmlSafeIdValue","createXmlSafeId","normalized","valueWithValidStart","createUniqueXmlSafeId","usedIds","id","index","nextId","createXmlSafeIdFactory","createArchiveFromUrls","urls","options","createSafeId","resources","url","opfFileData","mediaType","escapeXmlAttributeValue","filesFromUrl","configure","enableReport","getArchiveOpfInfo","readArchiveOpf","opsFile","basePath","raw","parseOpf","appleDisplayOptionsBasename","APPLE_IBOOKS_DISPLAY_OPTIONS_FILENAME","apple","manifest","infoFile","parsed","parseAppleDisplayOptionsXml","renditionLayout","resolveArchiveMetadata","e","comicInfoFilenameLower","COMIC_INFO_FILENAME","comicInfo","comicInfoFile","manifestWithoutComicInfo","_","items","parseComicInfo","resolved","createManifestResourceHref","baseUrl","resourcePath","hrefBaseUrl","defaultHook","filesWithIds","dir","getSpineItemFilesFromArchive","archiveOpf","opf","opfBasePath","spineRows","getItemsFromDoc","manifestItems","getBaseUrlForHref","el","href","manifestRenditionFlow","v","manifestRenditionSpread","manifestGuideType","epubHook","publisherRenditionLayout","packageRenditionLayout","title","readingDirection","totalSize","guideRefs","guide","elm","type","row","itemSize","hrefBaseUri","hasDocMetaViewport","doc","metaElm","allFilesHaveViewportMeta","result","current","isXmlBasedMimeType","XmlDocument","epubOptimizerHook","extractKoboInformationFromArchive","KOBO_DISPLAY_OPTIONS_FILENAME","layout","parseKoboXml","kobo","koboMeta","hasOpfExtension","isArchiveEpub","nonEpub","spineItem","archiveItem","parseContentType","isMediaContentMimeType","getXmlElementInnerText","child","XmlTextNode","XmlElement","manifestItemIsNavDocument","tokenizeXmlSpaceSeparatedList","extractNavChapter","li","chp","contentNode","urlJoin","sublistNode","children","buildTOCFromNav","toc","navDataChildren","parseTocFromNavPath","navItem","tocFile","tocFileBasePath","mapNcxChapter","point","src","out","pt","buildTOCFromNCX","ncxData","rootTagName","parseTocFromNcx","ncxId","ncxItem","ncxPath","parseToc","tocFromNav","tocFromNcx","normalizeFilenameAsTitle","filename","buildAudiobookToc","record","r","buildFolderFallbackToc","filesSortedByAlpha","combineWith","folder","subFolders","foundEntry","entry","nextFolderCursor","nextSubFolders","acc","folders","firstFolder","restFolders","resolveTocFromArchive","audiobookToc","tocHook","normalizeBaseUrl","generateManifestFromArchive","hooks","normalizedBaseUrl","createExternalHooks","hookFactories","hook","contentHooks","spineHooks","presentationHooks","navigationHooks","manifestHooks","baseManifestPromise","gen","manifestStr","hasCalibreCoverMeta","getBuggyCoverSvg","fixBuggyCover","resource","bodyToParse","opfXmlDoc","buggySvg","calibreFixHook","cssFixHook","newBody","getMetadata","getContentTypeFromExtension","metadata","invalidSelfClosingTags","selfClosingTagsFixHook","tagPattern","fixedBody","tagName","attributes","generateResourceFromArchive","defaultResource","resourceHooks","ArchiveEntry","cleanArchiveAfter","BehaviorSubject","update","map","locks","distinctUntilChanged","shareReplay","switchMap","isUnlocked","NEVER","timer","createArchiveLoader","getArchive","loadSubject","Subject","destroySubject","purgeSubject","archives","loadArchive$","mergeMap","archiveEntry","EMPTY","isClosed","cleanupArchive","locks$","isUnlocked$","newAccess$","pairwise","filter","prev","curent","startWith","from","tap","catchError","error","readyForPurge$","merge","first","takeUntil","access","releaseCalled","release","archive$","error$","ignoreElements","purge","parseByteRangePart","parseSingleByteRange","rangeHeader","rangeValue","matches","rawStart","rawEnd","parsedStart","parsedEnd","materializeForSlicing","body","start","endExclusive","bytes","createRangeResponse","contentType","headers","parsedRange","rangeBody","end","suffixLength","partial","fileProtocol","httpProtocolPattern","decodeResourcePath","stripFileProtocol","normalizeResourcePath","resourcePathWithoutFileProtocol","Streamer","onError","onManifestSuccess","rest","getResponse","response$","finalize","of","lastValueFrom","request","cleanedResourcePath","ServiceWorkerStreamer","getUriInfo","event","uriInfo","streamerPath"],"mappings":"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"}
|