Skip to content

Commit

Permalink
Merge branch 'next' into ts-tighten
Browse files Browse the repository at this point in the history
  • Loading branch information
Marsup authored Oct 22, 2024
2 parents 316d8c1 + 0c55843 commit dcfadc7
Show file tree
Hide file tree
Showing 35 changed files with 306 additions and 95 deletions.
9 changes: 0 additions & 9 deletions .eslintignore

This file was deleted.

20 changes: 20 additions & 0 deletions eslint.config.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
'use strict';

const HapiPlugin = require('@hapi/eslint-plugin');

module.exports = [
{
ignores: [
'node_modules/',
'test_runner/',
'test/coverage/',
'test/cli/',
'test/cli_*/',
'test/lint/',
'test/override/',
'test/plan/',
'test/transform/'
]
},
...HapiPlugin.configs.module
];
26 changes: 26 additions & 0 deletions lib/cli.js
Original file line number Diff line number Diff line change
Expand Up @@ -81,6 +81,28 @@ exports.run = async function () {
};


internals.addTsEsmHook = function () {

const Module = require('module');

const originalResolveFilename = Module._resolveFilename;

Module._resolveFilename = function (request, parent, ...rest) {

if (request.endsWith('.js') && (parent.filename.endsWith('.ts') || parent.filename.endsWith('.tsx'))) {
const target = Path.join(parent.path, request);
const tsEquivalent = `${target.slice(0, -3)}.ts`;

if (!Fs.existsSync(target) && Fs.existsSync(tsEquivalent)) {
return tsEquivalent;
}
}

return originalResolveFilename.call(this, request, parent, ...rest);
};
};


