@expofp/loader 1.0.61 → 1.0.65

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.
Files changed (35) hide show
  1. package/dist/bundle/bundle.js +125 -110
  2. package/dist/bundle/bundle.js.map +1 -1
  3. package/dist/bundle/{downloadZip-CohB2rQN.js → downloadOfflineZip-B8tTyZYe.js} +1001 -995
  4. package/dist/bundle/downloadOfflineZip-B8tTyZYe.js.map +1 -0
  5. package/dist/bundle/{makeOffline-DuRXI_Dd.js → makeOffline-DH6wJEem.js} +37 -37
  6. package/dist/bundle/makeOffline-DH6wJEem.js.map +1 -0
  7. package/dist/bundle/makeOfflineBundle-C-xleVMN.js +58 -0
  8. package/dist/bundle/makeOfflineBundle-C-xleVMN.js.map +1 -0
  9. package/dist/bundle/saveOfflineZip.browser-BTQeRUY_.js +7 -0
  10. package/dist/bundle/saveOfflineZip.browser-BTQeRUY_.js.map +1 -0
  11. package/dist/esm/importJson.js +4 -0
  12. package/dist/esm/index.d.ts +1 -1
  13. package/dist/esm/index.js +1 -1
  14. package/dist/esm/offline/downloadOfflineZip.d.ts +4 -0
  15. package/dist/esm/offline/downloadOfflineZip.js +15 -0
  16. package/dist/esm/offline/generateZip.d.ts +4 -0
  17. package/dist/esm/offline/{downloadZip.js → generateZip.js} +9 -8
  18. package/dist/esm/offline/index.d.ts +9 -1
  19. package/dist/esm/offline/index.js +12 -4
  20. package/dist/esm/offline/makeOffline.js +3 -2
  21. package/dist/esm/offline/makeOfflineBundle.js +19 -8
  22. package/dist/esm/offline/saveOfflineZip.browser.d.ts +1 -0
  23. package/dist/esm/offline/saveOfflineZip.browser.js +3 -0
  24. package/dist/esm/offline/saveOfflineZip.d.ts +1 -0
  25. package/dist/esm/offline/saveOfflineZip.js +16 -0
  26. package/dist/esm/offline/tools.js +4 -4
  27. package/dist/esm/resolve.d.ts +11 -4
  28. package/dist/esm/resolve.js +6 -34
  29. package/dist/esm/resolvers/legacyAssetUrlsResolver.js +12 -1
  30. package/package.json +10 -2
  31. package/dist/bundle/downloadZip-CohB2rQN.js.map +0 -1
  32. package/dist/bundle/makeOffline-DuRXI_Dd.js.map +0 -1
  33. package/dist/bundle/makeOfflineBundle-XjvmvDQE.js +0 -58
  34. package/dist/bundle/makeOfflineBundle-XjvmvDQE.js.map +0 -1
  35. package/dist/esm/offline/downloadZip.d.ts +0 -1
@@ -1,15 +1,15 @@
1
- import { l as w, r as x, a as v, b as R, d, e as b, p as O } from "./bundle.js";
1
+ import { d as w, r as O, b as R, e as S, f as b, h as x, p as E } from "./bundle.js";
2
2
  function u(t, n) {
3
3
  let r = n >>> 0;
4
4
  for (let a = 0; a < t.length; a++)
5
5
  r ^= t.charCodeAt(a), r = Math.imul(r, 16777619);
6
6
  return (r >>> 0).toString(16).padStart(8, "0");
7
7
  }
