|
|
|
|
|
|
|
|
|
|
|
import { $$ } from "basic-devtools"; |
|
|
|
import TYPES from "./types.js"; |
|
import allPlugins from "./plugins.js"; |
|
import { robustFetch as fetch, getText } from "./fetch.js"; |
|
import { ErrorCode } from "./exceptions.js"; |
|
|
|
const { BAD_CONFIG, CONFLICTING_CODE } = ErrorCode; |
|
|
|
const badURL = (url, expected = "") => { |
|
let message = `(${BAD_CONFIG}): Invalid URL: ${url}`; |
|
if (expected) message += `\nexpected ${expected} content`; |
|
throw new Error(message); |
|
}; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const configDetails = async (config, type) => { |
|
let text = config?.trim(); |
|
|
|
let url = "", |
|
toml = false, |
|
json = /^{/.test(text) && /}$/.test(text); |
|
|
|
if (!json && /\.(\w+)(?:\?\S*)?$/.test(text)) { |
|
const ext = RegExp.$1; |
|
if (ext === "json" && type !== "toml") json = true; |
|
else if (ext === "toml" && type !== "json") toml = true; |
|
else badURL(text, type); |
|
url = text; |
|
text = (await fetch(url).then(getText)).trim(); |
|
} |
|
return { json, toml: toml || (!json && !!text), text, url }; |
|
}; |
|
|
|
const conflictError = (reason) => new Error(`(${CONFLICTING_CODE}): ${reason}`); |
|
|
|
const syntaxError = (type, url, { message }) => { |
|
let str = `(${BAD_CONFIG}): Invalid ${type}`; |
|
if (url) str += ` @ ${url}`; |
|
return new SyntaxError(`${str}\n${message}`); |
|
}; |
|
|
|
const configs = new Map(); |
|
|
|
for (const [TYPE] of TYPES) { |
|
|
|
let plugins; |
|
|
|
|
|
let parsed; |
|
|
|
|
|
let error; |
|
|
|
|
|
let configURL; |
|
|
|
let config, |
|
type, |
|
pyElement, |
|
pyConfigs = $$(`${TYPE}-config`), |
|
attrConfigs = $$( |
|
[ |
|
`script[type="${TYPE}"][config]:not([worker])`, |
|
`${TYPE}-script[config]:not([worker])`, |
|
].join(","), |
|
); |
|
|
|
|
|
if (pyConfigs.length > 1) { |
|
error = conflictError(`Too many ${TYPE}-config`); |
|
} else { |
|
|
|
if (pyConfigs.length && attrConfigs.length) { |
|
error = conflictError( |
|
`Ambiguous ${TYPE}-config VS config attribute`, |
|
); |
|
} else if (pyConfigs.length) { |
|
[pyElement] = pyConfigs; |
|
config = pyElement.getAttribute("src") || pyElement.textContent; |
|
type = pyElement.getAttribute("type"); |
|
} else if (attrConfigs.length) { |
|
[pyElement, ...attrConfigs] = attrConfigs; |
|
config = pyElement.getAttribute("config"); |
|
|
|
if ( |
|
attrConfigs.some((el) => el.getAttribute("config") !== config) |
|
) { |
|
error = conflictError( |
|
"Unable to use different configs on main", |
|
); |
|
} |
|
} |
|
} |
|
|
|
|
|
if (!error && config) { |
|
try { |
|
const { json, toml, text, url } = await configDetails(config, type); |
|
if (url) configURL = new URL(url, location.href).href; |
|
config = text; |
|
if (json || type === "json") { |
|
try { |
|
parsed = JSON.parse(text); |
|
} catch (e) { |
|
error = syntaxError("JSON", url, e); |
|
} |
|
} else if (toml || type === "toml") { |
|
try { |
|
const { parse } = await import( |
|
"./3rd-party/toml.js" |
|
); |
|
parsed = parse(text); |
|
} catch (e) { |
|
error = syntaxError("TOML", url, e); |
|
} |
|
} |
|
} catch (e) { |
|
error = e; |
|
} |
|
} |
|
|
|
|
|
|
|
const toBeAwaited = []; |
|
for (const [key, value] of Object.entries(allPlugins)) { |
|
if (error) { |
|
if (key === "error") { |
|
|
|
|
|
|
|
value().then(({ notify }) => notify(error.message)); |
|
} |
|
} else if (!parsed?.plugins?.includes(`!${key}`)) { |
|
toBeAwaited.push(value().then(({ default: p }) => p)); |
|
} |
|
} |
|
|
|
|
|
plugins = Promise.all(toBeAwaited); |
|
|
|
configs.set(TYPE, { config: parsed, configURL, plugins, error }); |
|
} |
|
|
|
export default configs; |
|
|