/*
* Copyright (C) 2023-2024 Yomitan Authors
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
import {ExtensionError} from './extension-error.js';
/**
* @template {import('api-map').ApiSurface} [TApiSurface=never]
* @template {unknown[]} [TExtraParams=[]]
* @param {import('api-map').ApiMapInit} init
* @returns {import('api-map').ApiMap}
*/
export function createApiMap(init) {
return new Map(init);
}
/**
* @template {import('api-map').ApiSurface} [TApiSurface=never]
* @template {unknown[]} [TExtraParams=[]]
* @param {import('api-map').ApiMap} map
* @param {import('api-map').ApiMapInit} init
* @throws {Error}
*/
export function extendApiMap(map, init) {
for (const [key, value] of init) {
if (map.has(key)) { throw new Error(`The handler for ${String(key)} has already been registered`); }
map.set(key, value);
}
}
/**
* @template {import('api-map').ApiSurface} [TApiSurface=never]
* @template {unknown[]} [TExtraParams=[]]
* @param {import('api-map').ApiMap} map
* @param {string} name
* @returns {import('api-map').ApiHandlerAny|undefined}
*/
export function getApiMapHandler(map, name) {
return map.get(/** @type {import('api-map').ApiNames} */ (name));
}
/**
* @template {import('api-map').ApiSurface} [TApiSurface=never]
* @template {unknown[]} [TExtraParams=[]]
* @param {import('api-map').ApiMap} map
* @param {string} name
* @param {import('api-map').ApiParamsAny} params
* @param {TExtraParams} extraParams
* @param {(response: import('core').Response>) => void} callback
* @param {() => void} [handlerNotFoundCallback]
* @returns {boolean} `true` if async, `false` otherwise.
*/
export function invokeApiMapHandler(map, name, params, extraParams, callback, handlerNotFoundCallback) {
const handler = getApiMapHandler(map, name);
if (typeof handler === 'undefined') {
if (typeof handlerNotFoundCallback === 'function') {
try {
handlerNotFoundCallback();
} catch (error) {
// NOP
}
}
return false;
}
try {
const promiseOrResult = handler(/** @type {import('core').SafeAny} */ (params), ...extraParams);
if (promiseOrResult instanceof Promise) {
/** @type {Promise} */ (promiseOrResult).then(
(result) => { callback({result}); },
(error) => { callback({error: ExtensionError.serialize(error)}); }
);
return true;
} else {
callback({result: promiseOrResult});
return false;
}
} catch (error) {
callback({error: ExtensionError.serialize(error)});
return false;
}
}