160 lines
4.8 KiB
TypeScript
160 lines
4.8 KiB
TypeScript
import type { FileInfo, JSONSchema } from "../types/index.js";
|
|
import type { ParserOptions } from "../options.js";
|
|
import type { ResolverOptions } from "../types/index.js";
|
|
import type $Refs from "../refs.js";
|
|
import type { Plugin } from "../types/index.js";
|
|
|
|
/**
|
|
* Returns the given plugins as an array, rather than an object map.
|
|
* All other methods in this module expect an array of plugins rather than an object map.
|
|
*
|
|
* @returns
|
|
*/
|
|
export function all<S extends object = JSONSchema, O extends ParserOptions<S> = ParserOptions<S>>(
|
|
plugins: O["resolve"],
|
|
): Plugin[] {
|
|
return (Object.keys(plugins || {}) as (keyof ResolverOptions<S>)[])
|
|
.filter((key) => {
|
|
return typeof plugins![key] === "object";
|
|
})
|
|
.map((key) => {
|
|
(plugins![key] as ResolverOptions<S>)!.name = key;
|
|
return plugins![key] as Plugin;
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Filters the given plugins, returning only the ones return `true` for the given method.
|
|
*/
|
|
export function filter(plugins: Plugin[], method: any, file: any) {
|
|
return plugins.filter((plugin: Plugin) => {
|
|
return !!getResult(plugin, method, file);
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Sorts the given plugins, in place, by their `order` property.
|
|
*/
|
|
export function sort(plugins: Plugin[]) {
|
|
for (const plugin of plugins) {
|
|
plugin.order = plugin.order || Number.MAX_SAFE_INTEGER;
|
|
}
|
|
|
|
return plugins.sort((a: any, b: any) => {
|
|
return a.order - b.order;
|
|
});
|
|
}
|
|
|
|
export interface PluginResult<S extends object = JSONSchema> {
|
|
plugin: Plugin;
|
|
result?: string | Buffer | S;
|
|
error?: any;
|
|
}
|
|
|
|
/**
|
|
* Runs the specified method of the given plugins, in order, until one of them returns a successful result.
|
|
* Each method can return a synchronous value, a Promise, or call an error-first callback.
|
|
* If the promise resolves successfully, or the callback is called without an error, then the result
|
|
* is immediately returned and no further plugins are called.
|
|
* If the promise rejects, or the callback is called with an error, then the next plugin is called.
|
|
* If ALL plugins fail, then the last error is thrown.
|
|
*/
|
|
export async function run<S extends object = JSONSchema, O extends ParserOptions<S> = ParserOptions<S>>(
|
|
plugins: Plugin[],
|
|
method: keyof Plugin | keyof ResolverOptions<S>,
|
|
file: FileInfo,
|
|
$refs: $Refs<S, O>,
|
|
) {
|
|
let plugin: Plugin;
|
|
let lastError: PluginResult<S>;
|
|
let index = 0;
|
|
|
|
return new Promise<PluginResult<S>>((resolve, reject) => {
|
|
runNextPlugin();
|
|
|
|
function runNextPlugin() {
|
|
plugin = plugins[index++];
|
|
if (!plugin) {
|
|
// There are no more functions, so re-throw the last error
|
|
return reject(lastError);
|
|
}
|
|
|
|
try {
|
|
// console.log(' %s', plugin.name);
|
|
const result = getResult(plugin, method, file, callback, $refs);
|
|
if (result && typeof result.then === "function") {
|
|
// A promise was returned
|
|
result.then(onSuccess, onError);
|
|
} else if (result !== undefined) {
|
|
// A synchronous result was returned
|
|
onSuccess(result);
|
|
} else if (index === plugins.length) {
|
|
throw new Error("No promise has been returned or callback has been called.");
|
|
}
|
|
} catch (e) {
|
|
onError(e);
|
|
}
|
|
}
|
|
|
|
function callback(err: PluginResult<S>["error"], result: PluginResult<S>["result"]) {
|
|
if (err) {
|
|
onError(err);
|
|
} else {
|
|
onSuccess(result);
|
|
}
|
|
}
|
|
|
|
function onSuccess(result: PluginResult<S>["result"]) {
|
|
// console.log(' success');
|
|
resolve({
|
|
plugin,
|
|
result: result!,
|
|
});
|
|
}
|
|
|
|
function onError(error: PluginResult<S>["error"]) {
|
|
// console.log(' %s', err.message || err);
|
|
lastError = {
|
|
plugin,
|
|
error,
|
|
};
|
|
runNextPlugin();
|
|
}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Returns the value of the given property.
|
|
* If the property is a function, then the result of the function is returned.
|
|
* If the value is a RegExp, then it will be tested against the file URL.
|
|
* If the value is an array, then it will be compared against the file extension.
|
|
*/
|
|
function getResult<S extends object = JSONSchema, O extends ParserOptions<S> = ParserOptions<S>>(
|
|
obj: Plugin,
|
|
prop: keyof Plugin | keyof ResolverOptions<S>,
|
|
file: FileInfo,
|
|
callback?: (err?: Error, result?: any) => void,
|
|
$refs?: $Refs<S, O>,
|
|
) {
|
|
const value = obj[prop as keyof typeof obj] as unknown;
|
|
|
|
if (typeof value === "function") {
|
|
return value.apply(obj, [file, callback, $refs]);
|
|
}
|
|
|
|
if (!callback) {
|
|
// The synchronous plugin functions (canParse and canRead)
|
|
// allow a "shorthand" syntax, where the user can match
|
|
// files by RegExp or by file extension.
|
|
if (value instanceof RegExp) {
|
|
return value.test(file.url);
|
|
} else if (typeof value === "string") {
|
|
return value === file.extension;
|
|
} else if (Array.isArray(value)) {
|
|
return value.indexOf(file.extension) !== -1;
|
|
}
|
|
}
|
|
|
|
return value;
|
|
}
|