2024-03-25 01:38:18 +08:00
|
|
|
#!/usr/bin/env node
|
2020-01-16 11:40:46 +01:00
|
|
|
/**
|
2024-06-06 14:40:35 +02:00
|
|
|
* @file
|
2021-02-08 17:14:31 +01:00
|
|
|
* Linters for the package that can't easily be implemented in the existing
|
|
|
|
* linters (e.g. jsonlint/svglint).
|
2020-01-16 11:40:46 +01:00
|
|
|
*/
|
2021-01-15 21:47:00 +01:00
|
|
|
|
2024-06-06 14:40:35 +02:00
|
|
|
/**
|
|
|
|
* @typedef {import("../../sdk.mjs").IconData} IconData
|
|
|
|
* @typedef {import("../../types.js").CustomLicense} CustomLicense
|
|
|
|
* @typedef {IconData[]} IconsData
|
|
|
|
*/
|
|
|
|
|
2023-08-07 22:38:52 -06:00
|
|
|
import process from 'node:process';
|
2021-12-25 06:22:56 -08:00
|
|
|
import fakeDiff from 'fake-diff';
|
2024-03-25 01:38:18 +08:00
|
|
|
import {collator, getIconsDataString, normalizeNewlines} from '../../sdk.mjs';
|
2020-01-16 11:40:46 +01:00
|
|
|
|
|
|
|
/**
|
2021-02-08 17:14:31 +01:00
|
|
|
* Contains our tests so they can be isolated from each other.
|
2024-06-06 14:40:35 +02:00
|
|
|
* @type {{[k: string]: (arg0: {icons: IconsData}, arg1: string) => string | undefined}}
|
2020-01-16 11:40:46 +01:00
|
|
|
*/
|
|
|
|
const TESTS = {
|
2024-06-06 14:40:35 +02:00
|
|
|
/**
|
|
|
|
* Tests whether our icons are in alphabetical order
|
|
|
|
* @param {{icons: IconsData}} data Icons data
|
|
|
|
* @returns {string|undefined} Error message or undefined
|
|
|
|
*/
|
2024-03-25 01:38:18 +08:00
|
|
|
alphabetical(data) {
|
2024-06-06 14:40:35 +02:00
|
|
|
/**
|
|
|
|
* Collects invalid alphabet ordered icons
|
|
|
|
* @param {IconData[]} invalidEntries Invalid icons reference
|
|
|
|
* @param {IconData} icon Icon to check
|
|
|
|
* @param {number} index Index of the icon
|
|
|
|
* @param {IconData[]} array Array of icons
|
|
|
|
* @returns {IconData[]} Invalid icons
|
|
|
|
*/
|
2020-01-16 11:40:46 +01:00
|
|
|
const collector = (invalidEntries, icon, index, array) => {
|
|
|
|
if (index > 0) {
|
2024-03-25 01:38:18 +08:00
|
|
|
const previous = array[index - 1];
|
|
|
|
const comparison = collator.compare(icon.title, previous.title);
|
2022-09-24 17:29:43 +02:00
|
|
|
if (comparison < 0) {
|
2020-01-16 11:40:46 +01:00
|
|
|
invalidEntries.push(icon);
|
2024-03-25 01:38:18 +08:00
|
|
|
} else if (
|
|
|
|
comparison === 0 &&
|
|
|
|
previous.slug &&
|
|
|
|
(!icon.slug || collator.compare(icon.slug, previous.slug) < 0)
|
|
|
|
) {
|
|
|
|
invalidEntries.push(icon);
|
2020-01-16 11:40:46 +01:00
|
|
|
}
|
|
|
|
}
|
2024-03-25 01:38:18 +08:00
|
|
|
|
2020-01-16 11:40:46 +01:00
|
|
|
return invalidEntries;
|
|
|
|
};
|
2024-03-25 01:38:18 +08:00
|
|
|
|
2024-06-06 14:40:35 +02:00
|
|
|
/**
|
|
|
|
* Format an icon for display in the error message
|
|
|
|
* @param {IconData} icon Icon to format
|
|
|
|
* @returns {string} Formatted icon
|
|
|
|
*/
|
2021-10-25 21:13:10 +02:00
|
|
|
const format = (icon) => {
|
2021-04-16 18:05:44 +02:00
|
|
|
if (icon.slug) {
|
|
|
|
return `${icon.title} (${icon.slug})`;
|
|
|
|
}
|
2024-03-25 01:38:18 +08:00
|
|
|
|
2021-04-16 18:05:44 +02:00
|
|
|
return icon.title;
|
|
|
|
};
|
2020-01-16 11:40:46 +01:00
|
|
|
|
2024-03-25 01:38:18 +08:00
|
|
|
// eslint-disable-next-line unicorn/no-array-reduce, unicorn/no-array-callback-reference
|
2021-02-08 17:14:31 +01:00
|
|
|
const invalids = data.icons.reduce(collector, []);
|
2024-03-25 01:38:18 +08:00
|
|
|
if (invalids.length > 0) {
|
2020-01-16 11:40:46 +01:00
|
|
|
return `Some icons aren't in alphabetical order:
|
2021-10-25 21:13:10 +02:00
|
|
|
${invalids.map((icon) => format(icon)).join(', ')}`;
|
2020-01-16 11:40:46 +01:00
|
|
|
}
|
2020-12-13 21:17:41 +01:00
|
|
|
},
|
|
|
|
|
2021-02-08 17:14:31 +01:00
|
|
|
/* Check the formatting of the data file */
|
2024-03-25 01:38:18 +08:00
|
|
|
prettified(data, dataString) {
|
2022-03-31 06:30:30 -06:00
|
|
|
const normalizedDataString = normalizeNewlines(dataString);
|
2022-09-25 09:04:58 +08:00
|
|
|
const dataPretty = `${JSON.stringify(data, null, 4)}\n`;
|
2021-12-25 06:22:56 -08:00
|
|
|
|
|
|
|
if (normalizedDataString !== dataPretty) {
|
|
|
|
const dataDiff = fakeDiff(normalizedDataString, dataPretty);
|
2021-02-08 17:14:31 +01:00
|
|
|
return `Data file is formatted incorrectly:\n\n${dataDiff}`;
|
2020-12-13 21:17:41 +01:00
|
|
|
}
|
2021-10-25 21:13:10 +02:00
|
|
|
},
|
2023-05-29 09:34:33 +08:00
|
|
|
|
|
|
|
/* Check redundant trailing slash in URL */
|
2024-03-25 01:38:18 +08:00
|
|
|
checkUrl(data) {
|
2024-06-06 14:40:35 +02:00
|
|
|
/**
|
|
|
|
* Check if an URL has a redundant trailing slash.
|
|
|
|
* @param {string} url URL to check
|
|
|
|
* @returns {boolean} Whether the URL has a redundant trailing slash
|
|
|
|
*/
|
2023-05-29 09:34:33 +08:00
|
|
|
const hasRedundantTrailingSlash = (url) => {
|
2024-03-25 01:38:18 +08:00
|
|
|
const {origin} = new global.URL(url);
|
2023-05-29 09:34:33 +08:00
|
|
|
return /^\/+$/.test(url.replace(origin, ''));
|
|
|
|
};
|
|
|
|
|
|
|
|
const allUrlFields = [
|
|
|
|
...new Set(
|
2024-06-06 21:24:29 +02:00
|
|
|
data.icons.flatMap((icon) => {
|
|
|
|
/** @type {string[]} */
|
|
|
|
const license =
|
|
|
|
icon.license !== undefined && Object.hasOwn(icon.license, 'url')
|
|
|
|
? [
|
2024-09-17 06:11:21 +08:00
|
|
|
// eslint-disable-next-line no-warning-comments
|
2024-06-06 21:24:29 +02:00
|
|
|
// TODO: `hasOwn` is not currently supported by TS.
|
|
|
|
// See https://github.com/microsoft/TypeScript/issues/44253
|
|
|
|
/** @type {string} */
|
|
|
|
// @ts-ignore
|
|
|
|
icon.license.url,
|
|
|
|
]
|
|
|
|
: [];
|
|
|
|
const guidelines = icon.guidelines ? [icon.guidelines] : [];
|
|
|
|
return [icon.source, ...guidelines, ...license];
|
|
|
|
}),
|
2023-05-29 09:34:33 +08:00
|
|
|
),
|
|
|
|
];
|
|
|
|
|
|
|
|
const invalidUrls = allUrlFields.filter((url) =>
|
|
|
|
hasRedundantTrailingSlash(url),
|
|
|
|
);
|
|
|
|
|
|
|
|
if (invalidUrls.length > 0) {
|
|
|
|
return `Some URLs have a redundant trailing slash:\n\n${invalidUrls.join(
|
|
|
|
'\n',
|
|
|
|
)}`;
|
|
|
|
}
|
|
|
|
},
|
2020-01-16 11:40:46 +01:00
|
|
|
};
|
|
|
|
|
2024-06-06 14:40:35 +02:00
|
|
|
const iconsDataString = await getIconsDataString();
|
|
|
|
const iconsData = JSON.parse(iconsDataString);
|
2021-12-25 06:22:56 -08:00
|
|
|
|
2023-08-07 22:38:52 -06:00
|
|
|
const errors = (
|
2024-06-06 14:40:35 +02:00
|
|
|
await Promise.all(
|
|
|
|
Object.values(TESTS).map((test) => test(iconsData, iconsDataString)),
|
|
|
|
)
|
2024-03-25 01:38:18 +08:00
|
|
|
)
|
|
|
|
// eslint-disable-next-line unicorn/no-await-expression-member
|
|
|
|
.filter(Boolean);
|
2020-01-16 11:40:46 +01:00
|
|
|
|
2023-08-07 22:38:52 -06:00
|
|
|
if (errors.length > 0) {
|
2024-03-25 01:38:18 +08:00
|
|
|
for (const error of errors) console.error(`\u001B[31m${error}\u001B[0m`);
|
2023-08-07 22:38:52 -06:00
|
|
|
process.exit(1);
|
|
|
|
}
|