Files
Nitro_Render_V3/packages/utils/src/JsonParser.ts
T
medievalshell c3b15f02bf perf(gamedata): manifest ext by JSON mode, no double-probe
tryFetchManifestPair sceglie l'estensione in base a resolveJsonMode():
json5 -> .json5, legacy -> .json, auto -> entrambi. Evita le richieste
manifest.json fallite a ogni avvio in modalita json5.
2026-05-30 00:14:44 +02:00

161 lines
4.9 KiB
TypeScript

import JSON5 from 'json5';
declare const __NITRO_JSON_MODE__: 'legacy' | 'json5' | 'auto' | undefined;
const JSON5_EXTENSION = /\.json5(?:[?#]|$)/i;
const JSON5_MIME = /(?:application|text)\/(?:json5|x-json5)/i;
export type ConfigJsonErrorPhase = 'fetch' | 'parse';
export class ConfigJsonError extends Error
{
public readonly phase: ConfigJsonErrorPhase;
public readonly sourceUrl: string;
public readonly httpStatus?: number;
constructor(message: string, phase: ConfigJsonErrorPhase, sourceUrl: string, httpStatus?: number, cause?: unknown)
{
super(message);
this.name = 'ConfigJsonError';
this.phase = phase;
this.sourceUrl = sourceUrl;
this.httpStatus = httpStatus;
if(cause !== undefined) (this as any).cause = cause;
}
}
export const isMissingResource = (err: unknown): boolean =>
err instanceof ConfigJsonError && err.phase === 'fetch' && err.httpStatus === 404;
export const resolveJsonMode = (): 'legacy' | 'json5' | 'auto' =>
{
try
{
if(typeof __NITRO_JSON_MODE__ !== 'undefined' && __NITRO_JSON_MODE__)
{
if(__NITRO_JSON_MODE__ === 'legacy' || __NITRO_JSON_MODE__ === 'json5' || __NITRO_JSON_MODE__ === 'auto') return __NITRO_JSON_MODE__;
}
}
catch {}
return 'auto';
};
const looksLikeJson5Url = (url: string): boolean => !!url && JSON5_EXTENSION.test(url);
const looksLikeJson5ContentType = (contentType: string): boolean => !!contentType && JSON5_MIME.test(contentType);
const formatParseError = (sourceUrl: string, strictError: unknown, json5Error: unknown): string =>
{
const strictMessage = (strictError as Error)?.message || String(strictError);
const json5Message = (json5Error as Error)?.message || String(json5Error);
const source = sourceUrl ? ` in "${ sourceUrl }"` : '';
if(strictMessage === json5Message) return `Failed to parse JSON/JSON5${ source }${ json5Message }`;
return `Failed to parse JSON/JSON5${ source } — JSON5: ${ json5Message } (strict JSON: ${ strictMessage })`;
};
const formatStrictError = (sourceUrl: string, err: unknown): string =>
{
const message = (err as Error)?.message || String(err);
const source = sourceUrl ? ` in "${ sourceUrl }"` : '';
return `Failed to parse strict JSON${ source }${ message } (build is in 'legacy' mode; switch to JSON5 mode via 'yarn configure' to accept comments/trailing commas)`;
};
export const parseConfigJson = <T = any>(text: string, sourceUrl: string = ''): T =>
{
const trimmed = text ?? '';
const mode = resolveJsonMode();
if(mode === 'legacy')
{
try
{
return JSON.parse(trimmed) as T;
}
catch(err)
{
throw new ConfigJsonError(formatStrictError(sourceUrl, err), 'parse', sourceUrl, undefined, err);
}
}
if(mode === 'json5' || looksLikeJson5Url(sourceUrl))
{
try
{
return JSON5.parse<T>(trimmed);
}
catch(err)
{
throw new ConfigJsonError(formatParseError(sourceUrl, err, err), 'parse', sourceUrl, undefined, err);
}
}
let strictError: unknown;
try
{
return JSON.parse(trimmed) as T;
}
catch(err)
{
strictError = err;
}
try
{
return JSON5.parse<T>(trimmed);
}
catch(json5Error)
{
throw new ConfigJsonError(formatParseError(sourceUrl, strictError, json5Error), 'parse', sourceUrl, undefined, json5Error);
}
};
export const parseConfigJsonFromResponse = async <T = any>(response: Response, sourceUrl: string = ''): Promise<T> =>
{
const contentType = response.headers?.get?.('content-type') || '';
const text = await response.text();
const url = sourceUrl || (response as any).url || '';
const mode = resolveJsonMode();
if(mode === 'auto' && looksLikeJson5ContentType(contentType) && !looksLikeJson5Url(url))
{
try
{
return JSON5.parse<T>(text);
}
catch(err)
{
throw new ConfigJsonError(formatParseError(url, err, err), 'parse', url, undefined, err);
}
}
return parseConfigJson<T>(text, url);
};
export const fetchConfigJson = async <T = any>(url: string, init?: RequestInit): Promise<T> =>
{
let response: Response | undefined;
try
{
response = await fetch(url, init);
}
catch(networkErr)
{
const message = (networkErr as Error)?.message || String(networkErr);
throw new ConfigJsonError(`Network error fetching "${ url }" — ${ message }`, 'fetch', url, undefined, networkErr);
}
if(!response || response.status !== 200)
{
const status = response?.status;
throw new ConfigJsonError(`Failed to fetch "${ url }" — server returned HTTP ${ status ?? 'no response' }`, 'fetch', url, status);
}
return parseConfigJsonFromResponse<T>(response, url);
};