8
- function S(t) {
8
+ function j(t) {
9
9
  return u(t, 2166136261) + // your original seed
10
10
  u(t, 569420461) + u(t, 461845907) + u(t, 2246822507);
11
11
  }
12
- const m = /* @__PURE__ */ new Map(), p = /* @__PURE__ */ new Map(), E = /* @__PURE__ */ new Set([
12
+ const m = /* @__PURE__ */ new Map(), p = /* @__PURE__ */ new Map(), y = /* @__PURE__ */ new Set([
13
13
  "con",
14
14
  "prn",
15
15
  "aux",
@@ -33,14 +33,14 @@ const m = /* @__PURE__ */ new Map(), p = /* @__PURE__ */ new Map(), E = /* @__PU
33
33
  "lpt8",
34
34
  "lpt9"
35
35
  ]);
36
- function j(t) {
36
+ function I(t) {
37
37
  let n = t.normalize("NFKD").replace(/[^\p{Letter}\p{Number}]+/gu, "-").replace(/-+/g, "-").replace(/^-|-$/g, "").toLowerCase();
38
- return n = n.replace(/[<>:"/\\|?*]/g, ""), n = n.replace(/[. ]+$/g, ""), n || (n = "file"), E.has(n) && (n += "-file"), n;
38
+ return n = n.replace(/[<>:"/\\|?*]/g, ""), n = n.replace(/[. ]+$/g, ""), n || (n = "file"), y.has(n) && (n += "-file"), n;
39
39
  }
40
40
  function h(t) {
41
41
  const n = m.get(t);
42
42
  if (n) return n;
43
- const r = j(t);
43
+ const r = I(t);
44
44
  let a = r, o = 2;
45
45
  for (; ; ) {
46
46
  const e = p.get(a);
@@ -51,11 +51,12 @@ function h(t) {
51
51
  a = `${r}-${o++}`;
52
52
  }
53
53
  }
54
- function I(t, n = "") {
54
+ const d = w("efp:loader:offline");
55
+ function M(t, n = "") {
55
56
  let r = h(t);
56
57
  return n && (r = `${h(n)}$${r}`), r.endsWith("/") ? r += "index.json" : r.endsWith(".json") || (r += ".json"), "./" + r;
57
58
  }
58
- function M(t, n = "") {
59
+ function P(t, n = "") {
59
60
  try {
60
61
  new URL(t);
61
62
  } catch {
@@ -67,27 +68,26 @@ function M(t, n = "") {
67
68
  throw new Error(`Cannot make target path from URL without file extension: ${t}`);
68
69
  const s = e.substring(e.lastIndexOf("."));
69
70
  let l = e.substring(0, e.lastIndexOf("."));
70
- const f = l.match(/[^a-zA-Z0-9\-._\/]/g);
71
- if (f) {
72
- const c = h(l);
73
- console.warn(
74
- `Warning: pathname contains invalid filesystem characters (${[
75
- ...new Set(f)
76
- ].join(
71
+ const c = l.match(/[^a-zA-Z0-9\-._\/]/g);
72
+ if (c) {
73
+ const f = h(l);
74
+ d(
75
+ `Pathname contains invalid filesystem characters (${[...new Set(c)].join(
77
76
  ", "
78
- )}), slugifying it: ${l}${s} => ${c}${s}`
79
- ), l = c;
77
+ )}), slugifying it: ${l}${s} => ${f}${s}`
78
+ ), l = f;
80
79
  }
81
- if (e = l + s, e.length > 120 && (console.warn(
82
- `Warning: pathname is too long (${e.length} characters), truncating to 150 characters: ${e}`
80
+ if (e = l + s, e.length > 120 && (d(
81
+ `Pathname is too long (${e.length} characters), truncating to 150 characters: ${e}`
83
82
  ), e = e.substring(0, 120 - s.length) + s), i) {
84
- const c = S(i), g = e.lastIndexOf(".");
85
- g !== -1 ? e = `${e.slice(0, g)}.${c}${e.slice(g)}` : e = `${e}${c}`;
83
+ const f = j(i), g = e.lastIndexOf(".");
84
+ g !== -1 ? e = `${e.slice(0, g)}.${f}${e.slice(g)}` : e = `${e}${f}`;
86
85
  }
87
86
  return `./${n}${o}${e}`;
88
87
  }
89
- async function F(t) {
90
- w("makeOffline", t);
88
+ const v = w("efp:loader:makeOffline");
89
+ async function C(t) {
90
+ v("makeOffline", t);
91
91
  const n = await $(t);
92
92
  let r = !1, a;
93
93
  async function* o() {
@@ -102,7 +102,7 @@ async function F(t) {
102
102
  };
103
103
  }
104
104
  async function* $(t) {
105
- w("makeOfflineInternal", t);
105
+ v("makeOfflineInternal", t);
106
106
  const n = {
107
107
  refCache: /* @__PURE__ */ new Map(),
108
108
  forceFetch: !0,
@@ -112,7 +112,7 @@ async function* $(t) {
112
112
  async function* a(o, e) {
113
113
  if (!(typeof o[e] != "object" || o[e] === null)) {
114
114
  if ("$ref" in o[e]) {
115
- const i = x.filter((c) => v(c, o[e].$ref));
115
+ const i = O.filter((f) => R(f, o[e].$ref));
116
116
  if (i.length === 0)
117
117
  throw new Error(`No resolver found for ref: ${o[e].$ref}`);
118
118
  if (i.length > 1)
@@ -121,17 +121,17 @@ async function* $(t) {
121
121
  let l;
122
122
  switch (s.offlineMethod) {
123
123
  case "localizeRef":
124
- l = W;
124
+ l = U;
125
125
  break;
126
126
  case "resolveRef":
127
- l = y;
127
+ l = F;
128
128
  break;
129
129
  default:
130
130
  throw new Error(`Unknown offlineMethod: ${s.offlineMethod}`);
131
131
  }
132
- const f = yield* l(s, o[e].$ref, n);
133
- if (Object.isFrozen(o[e])) throw new Error("Unexpected frozen node during makeOffline");
134
- o[e] = R({ $ref: f }, o[e]);
132
+ const c = yield* l(s, o[e].$ref, n);
133
+ if (Object.isFrozen(o)) throw new Error("Unexpected frozen node during makeOffline");
134
+ o[e] = S({ $ref: c }, o[e]);
135
135
  }
136
136
  if (Array.isArray(o[e]))
137
137
  for (const [i] of o[e].entries())
@@ -142,17 +142,17 @@ async function* $(t) {
142
142
  }
143
143
  }
144
144
  }
145
- async function* W(t, n, r) {
146
- const a = d(await b(t, n, r));
145
+ async function* U(t, n, r) {
146
+ const a = b(await x(t, n, r));
147
147
  yield* $(a);
148
- const o = O(n), e = M(o);
148
+ const o = E(n), e = P(o);
149
149
  return yield { url: o, targetFilePath: e }, `${n.substring(0, n.length - o.length - 1)}:${e}`;
150
150
  }
151
- async function* y(t, n, r) {
152
- const a = d(await b(t, n, r)), o = I(n);
151
+ async function* F(t, n, r) {
152
+ const a = b(await x(t, n, r)), o = M(n);
153
153
  return yield { data: yield* $(a), targetFilePath: o }, o;
154
154
  }
155
155
  export {
156
- F as makeOffline
156
+ C as makeOffline
157
157
  };
158
- //# sourceMappingURL=makeOffline-DuRXI_Dd.js.map
158
+ //# sourceMappingURL=makeOffline-DH6wJEem.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"makeOffline-DH6wJEem.js","sources":["../../src/offline/hashString.ts","../../src/offline/slugify.ts","../../src/offline/tools.ts","../../src/offline/makeOffline.ts"],"sourcesContent":["// 32-bit murmur-ish hash with configurable seed\nfunction murmur32WithSeed(str: string, seed: number): string {\n let h = seed >>> 0;\n\n for (let i = 0; i < str.length; i++) {\n h ^= str.charCodeAt(i);\n h = Math.imul(h, 0x01000193); // FNV-ish prime, good mixer\n }\n\n return (h >>> 0).toString(16).padStart(8, '0'); // 8 hex chars\n}\n\n// 128-bit (MD5-width) hash: 4 × 32-bit parts concatenated\nexport function hashString(str: string): string {\n return (\n murmur32WithSeed(str, 0x811c9dc5) + // your original seed\n murmur32WithSeed(str, 0x21f0aaad) +\n murmur32WithSeed(str, 0x1b873593) +\n murmur32WithSeed(str, 0x85ebca6b)\n );\n}\n","// Module-level caches\nconst inputToSlug = new Map<string, string>();\nconst slugToInput = new Map<string, string>();\n\nconst WINDOWS_RESERVED = new Set([\n \"con\", \"prn\", \"aux\", \"nul\",\n \"com1\", \"com2\", \"com3\", \"com4\", \"com5\", \"com6\", \"com7\", \"com8\", \"com9\",\n \"lpt1\", \"lpt2\", \"lpt3\", \"lpt4\", \"lpt5\", \"lpt6\", \"lpt7\", \"lpt8\", \"lpt9\",\n]);\n\nfunction makeBaseSlug(input: string): string {\n let slug = input\n .normalize(\"NFKD\")\n // All non letters/numbers → \"-\"\n .replace(/[^\\p{Letter}\\p{Number}]+/gu, \"-\")\n // Collapse multiple \"-\"\n .replace(/-+/g, \"-\")\n // Trim \"-\" from start/end\n .replace(/^-|-$/g, \"\")\n .toLowerCase();\n\n // Strip forbidden Windows characters just in case\n slug = slug.replace(/[<>:\"/\\\\|?*]/g, \"\");\n\n // Windows forbids trailing space/period\n slug = slug.replace(/[. ]+$/g, \"\");\n\n // Fallback if everything was stripped\n if (!slug) slug = \"file\";\n\n // Avoid bare reserved device names\n if (WINDOWS_RESERVED.has(slug)) {\n slug += \"-file\";\n }\n\n return slug;\n}\n\nexport function slugifyFsUnique(input: string): string {\n // If we've seen this exact input before, return the same slug\n const existing = inputToSlug.get(input);\n if (existing) return existing;\n\n const base = makeBaseSlug(input);\n let candidate = base;\n let counter = 2;\n\n while (true) {\n const existingInput = slugToInput.get(candidate);\n\n if (!existingInput) {\n // Free slug → claim it for this input\n slugToInput.set(candidate, input);\n inputToSlug.set(input, candidate);\n return candidate;\n }\n\n if (existingInput === input) {\n // Same input somehow (super defensive)\n inputToSlug.set(input, candidate);\n return candidate;\n }\n\n // Collision: same slug already used by different input → add suffix\n candidate = `${base}-${counter++}`;\n }\n}\n\n// // Optional: to reset between runs/tests\n// export function resetSlugCache() {\n// inputToSlug.clear();\n// slugToInput.clear();\n// }","import debug from 'debug';\nimport { hashString } from './hashString';\nimport { slugifyFsUnique } from './slugify';\n\nconst log = debug('efp:loader:offline');\n\nexport function makeUniqueJsonTargetPathFromString(str: string, namespace: string = ''): string {\n // const hash = hashString(str);\n let result = slugifyFsUnique(str); // + '-' + hash;\n if (namespace) {\n result = `${slugifyFsUnique(namespace)}$${result}`;\n }\n\n if (result.endsWith('/')) {\n result += 'index.json';\n } else if (!result.endsWith('.json')) {\n result += '.json';\n }\n return './' + result;\n // handle directory case\n}\n\nexport function makeTargetPathFromUrl(url: string, prefix: string = ''): string {\n // https://example.com/dir1/dir2/a.js => \"{prefix}{origin-slug}/dir1/dir2/a.js\";\n // https://example.com/dir1/dir2/a.js?params => \"{prefix}{origin-slug}/dir1/dir2/a{paramsmd5hash}.js\";\n // use slugify.ts\n\n try {\n new URL(url);\n } catch {\n debugger;\n }\n const urlObj = new URL(url);\n const origin = `${urlObj.protocol}//${urlObj.host}`;\n const originSlug = slugifyFsUnique(origin);\n\n let pathname = urlObj.pathname;\n let search = urlObj.search;\n\n // if path doesn't end with extension, throw\n if (!pathname.match(/\\.[^\\/]+$/)) {\n throw new Error(`Cannot make target path from URL without file extension: ${url}`);\n }\n\n const extension = pathname.substring(pathname.lastIndexOf('.'));\n let pathnameWithoutExtension = pathname.substring(0, pathname.lastIndexOf('.'));\n\n // check the pathname contains only valid fs characters\n const invalidPathnameChars = pathnameWithoutExtension.match(/[^a-zA-Z0-9\\-._\\/]/g);\n if (invalidPathnameChars) {\n const fixedPathnameWithoutExtension = slugifyFsUnique(pathnameWithoutExtension);\n log(\n `Pathname contains invalid filesystem characters (${[...new Set(invalidPathnameChars)].join(\n ', '\n )}), slugifying it: ${pathnameWithoutExtension}${extension} => ${fixedPathnameWithoutExtension}${extension}`\n );\n pathnameWithoutExtension = fixedPathnameWithoutExtension;\n }\n\n pathname = pathnameWithoutExtension + extension;\n\n if (pathname.length > 120) {\n log(\n `Pathname is too long (${pathname.length} characters), truncating to 150 characters: ${pathname}`\n );\n pathname = pathname.substring(0, 120 - extension.length) + extension;\n }\n\n if (search) {\n // create a hash from search params\n const hash = hashString(search);\n const dotIndex = pathname.lastIndexOf('.');\n if (dotIndex !== -1) {\n pathname = `${pathname.slice(0, dotIndex)}.${hash}${pathname.slice(dotIndex)}`;\n } else {\n pathname = `${pathname}${hash}`;\n }\n }\n\n // // handle directory case\n // if (pathname.endsWith('/')) {\n // pathname += '__index.json';\n // }\n\n return `./${prefix}${originSlug}${pathname}`;\n}\n","import debug from 'debug';\nimport {\n canResolve,\n parseRefValue,\n resolverResolve,\n type ResolveContextInternal,\n} from '../resolve';\nimport { resolvers } from '../resolvers';\nimport { createMergedObjectWithOverridenNonRefProps, deepClone } from '../shared';\nimport type { LocalFile, MakeOfflineResult, ManifestData, Resolver } from '../types';\nimport { makeTargetPathFromUrl, makeUniqueJsonTargetPathFromString } from './tools';\n\nconst log = debug('efp:loader:makeOffline');\n\nexport async function makeOffline(manifest: any): Promise<MakeOfflineResult> {\n log('makeOffline', manifest);\n // if (typeof manifest !== 'object' || manifest === null) {\n // throw new Error('Manifest must be an object');\n // }\n\n const res = await makeOfflineInternal(manifest);\n let done = false;\n let resultManifest: any;\n async function* files() {\n resultManifest = yield* res;\n done = true;\n }\n\n return {\n get manifest(): ManifestData {\n if (!done) throw new Error('Iterate over files before getting manifest');\n return resultManifest;\n },\n files: files() as AsyncGenerator<LocalFile, void, void>,\n };\n}\n\nasync function* makeOfflineInternal(manifest: any): AsyncGenerator<LocalFile, ManifestData, void> {\n log('makeOfflineInternal', manifest);\n const resolveContext: ResolveContextInternal = {\n refCache: new Map<string, Promise<any>>(),\n forceFetch: true,\n signal: null,\n };\n const parent = { manifest };\n yield* walk(parent, 'manifest');\n return parent.manifest;\n\n async function* walk(node: any, key: string | number): AsyncGenerator<LocalFile> {\n // console.log('walk', node, key, node[key]);\n // let node[key] = node[key];\n if (typeof node[key] !== 'object' || node[key] === null) {\n return;\n }\n // let resolversToUse1 = resolvers.filter((r) => r.canResolve(node[key].$ref));\n // const usedResolvers = new Set<Resolver>();\n // do {\n if ('$ref' in node[key]) {\n const resolversToUse = resolvers.filter((r) => canResolve(r, node[key].$ref));\n if (resolversToUse.length === 0) {\n throw new Error(`No resolver found for ref: ${node[key].$ref}`);\n }\n if (resolversToUse.length > 1) {\n throw new Error(`Multiple resolvers can make offline ref: ${node[key].$ref}`);\n }\n\n const resolver = resolversToUse[0];\n\n let func: typeof offlineLocalizeRef;\n switch (resolver.offlineMethod) {\n case 'localizeRef':\n func = offlineLocalizeRef;\n break;\n case 'resolveRef':\n func = offlineResolveRef;\n break;\n default:\n throw new Error(`Unknown offlineMethod: ${resolver.offlineMethod}`);\n }\n\n const mergeRef = yield* func(resolver, node[key].$ref, resolveContext);\n\n if (Object.isFrozen(node)) throw new Error('Unexpected frozen node during makeOffline');\n\n node[key] = createMergedObjectWithOverridenNonRefProps({ $ref: mergeRef }, node[key]);\n }\n\n // recurse\n if (Array.isArray(node[key])) {\n for (const [index] of node[key].entries()) {\n yield* walk(node[key], index);\n }\n } else if (typeof node[key] === 'object' && node[key] !== null) {\n for (const key1 of Object.keys(node[key])) {\n // debugger;\n yield* walk(node[key], key1);\n }\n }\n }\n}\n\n// function canResolve(resolver: Resolver, ref: string): boolean {\n// if (resolver.canResolve) {\n// return resolver.canResolve(ref);\n// }\n// if (resolver.schema) {\n// return canResolveRefSchema(ref, resolver.schema);\n// }\n// throw new Error('Resolver is missing canResolve method and schema property');\n// }\n\nasync function* offlineLocalizeRef(\n resolver: Resolver,\n ref: string,\n context: ResolveContextInternal\n): AsyncGenerator<LocalFile, string, void> {\n const refData = deepClone(await resolverResolve(resolver, ref, context));\n // emit assets\n yield* makeOfflineInternal(refData);\n const url = parseRefValue(ref);\n const targetFilePath = makeTargetPathFromUrl(url);\n // TODO: handle CSS and other text files that may reference other assets\n yield { url, targetFilePath };\n const schema = ref.substring(0, ref.length - url.length - 1);\n return `${schema}:${targetFilePath}`;\n}\n\nasync function* offlineResolveRef(\n resolver: Resolver,\n ref: string,\n context: ResolveContextInternal\n): AsyncGenerator<LocalFile, string, void> {\n const refData = deepClone(await resolverResolve(resolver, ref, context));\n const targetFilePath = makeUniqueJsonTargetPathFromString(ref);\n const data = yield* makeOfflineInternal(refData);\n yield { data, targetFilePath };\n return targetFilePath;\n}\n\n// export function createMakeOfflineResult(\n// manifest: any,\n// walk: (node: any, key: string | number) => AsyncGenerator<LocalFile>\n// ): MakeOfflineResult {\n// log('createMakeOfflineResult', manifest);\n// if (typeof manifest !== 'object' || manifest === null) {\n// throw new Error('Manifest must be an object');\n// }\n// const parent = { manifest };\n// let done = false;\n// async function* files() {\n// yield* walk(parent, 'manifest');\n// done = true;\n// }\n\n// return {\n// get manifest() {\n// if (!done) {\n// throw new Error('Cannot access manifest before all files are generated');\n// }\n// return parent.manifest;\n// },\n// files: files(),\n// };\n// }\n"],"names":["murmur32WithSeed","str","seed","h","i","hashString","inputToSlug","slugToInput","WINDOWS_RESERVED","makeBaseSlug","input","slug","slugifyFsUnique","existing","base","candidate","counter","existingInput","log","debug","makeUniqueJsonTargetPathFromString","namespace","result","makeTargetPathFromUrl","url","prefix","urlObj","origin","originSlug","pathname","search","extension","pathnameWithoutExtension","invalidPathnameChars","fixedPathnameWithoutExtension","hash","dotIndex","makeOffline","manifest","res","makeOfflineInternal","done","resultManifest","files","resolveContext","parent","walk","node","key","resolversToUse","resolvers","r","canResolve","resolver","func","offlineLocalizeRef","offlineResolveRef","mergeRef","createMergedObjectWithOverridenNonRefProps","index","key1","ref","context","refData","deepClone","resolverResolve","parseRefValue","targetFilePath"],"mappings":"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"}
@@ -0,0 +1,58 @@
1
+ import { d, a as f, resolve as s } from "./bundle.js";
2
+ import { makeOffline as c } from "./makeOffline-DH6wJEem.js";
3
+ const l = d("efp:loader:makeOfflineBundle");
4
+ async function* p(e) {
5
+ l("Bundling", e), f(e);
6
+ const a = await c(e), n = /* @__PURE__ */ new Map();
7
+ for await (const t of a.files)
8
+ if ("data" in t) {
9
+ n.set(t.targetFilePath, t.data);
10
+ const i = JSON.stringify(t.data, null, 2);
11
+ l("Bundling file with data", t.targetFilePath, i), yield {
12
+ path: t.targetFilePath,
13
+ data: new TextEncoder().encode(i).buffer
14
+ };
15
+ } else if ("url" in t) {
16
+ const i = await w(t.url);
17
+ l("Bundling file from url", t.targetFilePath), yield {
18
+ path: t.targetFilePath,
19
+ data: await i.arrayBuffer()
20
+ };
21
+ }
22
+ const r = await g(a.manifest, n);
23
+ yield {
24
+ path: "./index.html",
25
+ data: new TextEncoder().encode(r).buffer
26
+ };
27
+ }
28
+ const u = 10, o = /* @__PURE__ */ new Set();
29
+ async function w(e) {
30
+ for (; o.size >= u; )
31
+ await Promise.race(o);
32
+ const a = (async () => {
33
+ l("Fetching file for zip:", e);
34
+ const n = await fetch(e);
35
+ return n.ok || (e.includes("/data/fp.svg.ViewBox.js"), console.warn(`Warning: Failed to fetch ${e}: ${n.status}`)), await n.blob();
36
+ })();
37
+ o.add(a);
38
+ try {
39
+ return await a;
40
+ } finally {
41
+ o.delete(a);
42
+ }
43
+ }
44
+ async function g(e, a) {
45
+ const n = await s(e, "/runtime/entry", { preResolvedRefs: a });
46
+ return `
47
+ <!DOCTYPE html>
48
+ <script type="module">
49
+ import { load } from ${JSON.stringify(n)};
50
+ await load(${JSON.stringify(e)});
51
+ console.info('🚀 loaded');
52
+ <\/script>
53
+ `;
54
+ }
55
+ export {
56
+ p as makeOfflineBundle
57
+ };
58
+ //# sourceMappingURL=makeOfflineBundle-C-xleVMN.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"makeOfflineBundle-C-xleVMN.js","sources":["../../src/offline/makeOfflineBundle.ts"],"sourcesContent":["import debug from 'debug';\nimport { resolve } from '..';\nimport { makeOffline } from './makeOffline';\nimport { deepFreeze } from '../shared';\n\nconst log = debug('efp:loader:makeOfflineBundle');\n\nexport async function* makeOfflineBundle(\n manifest: unknown\n): AsyncGenerator<{ path: string; data: ArrayBuffer }> {\n log('Bundling', manifest);\n deepFreeze(manifest);\n const offlineData = await makeOffline(manifest);\n const preResolvedRefs = new Map<string, any>();\n\n for await (const file of offlineData.files) {\n if ('data' in file) {\n preResolvedRefs.set(file.targetFilePath, file.data);\n const jsonString = JSON.stringify(file.data, null, 2);\n log('Bundling file with data', file.targetFilePath, jsonString);\n yield {\n path: file.targetFilePath,\n data: new TextEncoder().encode(jsonString).buffer,\n };\n } else if ('url' in file) {\n const blob = await downloadFile(file.url);\n log('Bundling file from url', file.targetFilePath);\n yield {\n path: file.targetFilePath,\n data: await blob.arrayBuffer(),\n };\n }\n }\n // yield* generateJsLoaderFiles();\n const html = await getIndexHtml(offlineData.manifest, preResolvedRefs);\n yield {\n path: './index.html',\n data: new TextEncoder().encode(html).buffer,\n };\n}\n\nconst MAX_CONCURRENT_DOWNLOADS = 10;\nconst queue = new Set<Promise<Blob>>();\n// have a queue to limit concurrent downloads\nasync function downloadFile(url: string): Promise<Blob> {\n while (queue.size >= MAX_CONCURRENT_DOWNLOADS) {\n await Promise.race(queue);\n }\n const downloadPromise = (async () => {\n log('Fetching file for zip:', url);\n const response = await fetch(url);\n if (!response.ok) {\n // Rodion's bug. May be missing file.\n if (!url.includes('/data/fp.svg.ViewBox.js')) {\n // throw new Error(`Failed to fetch ${url}: ${response.status} ${response.statusText}`);\n }\n console.warn(`Warning: Failed to fetch ${url}: ${response.status}`);\n }\n return await response.blob();\n })();\n queue.add(downloadPromise);\n try {\n return await downloadPromise;\n } finally {\n queue.delete(downloadPromise);\n }\n}\n\nasync function getIndexHtml(manifest: unknown, preResolvedRefs: Map<string, any>): Promise<string> {\n const entryPoint = await resolve(manifest, '/runtime/entry', { preResolvedRefs });\n const html = `\n<!DOCTYPE html>\n<script type=\"module\">\n import { load } from ${JSON.stringify(entryPoint)};\n await load(${JSON.stringify(manifest)});\n console.info('🚀 loaded');\n</script> \n`;\n return html;\n}\n"],"names":["log","debug","makeOfflineBundle","manifest","deepFreeze","offlineData","makeOffline","preResolvedRefs","file","jsonString","blob","downloadFile","html","getIndexHtml","MAX_CONCURRENT_DOWNLOADS","queue","url","downloadPromise","response","entryPoint","resolve"],"mappings":";;AAKA,MAAMA,IAAMC,EAAM,8BAA8B;AAEhD,gBAAuBC,EACrBC,GACqD;AACrD,EAAAH,EAAI,YAAYG,CAAQ,GACxBC,EAAWD,CAAQ;AACnB,QAAME,IAAc,MAAMC,EAAYH,CAAQ,GACxCI,wBAAsB,IAAA;AAE5B,mBAAiBC,KAAQH,EAAY;AACnC,QAAI,UAAUG,GAAM;AAClB,MAAAD,EAAgB,IAAIC,EAAK,gBAAgBA,EAAK,IAAI;AAClD,YAAMC,IAAa,KAAK,UAAUD,EAAK,MAAM,MAAM,CAAC;AACpD,MAAAR,EAAI,2BAA2BQ,EAAK,gBAAgBC,CAAU,GAC9D,MAAM;AAAA,QACJ,MAAMD,EAAK;AAAA,QACX,MAAM,IAAI,YAAA,EAAc,OAAOC,CAAU,EAAE;AAAA,MAAA;AAAA,IAE/C,WAAW,SAASD,GAAM;AACxB,YAAME,IAAO,MAAMC,EAAaH,EAAK,GAAG;AACxC,MAAAR,EAAI,0BAA0BQ,EAAK,cAAc,GACjD,MAAM;AAAA,QACJ,MAAMA,EAAK;AAAA,QACX,MAAM,MAAME,EAAK,YAAA;AAAA,MAAY;AAAA,IAEjC;AAGF,QAAME,IAAO,MAAMC,EAAaR,EAAY,UAAUE,CAAe;AACrE,QAAM;AAAA,IACJ,MAAM;AAAA,IACN,MAAM,IAAI,YAAA,EAAc,OAAOK,CAAI,EAAE;AAAA,EAAA;AAEzC;AAEA,MAAME,IAA2B,IAC3BC,wBAAY,IAAA;AAElB,eAAeJ,EAAaK,GAA4B;AACtD,SAAOD,EAAM,QAAQD;AACnB,UAAM,QAAQ,KAAKC,CAAK;AAE1B,QAAME,KAAmB,YAAY;AACnC,IAAAjB,EAAI,0BAA0BgB,CAAG;AACjC,UAAME,IAAW,MAAM,MAAMF,CAAG;AAChC,WAAKE,EAAS,OAEPF,EAAI,SAAS,yBAAyB,GAG3C,QAAQ,KAAK,4BAA4BA,CAAG,KAAKE,EAAS,MAAM,EAAE,IAE7D,MAAMA,EAAS,KAAA;AAAA,EACxB,GAAA;AACA,EAAAH,EAAM,IAAIE,CAAe;AACzB,MAAI;AACF,WAAO,MAAMA;AAAA,EACf,UAAA;AACE,IAAAF,EAAM,OAAOE,CAAe;AAAA,EAC9B;AACF;AAEA,eAAeJ,EAAaV,GAAmBI,GAAoD;AACjG,QAAMY,IAAa,MAAMC,EAAQjB,GAAU,kBAAkB,EAAE,iBAAAI,GAAiB;AAShF,SARa;AAAA;AAAA;AAAA,yBAGU,KAAK,UAAUY,CAAU,CAAC;AAAA,eACpC,KAAK,UAAUhB,CAAQ,CAAC;AAAA;AAAA;AAAA;AAKvC;"}
@@ -0,0 +1,7 @@
1
+ function e() {
2
+ throw new Error("saveOfflineZip is not available in the browser");
3
+ }
4
+ export {
5
+ e as saveOfflineZip
6
+ };
7
+ //# sourceMappingURL=saveOfflineZip.browser-BTQeRUY_.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"saveOfflineZip.browser-BTQeRUY_.js","sources":["../../src/offline/saveOfflineZip.browser.ts"],"sourcesContent":["export function saveOfflineZip() {\n throw new Error('saveOfflineZip is not available in the browser');\n}\n"],"names":["saveOfflineZip"],"mappings":"AAAO,SAASA,IAAiB;AAC/B,QAAM,IAAI,MAAM,gDAAgD;AAClE;"}
@@ -7,6 +7,10 @@ const jsonFrozen = new WeakSet();
7
7
  // Cons: can have CSP issues in some environments
8
8
  const importJsonNative = new Function('url', 'return import(url, { with: { type: "json" } });');
9
9
  export async function importJson(url, resolveContext) {
10
+ if (resolveContext.preResolvedRefs?.has(url)) {
11
+ log('importJson: using preResolvedRefs for', url);
12
+ return resolveContext.preResolvedRefs.get(url);
13
+ }
10
14
  // console.warn('importJson:', resolveContext.forceFetch);
11
15
  if (importJsonNotAvailable === undefined && !resolveContext.forceFetch) {
12
16
  try {
@@ -3,5 +3,5 @@ export declare const load: (manifest: unknown, ...args: IArguments[]) => Promise
3
3
  export declare const initialize: (manifest: unknown, ...args: IArguments[]) => Promise<any>;
4
4
  export declare function callFunction_Experimental(name: string, manifest: unknown, ...args: IArguments[]): Promise<any>;
5
5
  export { mutateManifest } from './mutateManifest';
6
- export { downloadZip, makeOfflineBundle } from './offline';
6
+ export { downloadOfflineZip, makeOfflineBundle, saveOfflineZip } from './offline';
7
7
  export { resolve } from './resolve';
package/dist/esm/index.js CHANGED
@@ -18,7 +18,7 @@ export async function callFunction_Experimental(name, manifest, ...args) {
18
18
  return await fn(manifest, ...args);
19
19
  }
20
20
  export { mutateManifest } from './mutateManifest';
21
- export { downloadZip, makeOfflineBundle } from './offline';
21
+ export { downloadOfflineZip, makeOfflineBundle, saveOfflineZip } from './offline';
22
22
  export { resolve } from './resolve';
23
23
  // preloadJson('https://efp-runtime.expofp.com/branches/main.json');
24
24
  // preconnectUrl('https://efp-runtime.expofp.com/');
@@ -0,0 +1,4 @@
1
+ export declare function downloadOfflineZip(manifest: unknown, extraFiles?: Array<{
2
+ path: string;
3
+ data: Uint8Array;
4
+ }>): Promise<void>;
@@ -0,0 +1,15 @@
1
+ import { generateZip } from './generateZip';
2
+ export async function downloadOfflineZip(manifest, extraFiles) {
3
+ // check browser only
4
+ if (typeof window === 'undefined' || typeof document === 'undefined') {
5
+ throw new Error('downloadOfflineZip can only be used in a browser environment');
6
+ }
7
+ // Generate archive
8
+ const blob = await generateZip(manifest, extraFiles);
9
+ // Create link & trigger download
10
+ const a = document.createElement('a');
11
+ a.href = URL.createObjectURL(blob);
12
+ a.download = 'offline.zip';
13
+ a.click();
14
+ URL.revokeObjectURL(a.href);
15
+ }
@@ -0,0 +1,4 @@
1
+ export declare function generateZip(manifest: unknown, extraFiles?: Array<{
2
+ path: string;
3
+ data: Uint8Array;
4
+ }>): Promise<Blob>;
@@ -1,20 +1,21 @@
1
+ import debug from 'debug';
1
2
  import JSZip from 'jszip';
2
- import { log } from '../logger';
3
3
  import { makeOfflineBundle } from './makeOfflineBundle';
4
- export async function downloadZip(manifest) {
4
+ const log = debug('efp:loader:generateZip');
5
+ export async function generateZip(manifest, extraFiles) {
5
6
  const zip = new JSZip();
6
7
  const bundle = await makeOfflineBundle(manifest);
7
8
  for await (const file of bundle) {
9
+ log('Adding file to zip:', file.path, 'size:', file.data.byteLength);
8
10
  zip.file(file.path, file.data);
9
11
  }
12
+ for (const extraFile of extraFiles ?? []) {
13
+ log('Adding extra file to zip:', extraFile.path, 'size:', extraFile.data.byteLength);
14
+ zip.file(extraFile.path, extraFile.data);
15
+ }
10
16
  // Generate archive
11
17
  const blob = await zip.generateAsync({ type: 'blob' });
12
18
  // console.info('Manifest for HTML:', offlineData.manifest);
13
19
  log('Generated offline ZIP bundle, size:', blob.size);
14
- // Create link & trigger download
15
- const a = document.createElement('a');
16
- a.href = URL.createObjectURL(blob);
17
- a.download = 'offline.zip';
18
- a.click();
19
- URL.revokeObjectURL(a.href);
20
+ return blob;
20
21
  }
@@ -2,4 +2,12 @@ export declare function makeOfflineBundle(manifest: unknown): Promise<AsyncGener
2
2
  path: string;
3
3
  data: ArrayBuffer;
4
4
  }, any, any>>;
5
- export declare function downloadZip(manifest: unknown): Promise<void>;
5
+ export declare function generateZip(manifest: unknown, extraFiles?: Array<{
6
+ path: string;
7
+ data: Uint8Array;
8
+ }>): Promise<Blob>;
9
+ export declare function downloadOfflineZip(manifest: unknown, extraFiles?: Array<{
10
+ path: string;
11
+ data: Uint8Array;
12
+ }>): Promise<void>;
13
+ export declare function saveOfflineZip(manifest: unknown, path: string): Promise<void>;
@@ -2,9 +2,17 @@ export async function makeOfflineBundle(manifest) {
2
2
  const { makeOfflineBundle } = await import('./makeOfflineBundle');
3
3
  return await makeOfflineBundle(manifest);
4
4
  }
5
- export async function downloadZip(manifest) {
6
- const { downloadZip } = await import('./downloadZip');
7
- return await downloadZip(manifest);
5
+ export async function generateZip(manifest, extraFiles) {
6
+ const { generateZip } = await import('./generateZip');
7
+ return await generateZip(manifest, extraFiles);
8
+ }
9
+ export async function downloadOfflineZip(manifest, extraFiles) {
10
+ const { downloadOfflineZip } = await import('./downloadOfflineZip');
11
+ return await downloadOfflineZip(manifest, extraFiles);
12
+ }
13
+ export async function saveOfflineZip(manifest, path) {
14
+ const { saveOfflineZip } = await import('./saveOfflineZip');
15
+ return await saveOfflineZip(manifest, path);
8
16
  }
9
17
  async function debugLogOfflineManifestFiles(manifest) {
10
18
  // console.log('Offline files:');
@@ -23,7 +31,7 @@ async function debugLogOfflineManifestFiles(manifest) {
23
31
  console.info('Offline data: ', result.manifest);
24
32
  }
25
33
  if (typeof window !== 'undefined') {
26
- window['__debugDownloadZip'] = downloadZip;
34
+ window['__debugDownloadOfflineZip'] = downloadOfflineZip;
27
35
  window['__debugMakeOfflineBundle'] = makeOfflineBundle;
28
36
  window['__debugLogOfflineManifestFiles'] = debugLogOfflineManifestFiles;
29
37
  }
@@ -1,8 +1,9 @@
1
- import { log } from '../logger';
1
+ import debug from 'debug';
2
2
  import { canResolve, parseRefValue, resolverResolve, } from '../resolve';
3
3
  import { resolvers } from '../resolvers';
4
4
  import { createMergedObjectWithOverridenNonRefProps, deepClone } from '../shared';
5
5
  import { makeTargetPathFromUrl, makeUniqueJsonTargetPathFromString } from './tools';
6
+ const log = debug('efp:loader:makeOffline');
6
7
  export async function makeOffline(manifest) {
7
8
  log('makeOffline', manifest);
8
9
  // if (typeof manifest !== 'object' || manifest === null) {
@@ -64,7 +65,7 @@ async function* makeOfflineInternal(manifest) {
64
65
  throw new Error(`Unknown offlineMethod: ${resolver.offlineMethod}`);
65
66
  }
66
67
  const mergeRef = yield* func(resolver, node[key].$ref, resolveContext);
67
- if (Object.isFrozen(node[key]))
68
+ if (Object.isFrozen(node))
68
69
  throw new Error('Unexpected frozen node during makeOffline');
69
70
  node[key] = createMergedObjectWithOverridenNonRefProps({ $ref: mergeRef }, node[key]);
70
71
  }
@@ -1,12 +1,18 @@
1
- import { log } from '../logger';
2
- import { makeOffline } from './makeOffline';
1
+ import debug from 'debug';
3
2
  import { resolve } from '..';
3
+ import { makeOffline } from './makeOffline';
4
+ import { deepFreeze } from '../shared';
5
+ const log = debug('efp:loader:makeOfflineBundle');
4
6
  export async function* makeOfflineBundle(manifest) {
5
- log('makeOfflineBundle', manifest);
7
+ log('Bundling', manifest);
8
+ deepFreeze(manifest);
6
9
  const offlineData = await makeOffline(manifest);
10
+ const preResolvedRefs = new Map();
7
11
  for await (const file of offlineData.files) {
8
12
  if ('data' in file) {
13
+ preResolvedRefs.set(file.targetFilePath, file.data);
9
14
  const jsonString = JSON.stringify(file.data, null, 2);
15
+ log('Bundling file with data', file.targetFilePath, jsonString);
10
16
  yield {
11
17
  path: file.targetFilePath,
12
18
  data: new TextEncoder().encode(jsonString).buffer,
@@ -14,6 +20,7 @@ export async function* makeOfflineBundle(manifest) {
14
20
  }
15
21
  else if ('url' in file) {
16
22
  const blob = await downloadFile(file.url);
23
+ log('Bundling file from url', file.targetFilePath);
17
24
  yield {
18
25
  path: file.targetFilePath,
19
26
  data: await blob.arrayBuffer(),
@@ -21,9 +28,9 @@ export async function* makeOfflineBundle(manifest) {
21
28
  }
22
29
  }
23
30
  // yield* generateJsLoaderFiles();
24
- const html = await getIndexHtml(offlineData.manifest);
31
+ const html = await getIndexHtml(offlineData.manifest, preResolvedRefs);
25
32
  yield {
26
- path: 'index.html',
33
+ path: './index.html',
27
34
  data: new TextEncoder().encode(html).buffer,
28
35
  };
29
36
  }
@@ -38,7 +45,11 @@ async function downloadFile(url) {
38
45
  log('Fetching file for zip:', url);
39
46
  const response = await fetch(url);
40
47
  if (!response.ok) {
41
- throw new Error(`Failed to fetch ${url}: ${response.status} ${response.statusText}`);
48
+ // Rodion's bug. May be missing file.
49
+ if (!url.includes('/data/fp.svg.ViewBox.js')) {
50
+ // throw new Error(`Failed to fetch ${url}: ${response.status} ${response.statusText}`);
51
+ }
52
+ console.warn(`Warning: Failed to fetch ${url}: ${response.status}`);
42
53
  }
43
54
  return await response.blob();
44
55
  })();
@@ -50,8 +61,8 @@ async function downloadFile(url) {
50
61
  queue.delete(downloadPromise);
51
62
  }
52
63
  }
53
- async function getIndexHtml(manifest) {
54
- const entryPoint = await resolve(manifest, '/runtime/entry');
64
+ async function getIndexHtml(manifest, preResolvedRefs) {
65
+ const entryPoint = await resolve(manifest, '/runtime/entry', { preResolvedRefs });
55
66
  const html = `
56
67
  <!DOCTYPE html>
57
68
  <script type="module">
@@ -0,0 +1 @@
1
+ export declare function saveOfflineZip(): void;
@@ -0,0 +1,3 @@
1
+ export function saveOfflineZip() {
2
+ throw new Error('saveOfflineZip is not available in the browser');
3
+ }
@@ -0,0 +1 @@
1
+ export declare function saveOfflineZip(manifest: unknown, path: string): Promise<void>;
@@ -0,0 +1,16 @@
1
+ import { generateZip } from './generateZip';
2
+ import fs from 'fs/promises';
3
+ export async function saveOfflineZip(manifest, path) {
4
+ if (!path) {
5
+ throw new Error('Path is required to save the ZIP file');
6
+ }
7
+ // should be zip path
8
+ if (!path.endsWith('.zip')) {
9
+ throw new Error('The specified path must end with .zip');
10
+ }
11
+ // Generate archive
12
+ const blob = await generateZip(manifest);
13
+ const arrayBuffer = await blob.arrayBuffer();
14
+ await fs.writeFile(path, Buffer.from(arrayBuffer));
15
+ console.log('Generated offline ZIP bundle at', path);
16
+ }
@@ -1,5 +1,7 @@
1
+ import debug from 'debug';
1
2
  import { hashString } from './hashString';
2
3
  import { slugifyFsUnique } from './slugify';
4
+ const log = debug('efp:loader:offline');
3
5
  export function makeUniqueJsonTargetPathFromString(str, namespace = '') {
4
6
  // const hash = hashString(str);
5
7
  let result = slugifyFsUnique(str); // + '-' + hash;
@@ -40,14 +42,12 @@ export function makeTargetPathFromUrl(url, prefix = '') {
40
42
  const invalidPathnameChars = pathnameWithoutExtension.match(/[^a-zA-Z0-9\-._\/]/g);
41
43
  if (invalidPathnameChars) {
42
44
  const fixedPathnameWithoutExtension = slugifyFsUnique(pathnameWithoutExtension);
43
- console.warn(`Warning: pathname contains invalid filesystem characters (${[
44
- ...new Set(invalidPathnameChars),
45
- ].join(', ')}), slugifying it: ${pathnameWithoutExtension}${extension} => ${fixedPathnameWithoutExtension}${extension}`);
45
+ log(`Pathname contains invalid filesystem characters (${[...new Set(invalidPathnameChars)].join(', ')}), slugifying it: ${pathnameWithoutExtension}${extension} => ${fixedPathnameWithoutExtension}${extension}`);
46
46
  pathnameWithoutExtension = fixedPathnameWithoutExtension;
47
47
  }
48
48
  pathname = pathnameWithoutExtension + extension;
49
49
  if (pathname.length > 120) {
50
- console.warn(`Warning: pathname is too long (${pathname.length} characters), truncating to 150 characters: ${pathname}`);
50
+ log(`Pathname is too long (${pathname.length} characters), truncating to 150 characters: ${pathname}`);
51
51
  pathname = pathname.substring(0, 120 - extension.length) + extension;
52
52
  }
53
53
  if (search) {
@@ -7,14 +7,19 @@ export interface ResolveContextInternal {
7
7
  */
8
8
  importCallback?: (url: string, type: 'json' | 'script') => void;
9
9
  signal: AbortSignal | null;
10
+ preResolvedRefs?: Map<string, any>;
10
11
  }
11
- export interface ResolveOptions extends Partial<ResolveContextInternal> {
12
+ interface ResolveOptionsNotMutate extends Partial<ResolveContextInternal> {
13
+ mutate?: false;
14
+ signal?: AbortSignal;
15
+ }
16
+ interface ResolveOptionMutate extends Omit<ResolveOptionsNotMutate, 'mutate'> {
12
17
  /**
13
18
  * Mutate original object during resolution.
14
19
  */
15
- mutate?: boolean;
16
- signal?: AbortSignal;
20
+ mutate: true;
17
21
  }
22
+ export type ResolveOptions = ResolveOptionsNotMutate | ResolveOptionMutate;
18
23
  /**
19
24
  * Resolves a value from an object using a JSON Pointer RFC 6901.
20
25
  *
@@ -31,8 +36,10 @@ export interface ResolveOptions extends Partial<ResolveContextInternal> {
31
36
  * // result === 'baz'
32
37
  * ```
33
38
  */
34
- export declare function resolve(object: any, jsonPointer: string, options?: ResolveOptions): Promise<any>;
39
+ export declare function resolve(object: any, jsonPointer: string, options: ResolveOptionMutate): Promise<void>;
40
+ export declare function resolve<T = any>(object: any, jsonPointer: string, options?: ResolveOptionsNotMutate): Promise<T>;
35
41
  export declare function resolverResolve(resolver: Resolver, ref: string, context: ResolveContextInternal): Promise<any>;
36
42
  export declare function canResolve(resolver: Resolver, ref: string): boolean;
37
43
  export declare function canResolveRefSchema(ref: string, prefixBase: string): boolean;
38
44
  export declare function parseRefValue(ref: string): string;
45
+ export {};