@riddance/env
Version:
88 lines • 15.2 kB
JavaScript
import { ESLint } from 'eslint';
import { readFile, rename, unlink, writeFile } from 'node:fs/promises';
import { basename, dirname, extname, join, relative } from 'node:path';
export function makeCache(path) {
return new ESLint({ cwd: path, globInputPaths: false });
}
export async function lint(reporter, path, files, cache) {
const results = await cache.lintFiles(files);
if (reporter) {
for (const result of results) {
for (const msg of result.messages) {
reporter.error(msg.message, relative(path, result.filePath), msg.line, msg.column);
}
}
const deprecations = results
.flatMap(r => r.usedDeprecatedRules)
.map(r => r.replacedBy.length === 0
? `${r.ruleId} deprecated`
: `${r.ruleId} deprecated, replaced by ${r.replacedBy.join(',')}`);
if (deprecations.length !== 0) {
for (const message of new Set(deprecations)) {
reporter.error(message);
}
}
}
return !results.some(r => {
return r.fatalErrorCount + r.errorCount + r.warningCount;
});
}
export async function fixLints(path, files) {
const cache = new ESLint({ cwd: path, fix: true });
const results = await cache.lintFiles(files);
const fixables = results.filter(r => r.output);
if (fixables.length === 0) {
return [];
}
await ESLint.outputFixes(results);
const [changed, mapping] = await kebabCaseFiles(path, results.map(r => relative(path, r.filePath)));
return [...changed, ...fixables.map(r => relative(path, r.filePath))].map(f => mapping.reduce((pv, [, camel, kebab]) => pv.replace(camel, kebab), f));
}
async function kebabCaseFiles(path, files) {
const ext = '.ts';
const sourceFiles = files.filter(f => extname(f) === ext);
const renamed = sourceFiles
.map(f => [dirname(f), basename(f, ext)])
.map(([dir, base]) => {
const kebab = base.replaceAll(/[A-Z]+(?![a-z])|[A-Z]/gu, ($, ofs) => (ofs ? '-' : '') + $.toLowerCase());
return kebab === base ? undefined : [dir, base, kebab];
})
.filter(r => !!r);
if (renamed.length === 0) {
return [[], renamed];
}
const changed = await Promise.all(sourceFiles.map(f => updateImports(path, f, renamed)));
await Promise.all([
...renamed.map(([p, camel, kebab]) => rename(join(path, p, camel + ext), join(path, p, kebab + ext))),
...renamed.flatMap(([p, camel]) => ['.d.ts', '.js'].map(e => ensureUnlinked(join(path, p, camel + e)))),
]);
return [changed.filter(c => c !== undefined), renamed];
}
async function updateImports(path, file, renamed) {
const fp = join(path, file);
const text = await readFile(fp, 'utf-8');
let updated = text;
for (const [, camel, kebab] of renamed) {
updated = updated.replaceAll(new RegExp(`import (.+) from '\\.(.*)/${camel}.js'`, 'gu'), (_, i, p) => `import ${i} from '.${p}/${kebab}.js'`);
}
if (updated !== text) {
await writeFile(fp, updated, 'utf-8');
return file;
}
return undefined;
}
async function ensureUnlinked(path) {
try {
await unlink(path);
}
catch (e) {
if (isFileNotFound(e)) {
return;
}
throw e;
}
}
function isFileNotFound(e) {
return e?.code === 'ENOENT';
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"linter.js","sourceRoot":"","sources":["linter.ts"],"names":[],"mappings":"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","sourcesContent":["import { ESLint } from 'eslint'\nimport { readFile, rename, unlink, writeFile } from 'node:fs/promises'\nimport { basename, dirname, extname, join, relative } from 'node:path'\nimport { Reporter } from './reporter.js'\n\nexport function makeCache(path: string) {\n    return new ESLint({ cwd: path, globInputPaths: false })\n}\n\nexport async function lint(\n    reporter: Reporter | undefined,\n    path: string,\n    files: string[],\n    cache: ESLint,\n) {\n    const results = await cache.lintFiles(files)\n    if (reporter) {\n        for (const result of results) {\n            for (const msg of result.messages) {\n                reporter.error(msg.message, relative(path, result.filePath), msg.line, msg.column)\n            }\n        }\n        const deprecations = results\n            .flatMap(r => r.usedDeprecatedRules)\n            .map(r =>\n                r.replacedBy.length === 0\n                    ? `${r.ruleId} deprecated`\n                    : `${r.ruleId} deprecated, replaced by ${r.replacedBy.join(',')}`,\n            )\n        if (deprecations.length !== 0) {\n            for (const message of new Set(deprecations)) {\n                reporter.error(message)\n            }\n        }\n    }\n    return !results.some(r => {\n        return r.fatalErrorCount + r.errorCount + r.warningCount\n    })\n}\n\nexport async function fixLints(path: string, files: string[]) {\n    const cache = new ESLint({ cwd: path, fix: true })\n    const results = await cache.lintFiles(files)\n    const fixables = results.filter(r => r.output)\n    if (fixables.length === 0) {\n        return []\n    }\n    await ESLint.outputFixes(results)\n    const [changed, mapping] = await kebabCaseFiles(\n        path,\n        results.map(r => relative(path, r.filePath)),\n    )\n    return [...changed, ...fixables.map(r => relative(path, r.filePath))].map(f =>\n        mapping.reduce((pv, [, camel, kebab]) => pv.replace(camel, kebab), f),\n    )\n}\n\nasync function kebabCaseFiles(path: string, files: string[]) {\n    const ext = '.ts'\n    const sourceFiles = files.filter(f => extname(f) === ext)\n    const renamed = sourceFiles\n        .map(f => [dirname(f), basename(f, ext)] as const)\n        .map(([dir, base]) => {\n            const kebab = base.replaceAll(\n                /[A-Z]+(?![a-z])|[A-Z]/gu,\n                ($, ofs) => (ofs ? '-' : '') + $.toLowerCase(),\n            )\n            return kebab === base ? undefined : ([dir, base, kebab] as const)\n        })\n        .filter(r => !!r)\n    if (renamed.length === 0) {\n        return [[] as string[], renamed] as const\n    }\n    const changed = await Promise.all(sourceFiles.map(f => updateImports(path, f, renamed)))\n    await Promise.all([\n        ...renamed.map(([p, camel, kebab]) =>\n            rename(join(path, p, camel + ext), join(path, p, kebab + ext)),\n        ),\n        ...renamed.flatMap(([p, camel]) =>\n            ['.d.ts', '.js'].map(e => ensureUnlinked(join(path, p, camel + e))),\n        ),\n    ])\n    return [changed.filter(c => c !== undefined), renamed] as const\n}\n\nasync function updateImports(\n    path: string,\n    file: string,\n    renamed: (readonly [string, string, string])[],\n) {\n    const fp = join(path, file)\n    const text = await readFile(fp, 'utf-8')\n    let updated = text\n    for (const [, camel, kebab] of renamed) {\n        updated = updated.replaceAll(\n            new RegExp(`import (.+) from '\\\\.(.*)/${camel}.js'`, 'gu'),\n            (_, i: string, p: string) => `import ${i} from '.${p}/${kebab}.js'`,\n        )\n    }\n    if (updated !== text) {\n        await writeFile(fp, updated, 'utf-8')\n        return file\n    }\n    return undefined\n}\n\nasync function ensureUnlinked(path: string) {\n    try {\n        await unlink(path)\n    } catch (e) {\n        if (isFileNotFound(e)) {\n            return\n        }\n        throw e\n    }\n}\n\nfunction isFileNotFound(e: unknown) {\n    return (e as { code?: unknown } | undefined)?.code === 'ENOENT'\n}\n"]}