1
0
mirror of https://github.com/simple-icons/simple-icons.git synced 2024-11-26 01:00:27 +02:00
simple-icons/scripts/lint/ourlint.js

104 lines
2.9 KiB
JavaScript
Raw Normal View History

2024-03-24 19:38:18 +02:00
#!/usr/bin/env node
/**
* @fileoverview
* Linters for the package that can't easily be implemented in the existing
* linters (e.g. jsonlint/svglint).
*/
import process from 'node:process';
import fakeDiff from 'fake-diff';
2024-03-24 19:38:18 +02:00
import {collator, getIconsDataString, normalizeNewlines} from '../../sdk.mjs';
/**
* Contains our tests so they can be isolated from each other.
* @type {{[k:string]: () => (string|undefined)}}
*/
const TESTS = {
/* Tests whether our icons are in alphabetical order */
2024-03-24 19:38:18 +02:00
alphabetical(data) {
const collector = (invalidEntries, icon, index, array) => {
if (index > 0) {
2024-03-24 19:38:18 +02:00
const previous = array[index - 1];
const comparison = collator.compare(icon.title, previous.title);
if (comparison < 0) {
invalidEntries.push(icon);
2024-03-24 19:38:18 +02:00
} else if (
comparison === 0 &&
previous.slug &&
(!icon.slug || collator.compare(icon.slug, previous.slug) < 0)
) {
invalidEntries.push(icon);
}
}
2024-03-24 19:38:18 +02:00
return invalidEntries;
};
2024-03-24 19:38:18 +02:00
const format = (icon) => {
if (icon.slug) {
return `${icon.title} (${icon.slug})`;
}
2024-03-24 19:38:18 +02:00
return icon.title;
};
2024-03-24 19:38:18 +02:00
// eslint-disable-next-line unicorn/no-array-reduce, unicorn/no-array-callback-reference
const invalids = data.icons.reduce(collector, []);
2024-03-24 19:38:18 +02:00
if (invalids.length > 0) {
return `Some icons aren't in alphabetical order:
${invalids.map((icon) => format(icon)).join(', ')}`;
}
},
/* Check the formatting of the data file */
2024-03-24 19:38:18 +02:00
prettified(data, dataString) {
const normalizedDataString = normalizeNewlines(dataString);
const dataPretty = `${JSON.stringify(data, null, 4)}\n`;
if (normalizedDataString !== dataPretty) {
const dataDiff = fakeDiff(normalizedDataString, dataPretty);
return `Data file is formatted incorrectly:\n\n${dataDiff}`;
}
},
/* Check redundant trailing slash in URL */
2024-03-24 19:38:18 +02:00
checkUrl(data) {
const hasRedundantTrailingSlash = (url) => {
2024-03-24 19:38:18 +02:00
const {origin} = new global.URL(url);
return /^\/+$/.test(url.replace(origin, ''));
};
const allUrlFields = [
...new Set(
data.icons
.flatMap((icon) => [icon.source, icon.guidelines, icon.license?.url])
.filter(Boolean),
),
];
const invalidUrls = allUrlFields.filter((url) =>
hasRedundantTrailingSlash(url),
);
if (invalidUrls.length > 0) {
return `Some URLs have a redundant trailing slash:\n\n${invalidUrls.join(
'\n',
)}`;
}
},
};
const dataString = await getIconsDataString();
const data = JSON.parse(dataString);
const errors = (
await Promise.all(Object.values(TESTS).map((test) => test(data, dataString)))
2024-03-24 19:38:18 +02:00
)
// eslint-disable-next-line unicorn/no-await-expression-member
.filter(Boolean);
if (errors.length > 0) {
2024-03-24 19:38:18 +02:00
for (const error of errors) console.error(`\u001B[31m${error}\u001B[0m`);
process.exit(1);
}