ipx 3.1.0 → 4.0.0-alpha.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,764 +0,0 @@
1
- import { defu } from 'defu';
2
- import { withLeadingSlash, hasProtocol, joinURL, decode } from 'ufo';
3
- import { createError, toNodeListener, createApp, defineEventHandler, setResponseStatus, getRequestHeader, appendResponseHeader, send, getResponseHeader, setResponseHeader, toWebHandler, toPlainHandler } from 'h3';
4
- import { imageMeta } from 'image-meta';
5
- import destr from 'destr';
6
- import { negotiate } from '@fastify/accept-negotiator';
7
- import getEtag from 'etag';
8
- import { ofetch } from 'ofetch';
9
- import { resolve, join, parse } from 'pathe';
10
-
11
- const Handlers = {
12
- __proto__: null,
13
- get b () { return b; },
14
- get background () { return background; },
15
- get blur () { return blur; },
16
- get crop () { return crop; },
17
- get enlarge () { return enlarge; },
18
- get extend () { return extend; },
19
- get extract () { return extract; },
20
- get fit () { return fit; },
21
- get flatten () { return flatten; },
22
- get flip () { return flip; },
23
- get flop () { return flop; },
24
- get gamma () { return gamma; },
25
- get grayscale () { return grayscale; },
26
- get h () { return h; },
27
- get height () { return height; },
28
- get kernel () { return kernel; },
29
- get median () { return median; },
30
- get modulate () { return modulate; },
31
- get negate () { return negate; },
32
- get normalize () { return normalize; },
33
- get pos () { return pos; },
34
- get position () { return position; },
35
- get q () { return q; },
36
- get quality () { return quality; },
37
- get resize () { return resize; },
38
- get rotate () { return rotate; },
39
- get s () { return s; },
40
- get sharpen () { return sharpen; },
41
- get threshold () { return threshold; },
42
- get tint () { return tint; },
43
- get trim () { return trim; },
44
- get w () { return w; },
45
- get width () { return width; }
46
- };
47
-
48
- function VArg(argument) {
49
- return destr(argument);
50
- }
51
- function parseArgs(arguments_, mappers) {
52
- const vargs = arguments_.split("_");
53
- return mappers.map((v, index) => v(vargs[index]));
54
- }
55
- function getHandler(key) {
56
- return Handlers[key];
57
- }
58
- function applyHandler(context, pipe, handler, argumentsString) {
59
- const arguments_ = handler.args ? parseArgs(argumentsString, handler.args) : [];
60
- return handler.apply(context, pipe, ...arguments_);
61
- }
62
- function clampDimensionsPreservingAspectRatio(sourceDimensions, desiredDimensions) {
63
- const desiredAspectRatio = desiredDimensions.width / desiredDimensions.height;
64
- let { width, height } = desiredDimensions;
65
- if (sourceDimensions.width && width > sourceDimensions.width) {
66
- width = sourceDimensions.width;
67
- height = Math.round(sourceDimensions.width / desiredAspectRatio);
68
- }
69
- if (sourceDimensions.height && height > sourceDimensions.height) {
70
- height = sourceDimensions.height;
71
- width = Math.round(sourceDimensions.height * desiredAspectRatio);
72
- }
73
- return { width, height };
74
- }
75
-
76
- const quality = {
77
- args: [VArg],
78
- order: -1,
79
- apply: (context, _pipe, quality2) => {
80
- context.quality = quality2;
81
- }
82
- };
83
- const fit = {
84
- args: [VArg],
85
- order: -1,
86
- apply: (context, _pipe, fit2) => {
87
- context.fit = fit2;
88
- }
89
- };
90
- const position = {
91
- args: [VArg],
92
- order: -1,
93
- apply: (context, _pipe, position2) => {
94
- context.position = position2;
95
- }
96
- };
97
- const HEX_RE = /^([\da-f]{2})([\da-f]{2})([\da-f]{2})$/i;
98
- const SHORTHEX_RE = /^([\da-f])([\da-f])([\da-f])$/i;
99
- const background = {
100
- args: [VArg],
101
- order: -1,
102
- apply: (context, _pipe, background2) => {
103
- background2 = String(background2);
104
- if (!background2.startsWith("#") && (HEX_RE.test(background2) || SHORTHEX_RE.test(background2))) {
105
- background2 = "#" + background2;
106
- }
107
- context.background = background2;
108
- }
109
- };
110
- const enlarge = {
111
- args: [],
112
- apply: (context) => {
113
- context.enlarge = true;
114
- }
115
- };
116
- const kernel = {
117
- args: [VArg],
118
- apply: (context, _pipe, kernel2) => {
119
- context.kernel = kernel2;
120
- }
121
- };
122
- const width = {
123
- args: [VArg],
124
- apply: (context, pipe, width2) => {
125
- return pipe.resize(width2, void 0, {
126
- withoutEnlargement: !context.enlarge
127
- });
128
- }
129
- };
130
- const height = {
131
- args: [VArg],
132
- apply: (context, pipe, height2) => {
133
- return pipe.resize(void 0, height2, {
134
- withoutEnlargement: !context.enlarge
135
- });
136
- }
137
- };
138
- const resize = {
139
- args: [VArg, VArg, VArg],
140
- apply: (context, pipe, size) => {
141
- let [width2, height2] = String(size).split("x").map(Number);
142
- if (!width2) {
143
- return;
144
- }
145
- if (!height2) {
146
- height2 = width2;
147
- }
148
- if (!context.enlarge) {
149
- const clamped = clampDimensionsPreservingAspectRatio(context.meta, {
150
- width: width2,
151
- height: height2
152
- });
153
- width2 = clamped.width;
154
- height2 = clamped.height;
155
- }
156
- return pipe.resize(width2, height2, {
157
- fit: context.fit,
158
- position: context.position,
159
- background: context.background,
160
- kernel: context.kernel
161
- });
162
- }
163
- };
164
- const trim = {
165
- args: [VArg],
166
- apply: (_context, pipe, threshold2) => {
167
- return pipe.trim(threshold2);
168
- }
169
- };
170
- const extend = {
171
- args: [VArg, VArg, VArg, VArg],
172
- apply: (context, pipe, top, right, bottom, left) => {
173
- return pipe.extend({
174
- top,
175
- left,
176
- bottom,
177
- right,
178
- background: context.background
179
- });
180
- }
181
- };
182
- const extract = {
183
- args: [VArg, VArg, VArg, VArg],
184
- apply: (_context, pipe, left, top, width2, height2) => {
185
- return pipe.extract({
186
- left,
187
- top,
188
- width: width2,
189
- height: height2
190
- });
191
- }
192
- };
193
- const rotate = {
194
- args: [VArg],
195
- apply: (context, pipe, angel) => {
196
- return pipe.rotate(angel, {
197
- background: context.background
198
- });
199
- }
200
- };
201
- const flip = {
202
- args: [],
203
- apply: (_context, pipe) => {
204
- return pipe.flip();
205
- }
206
- };
207
- const flop = {
208
- args: [],
209
- apply: (_context, pipe) => {
210
- return pipe.flop();
211
- }
212
- };
213
- const sharpen = {
214
- args: [VArg, VArg, VArg],
215
- apply: (_context, pipe, sigma, flat, jagged) => {
216
- return pipe.sharpen(sigma, flat, jagged);
217
- }
218
- };
219
- const median = {
220
- args: [VArg, VArg, VArg],
221
- apply: (_context, pipe, size) => {
222
- return pipe.median(size);
223
- }
224
- };
225
- const blur = {
226
- args: [VArg, VArg, VArg],
227
- apply: (_context, pipe, sigma) => {
228
- return pipe.blur(sigma);
229
- }
230
- };
231
- const flatten = {
232
- args: [VArg, VArg, VArg],
233
- apply: (context, pipe) => {
234
- return pipe.flatten({
235
- background: context.background
236
- });
237
- }
238
- };
239
- const gamma = {
240
- args: [VArg, VArg, VArg],
241
- apply: (_context, pipe, gamma2, gammaOut) => {
242
- return pipe.gamma(gamma2, gammaOut);
243
- }
244
- };
245
- const negate = {
246
- args: [VArg, VArg, VArg],
247
- apply: (_context, pipe) => {
248
- return pipe.negate();
249
- }
250
- };
251
- const normalize = {
252
- args: [VArg, VArg, VArg],
253
- apply: (_context, pipe) => {
254
- return pipe.normalize();
255
- }
256
- };
257
- const threshold = {
258
- args: [VArg],
259
- apply: (_context, pipe, threshold2) => {
260
- return pipe.threshold(threshold2);
261
- }
262
- };
263
- const modulate = {
264
- args: [VArg],
265
- apply: (_context, pipe, brightness, saturation, hue) => {
266
- return pipe.modulate({
267
- brightness,
268
- saturation,
269
- hue
270
- });
271
- }
272
- };
273
- const tint = {
274
- args: [VArg],
275
- apply: (_context, pipe, rgb) => {
276
- return pipe.tint(rgb);
277
- }
278
- };
279
- const grayscale = {
280
- args: [VArg],
281
- apply: (_context, pipe) => {
282
- return pipe.grayscale();
283
- }
284
- };
285
- const crop = extract;
286
- const q = quality;
287
- const b = background;
288
- const w = width;
289
- const h = height;
290
- const s = resize;
291
- const pos = position;
292
-
293
- function getEnv(name) {
294
- return name in process.env ? destr(process.env[name]) : void 0;
295
- }
296
- function cachedPromise(function_) {
297
- let p;
298
- return (...arguments_) => {
299
- if (p) {
300
- return p;
301
- }
302
- p = Promise.resolve(function_(...arguments_));
303
- return p;
304
- };
305
- }
306
-
307
- const SUPPORTED_FORMATS = /* @__PURE__ */ new Set([
308
- "jpeg",
309
- "png",
310
- "webp",
311
- "avif",
312
- "tiff",
313
- "heif",
314
- "gif",
315
- "heic"
316
- ]);
317
- function createIPX(userOptions) {
318
- const options = defu(userOptions, {
319
- alias: getEnv("IPX_ALIAS") || {},
320
- maxAge: getEnv("IPX_MAX_AGE") ?? 60,
321
- sharpOptions: {
322
- jpegProgressive: true
323
- }
324
- });
325
- options.alias = Object.fromEntries(
326
- Object.entries(options.alias || {}).map((e) => [
327
- withLeadingSlash(e[0]),
328
- e[1]
329
- ])
330
- );
331
- const getSharp = cachedPromise(async () => {
332
- return await import('sharp').then(
333
- (r) => r.default || r
334
- );
335
- });
336
- const getSVGO = cachedPromise(async () => {
337
- const { optimize } = await import('svgo');
338
- return { optimize };
339
- });
340
- return function ipx(id, modifiers = {}, opts = {}) {
341
- if (!id) {
342
- throw createError({
343
- statusCode: 400,
344
- statusText: `IPX_MISSING_ID`,
345
- message: `Resource id is missing`
346
- });
347
- }
348
- id = hasProtocol(id) ? id : withLeadingSlash(id);
349
- for (const base in options.alias) {
350
- if (id.startsWith(base)) {
351
- id = joinURL(options.alias[base], id.slice(base.length));
352
- }
353
- }
354
- const storage = hasProtocol(id) ? options.httpStorage || options.storage : options.storage || options.httpStorage;
355
- if (!storage) {
356
- throw createError({
357
- statusCode: 500,
358
- statusText: `IPX_NO_STORAGE`,
359
- message: "No storage configured!"
360
- });
361
- }
362
- const getSourceMeta = cachedPromise(async () => {
363
- const sourceMeta = await storage.getMeta(id, opts);
364
- if (!sourceMeta) {
365
- throw createError({
366
- statusCode: 404,
367
- statusText: `IPX_RESOURCE_NOT_FOUND`,
368
- message: `Resource not found: ${id}`
369
- });
370
- }
371
- const _maxAge = sourceMeta.maxAge ?? options.maxAge;
372
- return {
373
- maxAge: typeof _maxAge === "string" ? Number.parseInt(_maxAge) : _maxAge,
374
- mtime: sourceMeta.mtime ? new Date(sourceMeta.mtime) : void 0
375
- };
376
- });
377
- const getSourceData = cachedPromise(async () => {
378
- const sourceData = await storage.getData(id, opts);
379
- if (!sourceData) {
380
- throw createError({
381
- statusCode: 404,
382
- statusText: `IPX_RESOURCE_NOT_FOUND`,
383
- message: `Resource not found: ${id}`
384
- });
385
- }
386
- return Buffer.from(sourceData);
387
- });
388
- const process = cachedPromise(async () => {
389
- const sourceData = await getSourceData();
390
- let imageMeta$1;
391
- try {
392
- imageMeta$1 = imageMeta(sourceData);
393
- } catch {
394
- throw createError({
395
- statusCode: 400,
396
- statusText: `IPX_INVALID_IMAGE`,
397
- message: `Cannot parse image metadata: ${id}`
398
- });
399
- }
400
- let mFormat = modifiers.f || modifiers.format;
401
- if (mFormat === "jpg") {
402
- mFormat = "jpeg";
403
- }
404
- const format = mFormat && SUPPORTED_FORMATS.has(mFormat) ? mFormat : SUPPORTED_FORMATS.has(imageMeta$1.type || "") ? imageMeta$1.type : "jpeg";
405
- if (imageMeta$1.type === "svg" && !mFormat) {
406
- if (options.svgo === false) {
407
- return {
408
- data: sourceData,
409
- format: "svg+xml",
410
- meta: imageMeta$1
411
- };
412
- } else {
413
- const { optimize } = await getSVGO();
414
- const svg = optimize(sourceData.toString("utf8"), {
415
- ...options.svgo,
416
- plugins: ["removeScripts", ...options.svgo?.plugins || []]
417
- }).data;
418
- return {
419
- data: svg,
420
- format: "svg+xml",
421
- meta: imageMeta$1
422
- };
423
- }
424
- }
425
- const animated = modifiers.animated !== void 0 || modifiers.a !== void 0 || format === "gif";
426
- const Sharp = await getSharp();
427
- let sharp = Sharp(sourceData, { animated, ...options.sharpOptions });
428
- Object.assign(
429
- sharp.options,
430
- options.sharpOptions
431
- );
432
- const handlers = Object.entries(modifiers).map(([name, arguments_]) => ({
433
- handler: getHandler(name),
434
- name,
435
- args: arguments_
436
- })).filter((h) => h.handler).sort((a, b) => {
437
- const aKey = (a.handler.order || a.name || "").toString();
438
- const bKey = (b.handler.order || b.name || "").toString();
439
- return aKey.localeCompare(bKey);
440
- });
441
- const handlerContext = { meta: imageMeta$1 };
442
- for (const h of handlers) {
443
- sharp = applyHandler(handlerContext, sharp, h.handler, h.args) || sharp;
444
- }
445
- if (SUPPORTED_FORMATS.has(format || "")) {
446
- sharp = sharp.toFormat(format, {
447
- quality: handlerContext.quality
448
- });
449
- }
450
- const processedImage = await sharp.toBuffer();
451
- return {
452
- data: processedImage,
453
- format,
454
- meta: imageMeta$1
455
- };
456
- });
457
- return {
458
- getSourceMeta,
459
- process
460
- };
461
- };
462
- }
463
-
464
- const MODIFIER_SEP = /[&,]/g;
465
- const MODIFIER_VAL_SEP = /[:=_]/;
466
- function createIPXH3Handler(ipx) {
467
- const _handler = async (event) => {
468
- const [modifiersString = "", ...idSegments] = event.path.slice(
469
- 1
470
- /* leading slash */
471
- ).split("/");
472
- const id = safeString(decode(idSegments.join("/")));
473
- if (!modifiersString) {
474
- throw createError({
475
- statusCode: 400,
476
- statusText: `IPX_MISSING_MODIFIERS`,
477
- message: `Modifiers are missing: ${id}`
478
- });
479
- }
480
- if (!id || id === "/") {
481
- throw createError({
482
- statusCode: 400,
483
- statusText: `IPX_MISSING_ID`,
484
- message: `Resource id is missing: ${event.path}`
485
- });
486
- }
487
- const modifiers = /* @__PURE__ */ Object.create(null);
488
- if (modifiersString !== "_") {
489
- for (const p of modifiersString.split(MODIFIER_SEP)) {
490
- const [key, ...values] = p.split(MODIFIER_VAL_SEP);
491
- modifiers[safeString(key)] = values.map((v) => safeString(decode(v))).join("_");
492
- }
493
- }
494
- const mFormat = modifiers.f || modifiers.format;
495
- if (mFormat === "auto") {
496
- const acceptHeader = getRequestHeader(event, "accept") || "";
497
- const animated = modifiers.animated ?? modifiers.a;
498
- const autoFormat = autoDetectFormat(
499
- acceptHeader,
500
- // #234 "animated" param adds {animated: ''} to the modifiers
501
- // TODO: fix modifiers to normalized to boolean
502
- !!animated || animated === ""
503
- );
504
- delete modifiers.f;
505
- delete modifiers.format;
506
- if (autoFormat) {
507
- modifiers.format = autoFormat;
508
- appendResponseHeader(event, "vary", "Accept");
509
- }
510
- }
511
- const img = ipx(id, modifiers);
512
- const sourceMeta = await img.getSourceMeta();
513
- sendResponseHeaderIfNotSet(
514
- event,
515
- "content-security-policy",
516
- "default-src 'none'"
517
- );
518
- if (sourceMeta.mtime) {
519
- sendResponseHeaderIfNotSet(
520
- event,
521
- "last-modified",
522
- sourceMeta.mtime.toUTCString()
523
- );
524
- const _ifModifiedSince = getRequestHeader(event, "if-modified-since");
525
- if (_ifModifiedSince && new Date(_ifModifiedSince) >= sourceMeta.mtime) {
526
- setResponseStatus(event, 304);
527
- return send(event);
528
- }
529
- }
530
- const { data, format } = await img.process();
531
- if (typeof sourceMeta.maxAge === "number") {
532
- sendResponseHeaderIfNotSet(
533
- event,
534
- "cache-control",
535
- `max-age=${+sourceMeta.maxAge}, public, s-maxage=${+sourceMeta.maxAge}`
536
- );
537
- }
538
- const etag = getEtag(data);
539
- sendResponseHeaderIfNotSet(event, "etag", etag);
540
- if (etag && getRequestHeader(event, "if-none-match") === etag) {
541
- setResponseStatus(event, 304);
542
- return send(event);
543
- }
544
- if (format) {
545
- sendResponseHeaderIfNotSet(event, "content-type", `image/${format}`);
546
- }
547
- return data;
548
- };
549
- return defineEventHandler(async (event) => {
550
- try {
551
- return await _handler(event);
552
- } catch (_error) {
553
- const error = createError(_error);
554
- setResponseStatus(event, error.statusCode, error.statusMessage);
555
- return {
556
- error: {
557
- message: `[${error.statusCode}] [${error.statusMessage || "IPX_ERROR"}] ${error.message}`
558
- }
559
- };
560
- }
561
- });
562
- }
563
- function createIPXH3App(ipx) {
564
- const app = createApp({ debug: true });
565
- app.use(createIPXH3Handler(ipx));
566
- return app;
567
- }
568
- function createIPXWebServer(ipx) {
569
- return toWebHandler(createIPXH3App(ipx));
570
- }
571
- function createIPXNodeServer(ipx) {
572
- return toNodeListener(createIPXH3App(ipx));
573
- }
574
- function createIPXPlainServer(ipx) {
575
- return toPlainHandler(createIPXH3App(ipx));
576
- }
577
- function sendResponseHeaderIfNotSet(event, name, value) {
578
- if (!getResponseHeader(event, name)) {
579
- setResponseHeader(event, name, value);
580
- }
581
- }
582
- function autoDetectFormat(acceptHeader, animated) {
583
- if (animated) {
584
- const acceptMime2 = negotiate(acceptHeader, ["image/webp", "image/gif"]);
585
- return acceptMime2?.split("/")[1] || "gif";
586
- }
587
- const acceptMime = negotiate(acceptHeader, [
588
- "image/avif",
589
- "image/webp",
590
- "image/jpeg",
591
- "image/png",
592
- "image/tiff",
593
- "image/heif",
594
- "image/gif"
595
- ]);
596
- return acceptMime?.split("/")[1] || "jpeg";
597
- }
598
- function safeString(input) {
599
- return JSON.stringify(input).replace(/^"|"$/g, "").replace(/\\+/g, "\\").replace(/\\"/g, '"');
600
- }
601
-
602
- const HTTP_RE = /^https?:\/\//;
603
- function ipxHttpStorage(_options = {}) {
604
- const allowAllDomains = _options.allowAllDomains ?? getEnv("IPX_HTTP_ALLOW_ALL_DOMAINS") ?? false;
605
- let _domains = _options.domains || getEnv("IPX_HTTP_DOMAINS") || [];
606
- const defaultMaxAge = _options.maxAge || getEnv("IPX_HTTP_MAX_AGE") || 300;
607
- const fetchOptions = _options.fetchOptions || getEnv("IPX_HTTP_FETCH_OPTIONS") || {};
608
- if (typeof _domains === "string") {
609
- _domains = _domains.split(",").map((s) => s.trim());
610
- }
611
- const domains = new Set(
612
- _domains.map((d) => {
613
- if (!HTTP_RE.test(d)) {
614
- d = "http://" + d;
615
- }
616
- return new URL(d).hostname;
617
- }).filter(Boolean)
618
- );
619
- function validateId(id) {
620
- const url = new URL(decodeURIComponent(id));
621
- if (!url.hostname) {
622
- throw createError({
623
- statusCode: 403,
624
- statusText: `IPX_MISSING_HOSTNAME`,
625
- message: `Hostname is missing: ${id}`
626
- });
627
- }
628
- if (!allowAllDomains && !domains.has(url.hostname)) {
629
- throw createError({
630
- statusCode: 403,
631
- statusText: `IPX_FORBIDDEN_HOST`,
632
- message: `Forbidden host: ${url.hostname}`
633
- });
634
- }
635
- return url.toString();
636
- }
637
- function parseResponse(response) {
638
- let maxAge = defaultMaxAge;
639
- if (_options.ignoreCacheControl !== true) {
640
- const _cacheControl = response.headers.get("cache-control");
641
- if (_cacheControl) {
642
- const m = _cacheControl.match(/max-age=(\d+)/);
643
- if (m && m[1]) {
644
- maxAge = Number.parseInt(m[1]);
645
- }
646
- }
647
- }
648
- let mtime;
649
- const _lastModified = response.headers.get("last-modified");
650
- if (_lastModified) {
651
- mtime = new Date(_lastModified);
652
- }
653
- return { maxAge, mtime };
654
- }
655
- return {
656
- name: "ipx:http",
657
- async getMeta(id) {
658
- const url = validateId(id);
659
- try {
660
- const response = await ofetch.raw(url, {
661
- ...fetchOptions,
662
- method: "HEAD"
663
- });
664
- const { maxAge, mtime } = parseResponse(response);
665
- return { mtime, maxAge };
666
- } catch {
667
- return {};
668
- }
669
- },
670
- async getData(id) {
671
- const url = validateId(id);
672
- const response = await ofetch(url, {
673
- ...fetchOptions,
674
- method: "GET",
675
- responseType: "arrayBuffer"
676
- });
677
- return response;
678
- }
679
- };
680
- }
681
-
682
- function ipxFSStorage(_options = {}) {
683
- const dirs = resolveDirs(_options.dir);
684
- const maxAge = _options.maxAge || getEnv("IPX_FS_MAX_AGE");
685
- const _getFS = cachedPromise(
686
- () => import('node:fs/promises').catch(() => {
687
- throw createError({
688
- statusCode: 500,
689
- statusText: `IPX_FILESYSTEM_ERROR`,
690
- message: `Failed to resolve filesystem module`
691
- });
692
- })
693
- );
694
- const resolveFile = async (id) => {
695
- const fs = await _getFS();
696
- for (const dir of dirs) {
697
- const filePath = join(dir, id);
698
- if (!isValidPath(filePath) || !filePath.startsWith(dir)) {
699
- throw createError({
700
- statusCode: 403,
701
- statusText: `IPX_FORBIDDEN_PATH`,
702
- message: `Forbidden path: ${id}`
703
- });
704
- }
705
- try {
706
- const stats = await fs.stat(filePath);
707
- if (!stats.isFile()) {
708
- continue;
709
- }
710
- return {
711
- stats,
712
- read: () => fs.readFile(filePath)
713
- };
714
- } catch (error) {
715
- if (error.code === "ENOENT") {
716
- continue;
717
- }
718
- throw createError({
719
- statusCode: 403,
720
- statusText: `IPX_FORBIDDEN_FILE`,
721
- message: `Cannot access file: ${id}`
722
- });
723
- }
724
- }
725
- throw createError({
726
- statusCode: 404,
727
- statusText: `IPX_FILE_NOT_FOUND`,
728
- message: `File not found: ${id}`
729
- });
730
- };
731
- return {
732
- name: "ipx:node-fs",
733
- async getMeta(id) {
734
- const { stats } = await resolveFile(id);
735
- return {
736
- mtime: stats.mtime,
737
- maxAge
738
- };
739
- },
740
- async getData(id) {
741
- const { read } = await resolveFile(id);
742
- return read();
743
- }
744
- };
745
- }
746
- const isWindows = process.platform === "win32";
747
- function isValidPath(fp) {
748
- if (isWindows) {
749
- fp = fp.slice(parse(fp).root.length);
750
- }
751
- if (/["*:<>?|]/.test(fp)) {
752
- return false;
753
- }
754
- return true;
755
- }
756
- function resolveDirs(dirs) {
757
- if (!dirs || !Array.isArray(dirs)) {
758
- const dir = resolve(dirs || getEnv("IPX_FS_DIR") || ".");
759
- return [dir];
760
- }
761
- return dirs.map((dirs2) => resolve(dirs2));
762
- }
763
-
764
- export { createIPXH3Handler as a, createIPXH3App as b, createIPX as c, createIPXWebServer as d, createIPXNodeServer as e, createIPXPlainServer as f, ipxFSStorage as g, ipxHttpStorage as i };