mirror of
https://github.com/denoland/deno.git
synced 2025-01-10 08:09:06 -05:00
309 lines
8.2 KiB
JavaScript
Executable file
309 lines
8.2 KiB
JavaScript
Executable file
#!/usr/bin/env -S deno run --allow-all --config=tests/config/deno.json
|
|
// Copyright 2018-2025 the Deno authors. MIT license.
|
|
|
|
// deno-lint-ignore-file no-console
|
|
|
|
import {
|
|
buildMode,
|
|
dirname,
|
|
getPrebuilt,
|
|
getSources,
|
|
join,
|
|
parseJSONC,
|
|
ROOT_PATH,
|
|
walk,
|
|
} from "./util.js";
|
|
import { checkCopyright } from "./copyright_checker.js";
|
|
import * as ciFile from "../.github/workflows/ci.generate.ts";
|
|
|
|
const promises = [];
|
|
|
|
let js = Deno.args.includes("--js");
|
|
let rs = Deno.args.includes("--rs");
|
|
if (!js && !rs) {
|
|
js = true;
|
|
rs = true;
|
|
}
|
|
|
|
if (rs) {
|
|
promises.push(clippy());
|
|
}
|
|
|
|
if (js) {
|
|
promises.push(dlint());
|
|
promises.push(dlintPreferPrimordials());
|
|
promises.push(ensureCiYmlUpToDate());
|
|
promises.push(ensureNoNewITests());
|
|
promises.push(ensureNoUnusedOutFiles());
|
|
|
|
if (rs) {
|
|
promises.push(checkCopyright());
|
|
}
|
|
}
|
|
|
|
const results = await Promise.allSettled(promises);
|
|
for (const result of results) {
|
|
if (result.status === "rejected") {
|
|
console.error(result.reason);
|
|
Deno.exit(1);
|
|
}
|
|
}
|
|
|
|
async function dlint() {
|
|
const configFile = join(ROOT_PATH, ".dlint.json");
|
|
const execPath = await getPrebuilt("dlint");
|
|
|
|
const sourceFiles = await getSources(ROOT_PATH, [
|
|
"*.js",
|
|
"*.ts",
|
|
":!:.github/mtime_cache/action.js",
|
|
":!:cli/bench/testdata/npm/*",
|
|
":!:cli/bench/testdata/express-router.js",
|
|
":!:cli/bench/testdata/react-dom.js",
|
|
":!:cli/compilers/wasm_wrap.js",
|
|
":!:cli/tools/doc/prism.css",
|
|
":!:cli/tools/doc/prism.js",
|
|
":!:cli/tsc/dts/**",
|
|
":!:cli/tsc/*typescript.js",
|
|
":!:cli/tsc/compiler.d.ts",
|
|
":!:runtime/examples/",
|
|
":!:target/",
|
|
":!:tests/ffi/tests/test.js",
|
|
":!:tests/registry/**",
|
|
":!:tests/specs/**",
|
|
":!:tests/testdata/**",
|
|
":!:tests/unit_node/testdata/**",
|
|
":!:tests/wpt/runner/**",
|
|
":!:tests/wpt/suite/**",
|
|
]);
|
|
|
|
if (!sourceFiles.length) {
|
|
return;
|
|
}
|
|
|
|
const chunks = splitToChunks(sourceFiles, `${execPath} run`.length);
|
|
const pending = [];
|
|
for (const chunk of chunks) {
|
|
const cmd = new Deno.Command(execPath, {
|
|
cwd: ROOT_PATH,
|
|
args: ["run", "--config=" + configFile, ...chunk],
|
|
// capture to not conflict with clippy output
|
|
stderr: "piped",
|
|
});
|
|
pending.push(
|
|
cmd.output().then(({ stderr, code }) => {
|
|
if (code > 0) {
|
|
const decoder = new TextDecoder();
|
|
console.log("\n------ dlint ------");
|
|
console.log(decoder.decode(stderr));
|
|
throw new Error("dlint failed");
|
|
}
|
|
}),
|
|
);
|
|
}
|
|
const results = await Promise.allSettled(pending);
|
|
for (const result of results) {
|
|
if (result.status === "rejected") {
|
|
throw new Error(result.reason);
|
|
}
|
|
}
|
|
}
|
|
|
|
// `prefer-primordials` has to apply only to files related to bootstrapping,
|
|
// which is different from other lint rules. This is why this dedicated function
|
|
// is needed.
|
|
async function dlintPreferPrimordials() {
|
|
const execPath = await getPrebuilt("dlint");
|
|
const sourceFiles = await getSources(ROOT_PATH, [
|
|
"runtime/**/*.js",
|
|
"runtime/**/*.ts",
|
|
"ext/**/*.js",
|
|
"ext/**/*.ts",
|
|
":!:ext/**/*.d.ts",
|
|
"ext/node/polyfills/*.mjs",
|
|
]);
|
|
|
|
if (!sourceFiles.length) {
|
|
return;
|
|
}
|
|
|
|
const chunks = splitToChunks(sourceFiles, `${execPath} run`.length);
|
|
for (const chunk of chunks) {
|
|
const cmd = new Deno.Command(execPath, {
|
|
cwd: ROOT_PATH,
|
|
args: ["run", "--rule", "prefer-primordials", ...chunk],
|
|
stdout: "inherit",
|
|
stderr: "inherit",
|
|
});
|
|
const { code } = await cmd.output();
|
|
|
|
if (code > 0) {
|
|
throw new Error("prefer-primordials failed");
|
|
}
|
|
}
|
|
}
|
|
|
|
function splitToChunks(paths, initCmdLen) {
|
|
let cmdLen = initCmdLen;
|
|
const MAX_COMMAND_LEN = 30000;
|
|
const chunks = [[]];
|
|
for (const p of paths) {
|
|
if (cmdLen + p.length > MAX_COMMAND_LEN) {
|
|
chunks.push([p]);
|
|
cmdLen = initCmdLen;
|
|
} else {
|
|
chunks[chunks.length - 1].push(p);
|
|
cmdLen += p.length;
|
|
}
|
|
}
|
|
return chunks;
|
|
}
|
|
|
|
async function clippy() {
|
|
const currentBuildMode = buildMode();
|
|
const cmd = ["clippy", "--all-targets", "--all-features", "--locked"];
|
|
|
|
if (currentBuildMode != "debug") {
|
|
cmd.push("--release");
|
|
}
|
|
|
|
const cargoCmd = new Deno.Command("cargo", {
|
|
cwd: ROOT_PATH,
|
|
args: [
|
|
...cmd,
|
|
"--",
|
|
"-D",
|
|
"warnings",
|
|
"--deny",
|
|
"clippy::unused_async",
|
|
// generally prefer the `log` crate, but ignore
|
|
// these print_* rules if necessary
|
|
"--deny",
|
|
"clippy::print_stderr",
|
|
"--deny",
|
|
"clippy::print_stdout",
|
|
],
|
|
stdout: "inherit",
|
|
stderr: "inherit",
|
|
});
|
|
const { code } = await cargoCmd.output();
|
|
|
|
if (code > 0) {
|
|
throw new Error("clippy failed");
|
|
}
|
|
}
|
|
|
|
async function ensureCiYmlUpToDate() {
|
|
const expectedCiFileText = ciFile.generate();
|
|
const actualCiFileText = await Deno.readTextFile(ciFile.CI_YML_URL);
|
|
if (expectedCiFileText !== actualCiFileText) {
|
|
throw new Error(
|
|
"./.github/workflows/ci.yml is out of date. Run: ./.github/workflows/ci.generate.ts",
|
|
);
|
|
}
|
|
}
|
|
|
|
async function ensureNoNewITests() {
|
|
// Note: Only decrease these numbers. Never increase them!!
|
|
// This is to help ensure we slowly deprecate these tests and
|
|
// replace them with spec tests.
|
|
const iTestCounts = {
|
|
"bench_tests.rs": 0,
|
|
"cache_tests.rs": 0,
|
|
"cert_tests.rs": 0,
|
|
"check_tests.rs": 0,
|
|
"compile_tests.rs": 0,
|
|
"coverage_tests.rs": 0,
|
|
"eval_tests.rs": 0,
|
|
"flags_tests.rs": 0,
|
|
"fmt_tests.rs": 16,
|
|
"init_tests.rs": 0,
|
|
"inspector_tests.rs": 0,
|
|
"install_tests.rs": 0,
|
|
"jsr_tests.rs": 0,
|
|
"js_unit_tests.rs": 0,
|
|
"jupyter_tests.rs": 0,
|
|
// Read the comment above. Please don't increase these numbers!
|
|
"lsp_tests.rs": 0,
|
|
"node_compat_tests.rs": 0,
|
|
"node_unit_tests.rs": 2,
|
|
"npm_tests.rs": 5,
|
|
"pm_tests.rs": 0,
|
|
"publish_tests.rs": 0,
|
|
"repl_tests.rs": 0,
|
|
"run_tests.rs": 18,
|
|
"shared_library_tests.rs": 0,
|
|
"task_tests.rs": 2,
|
|
"test_tests.rs": 0,
|
|
"upgrade_tests.rs": 0,
|
|
"vendor_tests.rs": 1,
|
|
"watcher_tests.rs": 0,
|
|
"worker_tests.rs": 0,
|
|
};
|
|
const integrationDir = join(ROOT_PATH, "tests", "integration");
|
|
for await (const entry of Deno.readDir(integrationDir)) {
|
|
if (!entry.name.endsWith("_tests.rs")) {
|
|
continue;
|
|
}
|
|
const fileText = await Deno.readTextFile(join(integrationDir, entry.name));
|
|
const actualCount = fileText.match(/itest\!/g)?.length ?? 0;
|
|
const expectedCount = iTestCounts[entry.name] ?? 0;
|
|
// console.log(`"${entry.name}": ${actualCount},`);
|
|
if (actualCount > expectedCount) {
|
|
throw new Error(
|
|
`New itest added to ${entry.name}! The itest macro is deprecated. Please move your new test to ~/tests/specs.`,
|
|
);
|
|
} else if (actualCount < expectedCount) {
|
|
throw new Error(
|
|
`Thanks for removing an itest in ${entry.name}. ` +
|
|
`Please update the count in tools/lint.js for this file to ${actualCount}.`,
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
async function ensureNoUnusedOutFiles() {
|
|
const specsDir = join(ROOT_PATH, "tests", "specs");
|
|
const outFilePaths = new Set(
|
|
(await Array.fromAsync(
|
|
walk(specsDir, { exts: [".out"] }),
|
|
)).map((entry) => entry.path),
|
|
);
|
|
const testFiles = (await Array.fromAsync(
|
|
walk(specsDir, { exts: [".jsonc"] }),
|
|
)).filter((entry) => {
|
|
return entry.path.endsWith("__test__.jsonc");
|
|
});
|
|
|
|
function checkObject(baseDirPath, obj) {
|
|
for (const [key, value] of Object.entries(obj)) {
|
|
if (typeof value === "object") {
|
|
checkObject(baseDirPath, value);
|
|
} else if (key === "output" && typeof value === "string") {
|
|
const outFilePath = join(baseDirPath, value);
|
|
outFilePaths.delete(outFilePath);
|
|
}
|
|
}
|
|
}
|
|
|
|
for (const testFile of testFiles) {
|
|
try {
|
|
const text = await Deno.readTextFile(testFile.path);
|
|
const data = parseJSONC(text);
|
|
checkObject(dirname(testFile.path), data);
|
|
} catch (err) {
|
|
throw new Error("Failed reading: " + testFile.path, {
|
|
cause: err,
|
|
});
|
|
}
|
|
}
|
|
|
|
const notFoundPaths = Array.from(outFilePaths);
|
|
if (notFoundPaths.length > 0) {
|
|
notFoundPaths.sort(); // be deterministic
|
|
for (const file of notFoundPaths) {
|
|
console.error(`Unreferenced .out file: ${file}`);
|
|
}
|
|
throw new Error(`${notFoundPaths.length} unreferenced .out files`);
|
|
}
|
|
}
|