internals.traverse = async function (paths, options) {

let nextPath = null;
Expand Down Expand Up @@ -141,6 +163,10 @@ internals.traverse = async function (paths, options) {

const scripts = [];

if (options.typescript && testFiles.some(([, defaultToESM]) => defaultToESM)) {
internals.addTsEsmHook();
}

for (const [unresolvedFile, defaultToESM] of testFiles) {

global._labScriptRun = false;
Expand Down
6 changes: 3 additions & 3 deletions lib/linter/.eslintrc.js
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
'use strict';

module.exports = {
extends: 'plugin:@hapi/module'
};
const HapiPlugin = require('@hapi/eslint-plugin');

module.exports = [...HapiPlugin.configs.module];
50 changes: 40 additions & 10 deletions lib/linter/index.js
Original file line number Diff line number Diff line change
@@ -1,7 +1,6 @@
'use strict';

const Fs = require('fs');
const Path = require('path');

const Eslint = require('eslint');
const Hoek = require('@hapi/hoek');
Expand All @@ -18,23 +17,47 @@ exports.lint = async function () {

const options = process.argv[2] ? JSON.parse(process.argv[2]) : undefined;

if (!Fs.existsSync('.eslintrc.js') &&
!Fs.existsSync('.eslintrc.cjs') && // Needed for projects with "type": "module"
!Fs.existsSync('.eslintrc.yaml') &&
!Fs.existsSync('.eslintrc.yml') &&
!Fs.existsSync('.eslintrc.json') &&
!Fs.existsSync('.eslintrc')) {
configuration.overrideConfigFile = Path.join(__dirname, '.eslintrc.js');
let usingDefault = false;

if (!Fs.existsSync('eslint.config.js') &&
!Fs.existsSync('eslint.config.cjs') &&
!Fs.existsSync('eslint.config.mjs')) {
// No configuration file found, using the default one
usingDefault = true;
configuration.baseConfig = require('./.eslintrc.js');
configuration.overrideConfigFile = true;
}

if (options) {
Hoek.merge(configuration, options, true, false);
}

if (!configuration.extensions) {
configuration.extensions = ['.js', '.cjs', '.mjs'];
// Only the default configuration should be altered, otherwise the user's configuration should be used as is
if (usingDefault) {
if (!configuration.extensions) {
const extensions = ['js', 'cjs', 'mjs'];

if (configuration.typescript) {
extensions.push('ts');
}

configuration.baseConfig.unshift({
files: extensions.map((ext) => `**/*.${ext}`)
});
}

if (configuration.ignores) {
configuration.baseConfig.unshift({
ignores: configuration.ignores
});
}
}

delete configuration.extensions;
delete configuration.typescript;
delete configuration.ignores;


let results;
try {
const eslint = new Eslint.ESLint(configuration);
Expand All @@ -58,6 +81,13 @@ exports.lint = async function () {

transformed.errors = result.messages.map((err) => {

if (err.messageTemplate === 'all-matched-files-ignored') {
return {
severity: 'ERROR',
message: err.message
};
}

return {
line: err.line,
severity: err.severity === 1 ? 'WARNING' : 'ERROR',
Expand Down
42 changes: 35 additions & 7 deletions lib/modules/coverage.js
Original file line number Diff line number Diff line change
Expand Up @@ -16,8 +16,7 @@ const SourceMap = require('../source-map');
const Transform = require('./transform');

const internals = {
_state: Symbol.for('@hapi/lab/coverage/_state'),
eslint: new ESLint.ESLint({ baseConfig: Eslintrc })
_state: Symbol.for('@hapi/lab/coverage/_state')
};


Expand Down Expand Up @@ -111,7 +110,7 @@ internals.prime = function (extension, ctx) {
require.extensions[extension] = function (localModule, filename) {

// We never want to instrument eslint configs in order to avoid infinite recursion
if (Path.basename(filename, extension) !== '.eslintrc') {
if (!['.eslintrc', 'eslint.config'].includes(Path.basename(filename, extension))) {
for (let i = 0; i < internals.state.patterns.length; ++i) {
if (internals.state.patterns[i].test(filename.replace(/\\/g, '/'))) {
return localModule._compile(internals.instrument(filename, ctx), filename);
Expand Down Expand Up @@ -307,7 +306,7 @@ internals.instrument = function (filename, ctx) {
node.set(`(global.__$$labCov._statement(\'${filename}\',${left},${line},${node.left.source()})${node.operator}global.__$$labCov._statement(\'${filename}\',${right},${line},${node.right.source()}))`);
}
else if (node.parent?.type === 'ArrowFunctionExpression' &&
node.type.includes('Expression')) {
(node.type.includes('Expression') || node.type === 'Literal')) {

const id = addStatement(line, node, false);

Expand Down Expand Up @@ -761,11 +760,40 @@ internals.file = async function (filename, data, options) {

internals.context = async (options) => {

const filePath = Path.join(options.coveragePath || '', 'x.js');
let calculated;

// The parserOptions are shared by all files for coverage purposes, based on
// the effective eslint config for a hypothetical file {coveragePath}/x.js
const { parserOptions } = await internals.eslint.calculateConfigForFile(
Path.join(options.coveragePath || '', 'x.js')
);
try {
// Let's try first with eslint's native configuration detection
const eslint = new ESLint.ESLint({
ignore: false
});

calculated = await eslint.calculateConfigForFile(filePath);
}
catch (err) {
/* $lab:coverage:off$ */
if (err.messageTemplate !== 'config-file-missing') {
throw err;
}

// If the eslint config file is missing, we'll use the one provided by lab
const eslint = new ESLint.ESLint({
overrideConfig: Eslintrc,
overrideConfigFile: true,
ignore: false
});

calculated = await eslint.calculateConfigForFile(filePath);
/* $lab:coverage:on$ */
}

const parserOptions = {
...calculated.languageOptions,
...calculated.languageOptions?.parserOptions
};

return { parserOptions };
};
3 changes: 2 additions & 1 deletion lib/modules/leaks.js
Original file line number Diff line number Diff line change
Expand Up @@ -71,7 +71,8 @@ const internals = {
'__propKey',
'__setFunctionName',
'__addDisposableResource',
'__disposeResources'
'__disposeResources',
'__rewriteRelativeImportExtension'
]
};

Expand Down
6 changes: 5 additions & 1 deletion lib/modules/lint.js
Original file line number Diff line number Diff line change
Expand Up @@ -20,12 +20,16 @@ exports.lint = function (settings) {
try {
linterOptions = JSON.parse(settings['lint-options'] || '{}');
}
catch (err) {
catch {
return reject(new Error('lint-options could not be parsed'));
}

linterOptions.fix = settings['lint-fix'];

if (settings.typescript) {
linterOptions.typescript = true;
}

const child = ChildProcess.fork(linterPath, [JSON.stringify(linterOptions)], { cwd: settings.lintingPath });
child.once('message', (message) => {

Expand Down
2 changes: 1 addition & 1 deletion lib/modules/transform.js
Original file line number Diff line number Diff line change
Expand Up @@ -73,7 +73,7 @@ exports.retrieveFile = function (path) {
try {
contents = Fs.readFileSync(path, 'utf8');
}
catch (e) {
catch {
contents = null;
}

Expand Down
15 changes: 2 additions & 13 deletions lib/modules/types.js
Original file line number Diff line number Diff line change
Expand Up @@ -32,18 +32,7 @@ const internals = {
],

report: [
2304, // Cannot find name
2345, // Argument type is not assignable to parameter type
2339, // Property does not exist on type
2540, // Cannot assign to readonly property
2322, // Type is not assignable to other type
2314, // Generic type requires type arguments
2554, // Expected arguments but got other
2559, // Type T has no properties in common with type U
2769, // No overload matches this call
2673, // Constructor of class is private
2674, // Constructor of class is protected
2820 // Type T is not assignable to type U. Did you mean V?
1005 // Syntax error
]
};

Expand Down Expand Up @@ -164,7 +153,7 @@ internals.ignore = function (diagnostic, expectedErrors) {
return true;
}

if (!internals.report.includes(diagnostic.code)) {
if (internals.report.includes(diagnostic.code)) {
return false;
}

Expand Down
2 changes: 1 addition & 1 deletion lib/modules/typescript.js
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@ internals.transform = function (content, fileName) {
try {
var { config, error } = Typescript.readConfigFile(configFile, Typescript.sys.readFile);
}
catch (err) {
catch {
throw new Error(`Cannot find a tsconfig file for ${fileName}`);
}

Expand Down
2 changes: 2 additions & 0 deletions lib/runner.js
Original file line number Diff line number Diff line change
Expand Up @@ -13,10 +13,12 @@ const internals = {};

// Prevent libraries like Sinon from clobbering global time functions

/* eslint-disable no-redeclare */
const Date = global.Date;
const setTimeout = global.setTimeout;
const clearTimeout = global.clearTimeout;
const setImmediate = global.setImmediate;
/* eslint-enable no-redeclare */


Error.stackTraceLimit = Infinity; // Set Error stack size
Expand Down
19 changes: 8 additions & 11 deletions package.json
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
{
"name": "@hapi/lab",
"description": "Test utility",
"version": "25.1.3",
"version": "25.3.2",
"repository": "git://github.com/hapijs/lab",
"main": "lib/index.js",
"types": "lib/index.d.ts",
Expand All @@ -13,19 +13,14 @@
"bin/lab",
"lib"
],
"eslintConfig": {
"extends": [
"plugin:@hapi/module"
]
},
"dependencies": {
"@babel/core": "^7.16.0",
"@babel/eslint-parser": "^7.16.0",
"@babel/eslint-parser": "^7.25.1",
"@hapi/bossy": "^6.0.0",
"@hapi/eslint-plugin": "^6.0.0",
"@hapi/eslint-plugin": "^7.0.0",
"@hapi/hoek": "^11.0.2",
"diff": "^5.0.0",
"eslint": "8.x.x",
"eslint": "9.x.x",
"find-rc": "4.x.x",
"globby": "^11.1.0",
"handlebars": "4.x.x",
Expand All @@ -37,7 +32,7 @@
"will-call": "1.x.x"
},
"peerDependencies": {
"@hapi/eslint-plugin": "^6.0.0",
"@hapi/eslint-plugin": "^7.0.0",
"typescript": ">=4.4.0"
},
"peerDependenciesMeta": {
Expand All @@ -48,12 +43,14 @@
"devDependencies": {
"@hapi/code": "^9.0.0",
"@hapi/somever": "^4.0.0",
"@types/eslint": "^9.6.0",
"@types/node": "^18.11.17",
"cpr": "3.x.x",
"lab-event-reporter": "1.x.x",
"semver": "7.x.x",
"tsconfig-paths": "^4.0.0",
"typescript": "^4.5.4"
"typescript": "^4.5.4",
"typescript-eslint": "^8.1.0"
},
"bin": {
"lab": "./bin/lab"
Expand Down
3 changes: 2 additions & 1 deletion test/cli.js
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@
// Load modules

const ChildProcess = require('child_process');
// eslint-disable-next-line no-redeclare
const Crypto = require('crypto');
const Fs = require('fs');
const Http = require('http');
Expand Down Expand Up @@ -702,7 +703,7 @@ describe('CLI', () => {
try {
await unlink(outputPath);
}
catch (err) {
catch {

// Error is ok here
}
Expand Down
Loading

0 comments on commit dcfadc7

Please sign in to comment.