From 3285386cbed0acb40ef097e3d7005262a0ec6567 Mon Sep 17 00:00:00 2001 From: Patricio Palladino Date: Sat, 31 Aug 2024 14:58:44 +0000 Subject: [PATCH 01/14] Don't reexport ArgumentType --- v-next/hardhat/src/internal/builtin-global-options.ts | 3 ++- v-next/hardhat/src/internal/cli/helpers/utils.ts | 6 ++++-- v-next/hardhat/src/internal/cli/main.ts | 10 +++++----- v-next/hardhat/src/internal/core/config.ts | 2 -- v-next/hardhat/test/internal/core/config-validation.ts | 10 +++++----- v-next/hardhat/test/internal/core/global-options.ts | 6 ++---- v-next/hardhat/test/internal/core/tasks/builders.ts | 2 +- .../hardhat/test/internal/core/tasks/task-manager.ts | 6 ++---- 8 files changed, 21 insertions(+), 24 deletions(-) diff --git a/v-next/hardhat/src/internal/builtin-global-options.ts b/v-next/hardhat/src/internal/builtin-global-options.ts index 0d42b80394..eff895351c 100644 --- a/v-next/hardhat/src/internal/builtin-global-options.ts +++ b/v-next/hardhat/src/internal/builtin-global-options.ts @@ -1,6 +1,7 @@ import type { GlobalOptionDefinitions } from "../types/global-options.js"; -import { globalOption, ArgumentType } from "../config.js"; +import { globalOption } from "../config.js"; +import { ArgumentType } from "../types/arguments.js"; export const BUILTIN_GLOBAL_OPTIONS_DEFINITIONS: GlobalOptionDefinitions = new Map([ diff --git a/v-next/hardhat/src/internal/cli/helpers/utils.ts b/v-next/hardhat/src/internal/cli/helpers/utils.ts index e948b247d1..d77d9f29a4 100644 --- a/v-next/hardhat/src/internal/cli/helpers/utils.ts +++ b/v-next/hardhat/src/internal/cli/helpers/utils.ts @@ -1,7 +1,9 @@ -import type { ArgumentTypeToValueType } from "../../../types/arguments.js"; +import type { + ArgumentType, + ArgumentTypeToValueType, +} from "../../../types/arguments.js"; import type { GlobalOptionDefinitions } from "../../../types/global-options.js"; import type { Task } from "../../../types/tasks.js"; -import type { ArgumentType } from "../../core/config.js"; import { camelToKebabCase } from "@ignored/hardhat-vnext-utils/string"; diff --git a/v-next/hardhat/src/internal/cli/main.ts b/v-next/hardhat/src/internal/cli/main.ts index 257e67b5f5..68946f5ca7 100644 --- a/v-next/hardhat/src/internal/cli/main.ts +++ b/v-next/hardhat/src/internal/cli/main.ts @@ -1,7 +1,3 @@ -import type { - OptionDefinition, - PositionalArgumentDefinition, -} from "../../types/arguments.js"; import type { GlobalOptionDefinitions, GlobalOptions, @@ -20,9 +16,13 @@ import debug from "debug"; import { resolveHardhatConfigPath } from "../../config.js"; import { createHardhatRuntimeEnvironment } from "../../hre.js"; +import { + ArgumentType, + type OptionDefinition, + type PositionalArgumentDefinition, +} from "../../types/arguments.js"; import { BUILTIN_GLOBAL_OPTIONS_DEFINITIONS } from "../builtin-global-options.js"; import { builtinPlugins } from "../builtin-plugins/index.js"; -import { ArgumentType } from "../core/config.js"; import { buildGlobalOptionDefinitions, parseArgumentValue, diff --git a/v-next/hardhat/src/internal/core/config.ts b/v-next/hardhat/src/internal/core/config.ts index ce1e84aa18..d374f2d5e2 100644 --- a/v-next/hardhat/src/internal/core/config.ts +++ b/v-next/hardhat/src/internal/core/config.ts @@ -20,8 +20,6 @@ import { export type { HardhatUserConfig } from "../../types/config.js"; -export { ArgumentType } from "../../types/arguments.js"; - /** * Creates a configuration variable, which will be fetched at runtime. */ diff --git a/v-next/hardhat/test/internal/core/config-validation.ts b/v-next/hardhat/test/internal/core/config-validation.ts index 576082bd7f..a9c3ae33f3 100644 --- a/v-next/hardhat/test/internal/core/config-validation.ts +++ b/v-next/hardhat/test/internal/core/config-validation.ts @@ -1,8 +1,4 @@ import type { HardhatUserConfig } from "../../../src/config.js"; -import type { - PositionalArgumentDefinition, - OptionDefinition, -} from "../../../src/types/arguments.js"; import type { HardhatPlugin } from "../../../src/types/plugins.js"; import type { EmptyTaskDefinition, @@ -14,7 +10,6 @@ import type { import assert from "node:assert/strict"; import { describe, it } from "node:test"; -import { ArgumentType } from "../../../src/config.js"; import { validatePositionalArguments, validateOptions, @@ -25,6 +20,11 @@ import { validatePluginsConfig, collectValidationErrorsForUserConfig, } from "../../../src/internal/core/config-validation.js"; +import { + type PositionalArgumentDefinition, + type OptionDefinition, + ArgumentType, +} from "../../../src/types/arguments.js"; import { TaskDefinitionType } from "../../../src/types/tasks.js"; describe("config validation", function () { diff --git a/v-next/hardhat/test/internal/core/global-options.ts b/v-next/hardhat/test/internal/core/global-options.ts index 18cf949ebb..2e7ed4198c 100644 --- a/v-next/hardhat/test/internal/core/global-options.ts +++ b/v-next/hardhat/test/internal/core/global-options.ts @@ -7,15 +7,13 @@ import { HardhatError } from "@ignored/hardhat-vnext-errors"; import { assertThrowsHardhatError } from "@nomicfoundation/hardhat-test-utils"; import { RESERVED_ARGUMENT_NAMES } from "../../../src/internal/core/arguments.js"; -import { - ArgumentType, - globalOption, -} from "../../../src/internal/core/config.js"; +import { globalOption } from "../../../src/internal/core/config.js"; import { buildGlobalOptionDefinition, buildGlobalOptionDefinitions, resolveGlobalOptions, } from "../../../src/internal/core/global-options.js"; +import { ArgumentType } from "../../../src/types/arguments.js"; import { createTestEnvManager } from "./utils.js"; diff --git a/v-next/hardhat/test/internal/core/tasks/builders.ts b/v-next/hardhat/test/internal/core/tasks/builders.ts index dc7a58c1cb..0d85dee3f6 100644 --- a/v-next/hardhat/test/internal/core/tasks/builders.ts +++ b/v-next/hardhat/test/internal/core/tasks/builders.ts @@ -4,13 +4,13 @@ import { after, before, describe, it } from "node:test"; import { HardhatError } from "@ignored/hardhat-vnext-errors"; import { assertThrowsHardhatError } from "@nomicfoundation/hardhat-test-utils"; -import { ArgumentType } from "../../../../src/config.js"; import { RESERVED_ARGUMENT_NAMES } from "../../../../src/internal/core/arguments.js"; import { EmptyTaskDefinitionBuilderImplementation, NewTaskDefinitionBuilderImplementation, TaskOverrideDefinitionBuilderImplementation, } from "../../../../src/internal/core/tasks/builders.js"; +import { ArgumentType } from "../../../../src/types/arguments.js"; import { TaskDefinitionType } from "../../../../src/types/tasks.js"; describe("Task builders", () => { diff --git a/v-next/hardhat/test/internal/core/tasks/task-manager.ts b/v-next/hardhat/test/internal/core/tasks/task-manager.ts index 33e837fb1a..a8a97b1ac0 100644 --- a/v-next/hardhat/test/internal/core/tasks/task-manager.ts +++ b/v-next/hardhat/test/internal/core/tasks/task-manager.ts @@ -8,16 +8,14 @@ import { } from "@nomicfoundation/hardhat-test-utils"; import { RESERVED_ARGUMENT_NAMES } from "../../../../src/internal/core/arguments.js"; -import { - ArgumentType, - globalOption, -} from "../../../../src/internal/core/config.js"; +import { globalOption } from "../../../../src/internal/core/config.js"; import { createBaseHardhatRuntimeEnvironment } from "../../../../src/internal/core/index.js"; import { EmptyTaskDefinitionBuilderImplementation, NewTaskDefinitionBuilderImplementation, TaskOverrideDefinitionBuilderImplementation, } from "../../../../src/internal/core/tasks/builders.js"; +import { ArgumentType } from "../../../../src/types/arguments.js"; import { TaskDefinitionType } from "../../../../src/types/tasks.js"; /** From b1512b6007fddf1bb159ef862b0959ca8ee3f16e Mon Sep 17 00:00:00 2001 From: Patricio Palladino Date: Sat, 31 Aug 2024 15:08:47 +0000 Subject: [PATCH 02/14] Only re-export HardhatUserConfig from config.ts --- v-next/hardhat/src/config.ts | 2 ++ v-next/hardhat/src/internal/core/config-validation.ts | 2 +- v-next/hardhat/src/internal/core/config.ts | 2 -- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/v-next/hardhat/src/config.ts b/v-next/hardhat/src/config.ts index 7107a18dd2..d7dc5e303b 100644 --- a/v-next/hardhat/src/config.ts +++ b/v-next/hardhat/src/config.ts @@ -7,6 +7,8 @@ import "./internal/builtin-plugins/index.js"; export type * from "./internal/core/config.js"; export * from "./internal/core/config.js"; +export type { HardhatUserConfig } from "./types/config.js"; + /** * Attempts to find the nearest Hardhat config file, starting from the current * working directory. If no config file is found, an error is thrown. diff --git a/v-next/hardhat/src/internal/core/config-validation.ts b/v-next/hardhat/src/internal/core/config-validation.ts index c2a7a9b2e0..9acd11769a 100644 --- a/v-next/hardhat/src/internal/core/config-validation.ts +++ b/v-next/hardhat/src/internal/core/config-validation.ts @@ -1,4 +1,3 @@ -import type { HardhatUserConfig } from "./config.js"; import type { HardhatUserConfigValidationError, HookManager, @@ -19,6 +18,7 @@ import { TaskDefinitionType, type TaskOverrideDefinition, } from "../../types/tasks.js"; +import { HardhatUserConfig } from "../../types/config.js"; function isValidEnumValue( theEnum: Record, diff --git a/v-next/hardhat/src/internal/core/config.ts b/v-next/hardhat/src/internal/core/config.ts index d374f2d5e2..f4e0a04059 100644 --- a/v-next/hardhat/src/internal/core/config.ts +++ b/v-next/hardhat/src/internal/core/config.ts @@ -18,8 +18,6 @@ import { TaskOverrideDefinitionBuilderImplementation, } from "./tasks/builders.js"; -export type { HardhatUserConfig } from "../../types/config.js"; - /** * Creates a configuration variable, which will be fetched at runtime. */ From 0e0b0ccda714f3aef5c9f83397b2fb6beaafa2bb Mon Sep 17 00:00:00 2001 From: Patricio Palladino Date: Sat, 31 Aug 2024 15:45:06 +0000 Subject: [PATCH 03/14] Simplify the config loading logic --- v-next/hardhat/src/cli.ts | 4 + v-next/hardhat/src/config.ts | 22 ----- v-next/hardhat/src/hre.ts | 2 + v-next/hardhat/src/index.ts | 10 ++- v-next/hardhat/src/internal/cli/main.ts | 29 ++++--- .../src/internal/core/config-validation.ts | 2 +- .../src/internal/helpers/config-loading.ts | 81 ++++++++++++++++--- .../config-custom-path/other.config.js | 1 + v-next/hardhat/test/hre/index.ts | 38 +++++++-- 9 files changed, 133 insertions(+), 56 deletions(-) create mode 100644 v-next/hardhat/test/fixture-projects/config-custom-path/other.config.js diff --git a/v-next/hardhat/src/cli.ts b/v-next/hardhat/src/cli.ts index 6a768cd033..8f76b6cf67 100644 --- a/v-next/hardhat/src/cli.ts +++ b/v-next/hardhat/src/cli.ts @@ -1,5 +1,9 @@ import { register } from "tsx/esm/api"; +// Note: We import the builtin plugins' types here, so that any type extension +// they may have gets loaded. +import "./internal/builtin-plugins/index.js"; + // We enable the sourcemaps before loading main, so that everything except this // small file is loaded with sourcemaps enabled. process.setSourceMapsEnabled(true); diff --git a/v-next/hardhat/src/config.ts b/v-next/hardhat/src/config.ts index d7dc5e303b..318ca01085 100644 --- a/v-next/hardhat/src/config.ts +++ b/v-next/hardhat/src/config.ts @@ -1,26 +1,4 @@ -import { findClosestHardhatConfig } from "./internal/helpers/config-loading.js"; - -// Note: We import the builtin plugins' types here, so that any type extension -// they may have gets loaded. -import "./internal/builtin-plugins/index.js"; - export type * from "./internal/core/config.js"; export * from "./internal/core/config.js"; export type { HardhatUserConfig } from "./types/config.js"; - -/** - * Attempts to find the nearest Hardhat config file, starting from the current - * working directory. If no config file is found, an error is thrown. - * - * @returns The path to the nearest Hardhat config file. - */ -export async function resolveHardhatConfigPath(): Promise { - const configPath = process.env.HARDHAT_CONFIG; - - if (configPath !== undefined) { - return configPath; - } - - return findClosestHardhatConfig(); -} diff --git a/v-next/hardhat/src/hre.ts b/v-next/hardhat/src/hre.ts index 59678b3c31..2beb11559e 100644 --- a/v-next/hardhat/src/hre.ts +++ b/v-next/hardhat/src/hre.ts @@ -12,6 +12,8 @@ import { resolvePluginList, } from "./internal/core/index.js"; +export { resolveHardhatConfigPath } from "./internal/helpers/config-loading.js"; + /** * Creates an instances of the Hardhat Runtime Environment. * diff --git a/v-next/hardhat/src/index.ts b/v-next/hardhat/src/index.ts index 75e3334c7c..05a29a2c25 100644 --- a/v-next/hardhat/src/index.ts +++ b/v-next/hardhat/src/index.ts @@ -5,18 +5,24 @@ import type { HardhatRuntimeEnvironment } from "./types/hre.js"; import type { TaskManager } from "./types/tasks.js"; import type { UserInterruptionManager } from "./types/user-interruptions.js"; -import { resolveHardhatConfigPath } from "./config.js"; import { createHardhatRuntimeEnvironment } from "./hre.js"; import { resolveProjectRoot } from "./internal/core/index.js"; import { getGlobalHardhatRuntimeEnvironment, setGlobalHardhatRuntimeEnvironment, } from "./internal/global-hre-instance.js"; -import { importUserConfig } from "./internal/helpers/config-loading.js"; +import { + importUserConfig, + resolveHardhatConfigPath, +} from "./internal/helpers/config-loading.js"; let maybeHre: HardhatRuntimeEnvironment | undefined = getGlobalHardhatRuntimeEnvironment(); +// Note: We import the builtin plugins' types here, so that any type extension +// they may have gets loaded. +import "./internal/builtin-plugins/index.js"; + if (maybeHre === undefined) { /* eslint-disable no-restricted-syntax -- Allow top-level await here */ const configPath = await resolveHardhatConfigPath(); diff --git a/v-next/hardhat/src/internal/cli/main.ts b/v-next/hardhat/src/internal/cli/main.ts index 68946f5ca7..8b5f1868e8 100644 --- a/v-next/hardhat/src/internal/cli/main.ts +++ b/v-next/hardhat/src/internal/cli/main.ts @@ -10,11 +10,9 @@ import { assertHardhatInvariant, } from "@ignored/hardhat-vnext-errors"; import { isCi } from "@ignored/hardhat-vnext-utils/ci"; -import { getRealPath } from "@ignored/hardhat-vnext-utils/fs"; import { kebabToCamelCase } from "@ignored/hardhat-vnext-utils/string"; import debug from "debug"; -import { resolveHardhatConfigPath } from "../../config.js"; import { createHardhatRuntimeEnvironment } from "../../hre.js"; import { ArgumentType, @@ -30,7 +28,10 @@ import { resolveProjectRoot, } from "../core/index.js"; import { setGlobalHardhatRuntimeEnvironment } from "../global-hre-instance.js"; -import { importUserConfig } from "../helpers/config-loading.js"; +import { + importUserConfig, + resolveHardhatConfigPath, +} from "../helpers/config-loading.js"; import { printErrorMessages } from "./error-handler.js"; import { getGlobalHelpString } from "./helpers/getGlobalHelpString.js"; @@ -73,17 +74,13 @@ export async function main( log("Retrieved telemetry consent"); - if (builtinGlobalOptions.configPath === undefined) { - builtinGlobalOptions.configPath = await resolveHardhatConfigPath(); - - log("Resolved config path"); - } - - const projectRoot = await resolveProjectRoot( - await getRealPath(builtinGlobalOptions.configPath), + const configPath = await resolveHardhatConfigPath( + builtinGlobalOptions.configPath, ); - const userConfig = await importUserConfig(builtinGlobalOptions.configPath); + const projectRoot = await resolveProjectRoot(configPath); + + const userConfig = await importUserConfig(configPath); log("User config imported"); @@ -97,10 +94,12 @@ export async function main( const pluginGlobalOptionDefinitions = buildGlobalOptionDefinitions(resolvedPlugins); + const globalOptionDefinitions = new Map([ ...BUILTIN_GLOBAL_OPTIONS_DEFINITIONS, ...pluginGlobalOptionDefinitions, ]); + const userProvidedGlobalOptions = await parseGlobalOptions( globalOptionDefinitions, cliArguments, @@ -111,7 +110,11 @@ export async function main( const hre = await createHardhatRuntimeEnvironment( userConfig, - { ...builtinGlobalOptions, ...userProvidedGlobalOptions }, + { + ...builtinGlobalOptions, + config: configPath, + ...userProvidedGlobalOptions, + }, projectRoot, { resolvedPlugins, globalOptionDefinitions }, ); diff --git a/v-next/hardhat/src/internal/core/config-validation.ts b/v-next/hardhat/src/internal/core/config-validation.ts index 9acd11769a..989f0d83cc 100644 --- a/v-next/hardhat/src/internal/core/config-validation.ts +++ b/v-next/hardhat/src/internal/core/config-validation.ts @@ -1,3 +1,4 @@ +import type { HardhatUserConfig } from "../../types/config.js"; import type { HardhatUserConfigValidationError, HookManager, @@ -18,7 +19,6 @@ import { TaskDefinitionType, type TaskOverrideDefinition, } from "../../types/tasks.js"; -import { HardhatUserConfig } from "../../types/config.js"; function isValidEnumValue( theEnum: Record, diff --git a/v-next/hardhat/src/internal/helpers/config-loading.ts b/v-next/hardhat/src/internal/helpers/config-loading.ts index 474ae5366f..8e9374c35e 100644 --- a/v-next/hardhat/src/internal/helpers/config-loading.ts +++ b/v-next/hardhat/src/internal/helpers/config-loading.ts @@ -4,9 +4,47 @@ import { isAbsolute, resolve } from "node:path"; import { pathToFileURL } from "node:url"; import { HardhatError } from "@ignored/hardhat-vnext-errors"; -import { findUp } from "@ignored/hardhat-vnext-utils/fs"; +import { exists, findUp } from "@ignored/hardhat-vnext-utils/fs"; import { isObject } from "@ignored/hardhat-vnext-utils/lang"; +import debug from "debug"; +const log = debug("hardhat:core:config-loading"); + +/** + * Resolves the path to the Hardhat config file using these rules: + * - If the user provided a path, that path is returned. + * - Otherwise, if the HARDHAT_CONFIG env var is set, that path is returned. + * - Otherwise, the closest Hardhat config file to the current working + * directory is returned. + * + * @param userProvidedPath An optional path to the Hardhat config file, provided + * by the user. + * @returns The path to the Hardhat config file, as an absolute path. + * @throws HardhatError If no Hardhat config file can be found. + */ +export async function resolveHardhatConfigPath( + userProvidedPath?: string, +): Promise { + const configPath = userProvidedPath ?? process.env.HARDHAT_CONFIG; + + if (configPath !== undefined) { + return normalizeConfigPath(configPath); + } + + if (process.env.HARDHAT_CONFIG !== undefined) { + log("Using config file path from the HARDHAT_CONFIG env var"); + return normalizeConfigPath(process.env.HARDHAT_CONFIG); + } + + return findClosestHardhatConfig(); +} + +/** + * Finds the closest Hardhat config file to the current working directory. + * + * @returns The absolute path to the closest Hardhat config file. + * @throw HardhatError if no Hardhat config file can be found. + */ export async function findClosestHardhatConfig(): Promise { let hardhatConfigPath = await findUp("hardhat.config.ts"); @@ -23,20 +61,15 @@ export async function findClosestHardhatConfig(): Promise { throw new HardhatError(HardhatError.ERRORS.GENERAL.NO_CONFIG_FILE_FOUND); } +/** + * Imports the user config and returns it. + * @param configPath The path to the config file. + * @returns The user config. + */ export async function importUserConfig( configPath: string, ): Promise { - const normalizedPath = isAbsolute(configPath) - ? configPath - : resolve(process.cwd(), configPath); - - const { exists } = await import("@ignored/hardhat-vnext-utils/fs"); - - if (!(await exists(normalizedPath))) { - throw new HardhatError(HardhatError.ERRORS.GENERAL.INVALID_CONFIG_PATH, { - configPath, - }); - } + const normalizedPath = await normalizeConfigPath(configPath); const imported = await import(pathToFileURL(normalizedPath).href); @@ -48,7 +81,7 @@ export async function importUserConfig( const config = imported.default; - if (!isObject(config) || config === null) { + if (!isObject(config)) { throw new HardhatError(HardhatError.ERRORS.GENERAL.INVALID_CONFIG_OBJECT, { configPath, }); @@ -56,3 +89,25 @@ export async function importUserConfig( return config; } + +/** + * Returns an abolute version of the config path, throwing if the path + * doesn't exist. + * + * @param configPath The path to the config file. + * @returns The absolute path to the config file. + * @throws HardhatError if the path doesn't exist. + */ +async function normalizeConfigPath(configPath: string): Promise { + const normalizedPath = isAbsolute(configPath) + ? configPath + : resolve(process.cwd(), configPath); + + if (!(await exists(normalizedPath))) { + throw new HardhatError(HardhatError.ERRORS.GENERAL.INVALID_CONFIG_PATH, { + configPath, + }); + } + + return normalizedPath; +} diff --git a/v-next/hardhat/test/fixture-projects/config-custom-path/other.config.js b/v-next/hardhat/test/fixture-projects/config-custom-path/other.config.js new file mode 100644 index 0000000000..ff8b4c5632 --- /dev/null +++ b/v-next/hardhat/test/fixture-projects/config-custom-path/other.config.js @@ -0,0 +1 @@ +export default {}; diff --git a/v-next/hardhat/test/hre/index.ts b/v-next/hardhat/test/hre/index.ts index 5a3722921a..8376ec55f3 100644 --- a/v-next/hardhat/test/hre/index.ts +++ b/v-next/hardhat/test/hre/index.ts @@ -2,12 +2,12 @@ import assert from "node:assert/strict"; import { afterEach, describe, it } from "node:test"; import { HardhatError } from "@ignored/hardhat-vnext-errors"; +import { getRealPath } from "@ignored/hardhat-vnext-utils/fs"; import { assertRejectsWithHardhatError, useFixtureProject, } from "@nomicfoundation/hardhat-test-utils"; -import { resolveHardhatConfigPath } from "../../src/config.js"; import { createHardhatRuntimeEnvironment } from "../../src/hre.js"; import { builtinPlugins } from "../../src/internal/builtin-plugins/index.js"; import { @@ -15,6 +15,7 @@ import { resetGlobalHardhatRuntimeEnvironment, setGlobalHardhatRuntimeEnvironment, } from "../../src/internal/global-hre-instance.js"; +import { resolveHardhatConfigPath } from "../../src/internal/helpers/config-loading.js"; describe("HRE", () => { afterEach(() => { @@ -58,12 +59,39 @@ describe("HRE", () => { describe("config loading", () => { describe("resolveConfigPath", async () => { - it("should return the HARDHAT_CONFIG env variable if it is set", async () => { - process.env.HARDHAT_CONFIG = "env.config.js"; + describe("With custom config path", () => { + useFixtureProject("config-custom-path"); - assert.equal(await resolveHardhatConfigPath(), "env.config.js"); + it("should return the HARDHAT_CONFIG env variable if it is set", async () => { + try { + // We set the env var to a hardhat config and then clean it up + process.env.HARDHAT_CONFIG = "other.config.js"; - delete process.env.HARDHAT_CONFIG; + assert.equal( + await resolveHardhatConfigPath(), + await getRealPath("other.config.js"), + ); + } finally { + delete process.env.HARDHAT_CONFIG; + } + }); + + it("should noramlize and return the provided path", async () => { + assert.equal( + await resolveHardhatConfigPath("other.config.js"), + await getRealPath("other.config.js"), + ); + }); + + it("should throw if the config file is not found", async () => { + await assertRejectsWithHardhatError( + resolveHardhatConfigPath("non-existent.config.js"), + HardhatError.ERRORS.GENERAL.INVALID_CONFIG_PATH, + { + configPath: "non-existent.config.js", + }, + ); + }); }); it("should throw if the config file is not found", async () => { From d6e8c14ba2467833506afd5423bc9c9dfc088e99 Mon Sep 17 00:00:00 2001 From: Patricio Palladino Date: Sat, 31 Aug 2024 15:58:04 +0000 Subject: [PATCH 04/14] Remove core's barrel file --- v-next/hardhat/src/hre.ts | 12 +- v-next/hardhat/src/index.ts | 2 +- v-next/hardhat/src/internal/cli/main.ts | 10 +- v-next/hardhat/src/internal/core/index.ts | 42 - .../configuration-variables.ts | 2 +- .../test/internal/core/tasks/task-manager.ts | 2496 +++++++++-------- .../user-interruptions-manager.ts | 2 +- 7 files changed, 1346 insertions(+), 1220 deletions(-) delete mode 100644 v-next/hardhat/src/internal/core/index.ts diff --git a/v-next/hardhat/src/hre.ts b/v-next/hardhat/src/hre.ts index 2beb11559e..8bc296ef95 100644 --- a/v-next/hardhat/src/hre.ts +++ b/v-next/hardhat/src/hre.ts @@ -5,12 +5,12 @@ import type { HardhatRuntimeEnvironment } from "./types/hre.js"; import { BUILTIN_GLOBAL_OPTIONS_DEFINITIONS } from "./internal/builtin-global-options.js"; import { builtinPlugins } from "./internal/builtin-plugins/index.js"; -import { resolveProjectRoot } from "./internal/core/hre.js"; +import { buildGlobalOptionDefinitions } from "./internal/core/global-options.js"; import { - buildGlobalOptionDefinitions, - createBaseHardhatRuntimeEnvironment, - resolvePluginList, -} from "./internal/core/index.js"; + HardhatRuntimeEnvironmentImplementation, + resolveProjectRoot, +} from "./internal/core/hre.js"; +import { resolvePluginList } from "./internal/core/plugins/resolve-plugin-list.js"; export { resolveHardhatConfigPath } from "./internal/helpers/config-loading.js"; @@ -57,7 +57,7 @@ export async function createHardhatRuntimeEnvironment( unsafeOptions.globalOptionDefinitions = globalOptionDefinitions; } - return createBaseHardhatRuntimeEnvironment( + return HardhatRuntimeEnvironmentImplementation.create( config, userProvidedGlobalOptions, resolvedProjectRoot, diff --git a/v-next/hardhat/src/index.ts b/v-next/hardhat/src/index.ts index 05a29a2c25..4e19931670 100644 --- a/v-next/hardhat/src/index.ts +++ b/v-next/hardhat/src/index.ts @@ -6,7 +6,7 @@ import type { TaskManager } from "./types/tasks.js"; import type { UserInterruptionManager } from "./types/user-interruptions.js"; import { createHardhatRuntimeEnvironment } from "./hre.js"; -import { resolveProjectRoot } from "./internal/core/index.js"; +import { resolveProjectRoot } from "./internal/core/hre.js"; import { getGlobalHardhatRuntimeEnvironment, setGlobalHardhatRuntimeEnvironment, diff --git a/v-next/hardhat/src/internal/cli/main.ts b/v-next/hardhat/src/internal/cli/main.ts index 8b5f1868e8..79acf5a789 100644 --- a/v-next/hardhat/src/internal/cli/main.ts +++ b/v-next/hardhat/src/internal/cli/main.ts @@ -21,12 +21,10 @@ import { } from "../../types/arguments.js"; import { BUILTIN_GLOBAL_OPTIONS_DEFINITIONS } from "../builtin-global-options.js"; import { builtinPlugins } from "../builtin-plugins/index.js"; -import { - buildGlobalOptionDefinitions, - parseArgumentValue, - resolvePluginList, - resolveProjectRoot, -} from "../core/index.js"; +import { parseArgumentValue } from "../core/arguments.js"; +import { buildGlobalOptionDefinitions } from "../core/global-options.js"; +import { resolveProjectRoot } from "../core/hre.js"; +import { resolvePluginList } from "../core/plugins/resolve-plugin-list.js"; import { setGlobalHardhatRuntimeEnvironment } from "../global-hre-instance.js"; import { importUserConfig, diff --git a/v-next/hardhat/src/internal/core/index.ts b/v-next/hardhat/src/internal/core/index.ts deleted file mode 100644 index d2476d9cee..0000000000 --- a/v-next/hardhat/src/internal/core/index.ts +++ /dev/null @@ -1,42 +0,0 @@ -import type { UnsafeHardhatRuntimeEnvironmentOptions } from "./types.js"; -import type { HardhatUserConfig } from "../../types/config.js"; -import type { GlobalOptions } from "../../types/global-options.js"; -import type { HardhatRuntimeEnvironment } from "../../types/hre.js"; - -import { HardhatRuntimeEnvironmentImplementation } from "./hre.js"; - -/** - * Creates an instances of the Hardhat Runtime Environment without any of the - * built-in plugins. - * - * To get the built-in plugins, use `createHardhatRuntimeEnvironment` from - * `hardhat/hre` instead. - * - * @param config - The user's Hardhat configuration. - * @param userProvidedGlobalOptions - The global options provided by the - * user. - * @param projectRoot - The root of the Hardhat project. Hardhat expects this - * to be the root of the npm project containing your config file. If none is - * provided, it will use the root of the npm project that contains the CWD. - * @param unsafeOptions - Options used to bypass some initialization, to avoid - * redoing it in the CLI. Should only be used in the official CLI. - * @returns The Hardhat Runtime Environment. - */ -export async function createBaseHardhatRuntimeEnvironment( - config: HardhatUserConfig, - userProvidedGlobalOptions: Partial = {}, - projectRoot?: string, - unsafeOptions?: UnsafeHardhatRuntimeEnvironmentOptions, -): Promise { - return HardhatRuntimeEnvironmentImplementation.create( - config, - userProvidedGlobalOptions, - projectRoot, - unsafeOptions, - ); -} - -export { resolveProjectRoot } from "./hre.js"; -export { parseArgumentValue } from "./arguments.js"; -export { resolvePluginList } from "./plugins/resolve-plugin-list.js"; -export { buildGlobalOptionDefinitions } from "./global-options.js"; diff --git a/v-next/hardhat/test/internal/core/configuration-variables/configuration-variables.ts b/v-next/hardhat/test/internal/core/configuration-variables/configuration-variables.ts index d3cfbec6e4..e6cb595ff4 100644 --- a/v-next/hardhat/test/internal/core/configuration-variables/configuration-variables.ts +++ b/v-next/hardhat/test/internal/core/configuration-variables/configuration-variables.ts @@ -6,7 +6,7 @@ import { assertRejectsWithHardhatError } from "@nomicfoundation/hardhat-test-uti import { ResolvedConfigurationVariableImplementation } from "../../../../src/internal/core/configuration-variables.js"; import { HookManagerImplementation } from "../../../../src/internal/core/hook-manager.js"; -import { resolveProjectRoot } from "../../../../src/internal/core/index.js"; +import { resolveProjectRoot } from "../../../../src/internal/core/hre.js"; import { UserInterruptionManagerImplementation } from "../../../../src/internal/core/user-interruptions.js"; describe("ResolvedConfigurationVariable", () => { diff --git a/v-next/hardhat/test/internal/core/tasks/task-manager.ts b/v-next/hardhat/test/internal/core/tasks/task-manager.ts index a8a97b1ac0..d73aecc47d 100644 --- a/v-next/hardhat/test/internal/core/tasks/task-manager.ts +++ b/v-next/hardhat/test/internal/core/tasks/task-manager.ts @@ -9,7 +9,7 @@ import { import { RESERVED_ARGUMENT_NAMES } from "../../../../src/internal/core/arguments.js"; import { globalOption } from "../../../../src/internal/core/config.js"; -import { createBaseHardhatRuntimeEnvironment } from "../../../../src/internal/core/index.js"; +import { HardhatRuntimeEnvironmentImplementation } from "../../../../src/internal/core/hre.js"; import { EmptyTaskDefinitionBuilderImplementation, NewTaskDefinitionBuilderImplementation, @@ -30,47 +30,50 @@ import { TaskDefinitionType } from "../../../../src/types/tasks.js"; */ describe("TaskManagerImplementation", () => { it("should initialize the task manager with an empty set of tasks if no plugins or tasks are provided", async () => { - const hre = await createBaseHardhatRuntimeEnvironment({}); + const hre = await HardhatRuntimeEnvironmentImplementation.create({}, {}); assert.equal(hre.tasks.rootTasks.size, 0); }); it("should initialize the task manager with the tasks from the plugins", async () => { - const hre = await createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .addOption({ name: "arg1", defaultValue: "default" }) - .setAction(() => {}) - .build(), - new NewTaskDefinitionBuilderImplementation("task2") - .addFlag({ name: "flag1" }) - .setAction(() => {}) - .build(), - ], - globalOptions: [ - globalOption({ - name: "globalOption1", - description: "", - type: ArgumentType.STRING, - defaultValue: "", - }), - ], - }, - { - id: "plugin2", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task3") - .addPositionalArgument({ name: "posArg1" }) - .addVariadicArgument({ name: "varArg1" }) - .setAction(() => {}) - .build(), - ], - }, - ], - }); + const hre = await HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .addOption({ name: "arg1", defaultValue: "default" }) + .setAction(() => {}) + .build(), + new NewTaskDefinitionBuilderImplementation("task2") + .addFlag({ name: "flag1" }) + .setAction(() => {}) + .build(), + ], + globalOptions: [ + globalOption({ + name: "globalOption1", + description: "", + type: ArgumentType.STRING, + defaultValue: "", + }), + ], + }, + { + id: "plugin2", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task3") + .addPositionalArgument({ name: "posArg1" }) + .addVariadicArgument({ name: "varArg1" }) + .setAction(() => {}) + .build(), + ], + }, + ], + }, + {}, + ); // task1 in plugin1 should be available const task1 = hre.tasks.getTask("task1"); @@ -95,23 +98,26 @@ describe("TaskManagerImplementation", () => { }); it("should initialize the task manager with the tasks from the config", async () => { - const hre = await createBaseHardhatRuntimeEnvironment({ - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .addOption({ name: "arg1", defaultValue: "default" }) - .setAction(() => {}) - .build(), - new NewTaskDefinitionBuilderImplementation("task2") - .addFlag({ name: "flag1" }) - .setAction(() => {}) - .build(), - new NewTaskDefinitionBuilderImplementation("task3") - .addPositionalArgument({ name: "posArg1" }) - .addVariadicArgument({ name: "varArg1" }) - .setAction(() => {}) - .build(), - ], - }); + const hre = await HardhatRuntimeEnvironmentImplementation.create( + { + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .addOption({ name: "arg1", defaultValue: "default" }) + .setAction(() => {}) + .build(), + new NewTaskDefinitionBuilderImplementation("task2") + .addFlag({ name: "flag1" }) + .setAction(() => {}) + .build(), + new NewTaskDefinitionBuilderImplementation("task3") + .addPositionalArgument({ name: "posArg1" }) + .addVariadicArgument({ name: "varArg1" }) + .setAction(() => {}) + .build(), + ], + }, + {}, + ); // task1 in plugin1 should be available const task1 = hre.tasks.getTask("task1"); @@ -136,30 +142,33 @@ describe("TaskManagerImplementation", () => { }); it("should override a task within the same plugin", async () => { - const hre = await createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .setDescription("description1") - .addOption({ name: "arg1", defaultValue: "default" }) - .addFlag({ name: "flag1" }) - .addPositionalArgument({ name: "posArg1" }) - .addVariadicArgument({ name: "varArg1" }) - .setAction(() => {}) - .build(), - // overriding task1 with a new description and arguments - new TaskOverrideDefinitionBuilderImplementation("task1") - .setDescription("description2") - .addOption({ name: "arg2", defaultValue: "default" }) - .addFlag({ name: "flag2" }) - .setAction(() => {}) - .build(), - ], - }, - ], - }); + const hre = await HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .setDescription("description1") + .addOption({ name: "arg1", defaultValue: "default" }) + .addFlag({ name: "flag1" }) + .addPositionalArgument({ name: "posArg1" }) + .addVariadicArgument({ name: "varArg1" }) + .setAction(() => {}) + .build(), + // overriding task1 with a new description and arguments + new TaskOverrideDefinitionBuilderImplementation("task1") + .setDescription("description2") + .addOption({ name: "arg2", defaultValue: "default" }) + .addFlag({ name: "flag2" }) + .setAction(() => {}) + .build(), + ], + }, + ], + }, + {}, + ); const task1 = hre.tasks.getTask("task1"); assert.deepEqual(task1.id, ["task1"]); @@ -186,35 +195,38 @@ describe("TaskManagerImplementation", () => { }); it("should override a task from a different plugin", async () => { - const hre = await createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .setDescription("description1") - .addOption({ name: "arg1", defaultValue: "default" }) - .addFlag({ name: "flag1" }) - .addPositionalArgument({ name: "posArg1" }) - .addVariadicArgument({ name: "varArg1" }) - .setAction(() => {}) - .build(), - ], - }, - { - id: "plugin2", - tasks: [ - // overriding task1 with a new description and arguments - new TaskOverrideDefinitionBuilderImplementation("task1") - .setDescription("description2") - .addOption({ name: "arg2", defaultValue: "default" }) - .addFlag({ name: "flag2" }) - .setAction(() => {}) - .build(), - ], - }, - ], - }); + const hre = await HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .setDescription("description1") + .addOption({ name: "arg1", defaultValue: "default" }) + .addFlag({ name: "flag1" }) + .addPositionalArgument({ name: "posArg1" }) + .addVariadicArgument({ name: "varArg1" }) + .setAction(() => {}) + .build(), + ], + }, + { + id: "plugin2", + tasks: [ + // overriding task1 with a new description and arguments + new TaskOverrideDefinitionBuilderImplementation("task1") + .setDescription("description2") + .addOption({ name: "arg2", defaultValue: "default" }) + .addFlag({ name: "flag2" }) + .setAction(() => {}) + .build(), + ], + }, + ], + }, + {}, + ); const task1 = hre.tasks.getTask("task1"); assert.deepEqual(task1.id, ["task1"]); @@ -241,42 +253,45 @@ describe("TaskManagerImplementation", () => { }); it("should override the same task multiple times", async () => { - const hre = await createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .setDescription("description1") - .addOption({ name: "arg1", defaultValue: "default" }) - .addFlag({ name: "flag1" }) - .addPositionalArgument({ name: "posArg1" }) - .addVariadicArgument({ name: "varArg1" }) - .setAction(() => {}) - .build(), - // overriding task1 with a new description and arguments - new TaskOverrideDefinitionBuilderImplementation("task1") - .setDescription("description2") - .addOption({ name: "arg2", defaultValue: "default" }) - .addFlag({ name: "flag2" }) - .setAction(() => {}) - .build(), - ], - }, - { - id: "plugin2", - tasks: [ - // overriding task1 with a new description and arguments - new TaskOverrideDefinitionBuilderImplementation("task1") - .setDescription("description3") - .addOption({ name: "arg3", defaultValue: "default" }) - .addFlag({ name: "flag3" }) - .setAction(() => {}) - .build(), - ], - }, - ], - }); + const hre = await HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .setDescription("description1") + .addOption({ name: "arg1", defaultValue: "default" }) + .addFlag({ name: "flag1" }) + .addPositionalArgument({ name: "posArg1" }) + .addVariadicArgument({ name: "varArg1" }) + .setAction(() => {}) + .build(), + // overriding task1 with a new description and arguments + new TaskOverrideDefinitionBuilderImplementation("task1") + .setDescription("description2") + .addOption({ name: "arg2", defaultValue: "default" }) + .addFlag({ name: "flag2" }) + .setAction(() => {}) + .build(), + ], + }, + { + id: "plugin2", + tasks: [ + // overriding task1 with a new description and arguments + new TaskOverrideDefinitionBuilderImplementation("task1") + .setDescription("description3") + .addOption({ name: "arg3", defaultValue: "default" }) + .addFlag({ name: "flag3" }) + .setAction(() => {}) + .build(), + ], + }, + ], + }, + {}, + ); const task1 = hre.tasks.getTask("task1"); assert.deepEqual(task1.id, ["task1"]); @@ -306,19 +321,22 @@ describe("TaskManagerImplementation", () => { }); it("should add an empty task", async () => { - const hre = await createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new EmptyTaskDefinitionBuilderImplementation( - "task1", - "description1", - ).build(), - ], - }, - ], - }); + const hre = await HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new EmptyTaskDefinitionBuilderImplementation( + "task1", + "description1", + ).build(), + ], + }, + ], + }, + {}, + ); const task1 = hre.tasks.getTask("task1"); assert.deepEqual(task1.id, ["task1"]); @@ -326,37 +344,40 @@ describe("TaskManagerImplementation", () => { }); it("should add subtasks", async () => { - const hre = await createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new EmptyTaskDefinitionBuilderImplementation( - "task1", - "description1", - ).build(), - // adds a subtask to the empty task - new NewTaskDefinitionBuilderImplementation(["task1", "subtask1"]) - .setAction(() => {}) - .build(), - ], - }, + const hre = await HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new EmptyTaskDefinitionBuilderImplementation( + "task1", + "description1", + ).build(), + // adds a subtask to the empty task + new NewTaskDefinitionBuilderImplementation(["task1", "subtask1"]) + .setAction(() => {}) + .build(), + ], + }, - { - id: "plugin2", - tasks: [ - // adds a subtask to the non-empty task - new NewTaskDefinitionBuilderImplementation([ - "task1", - "subtask1", - "subsubtask1", - ]) - .setAction(() => {}) - .build(), - ], - }, - ], - }); + { + id: "plugin2", + tasks: [ + // adds a subtask to the non-empty task + new NewTaskDefinitionBuilderImplementation([ + "task1", + "subtask1", + "subsubtask1", + ]) + .setAction(() => {}) + .build(), + ], + }, + ], + }, + {}, + ); const task1 = hre.tasks.getTask("task1"); assert.deepEqual(task1.id, ["task1"]); @@ -383,30 +404,33 @@ describe("TaskManagerImplementation", () => { describe("errors", () => { it("should throw if there's a global option with the same name as a task option", async () => { await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .addOption({ name: "arg1", defaultValue: "default" }) - .setAction(() => {}) - .build(), - ], - }, - { - id: "plugin2", - globalOptions: [ - globalOption({ - name: "arg1", - description: "", - type: ArgumentType.STRING, - defaultValue: "", - }), - ], - }, - ], - }), + HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .addOption({ name: "arg1", defaultValue: "default" }) + .setAction(() => {}) + .build(), + ], + }, + { + id: "plugin2", + globalOptions: [ + globalOption({ + name: "arg1", + description: "", + type: ArgumentType.STRING, + defaultValue: "", + }), + ], + }, + ], + }, + {}, + ), HardhatError.ERRORS.TASK_DEFINITIONS.TASK_OPTION_ALREADY_DEFINED, { @@ -420,32 +444,35 @@ describe("TaskManagerImplementation", () => { it("should throw if there's a global option with the same name as a task positional argument", async () => { await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .addPositionalArgument({ name: "arg1" }) - .setAction(() => {}) - .build(), - ], - }, - { - id: "plugin2", - globalOptions: [ - globalOption({ - name: "arg1", - description: "", - type: ArgumentType.STRING, - defaultValue: "", - }), - ], - }, - ], - }), - - HardhatError.ERRORS.TASK_DEFINITIONS.TASK_OPTION_ALREADY_DEFINED, + HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .addPositionalArgument({ name: "arg1" }) + .setAction(() => {}) + .build(), + ], + }, + { + id: "plugin2", + globalOptions: [ + globalOption({ + name: "arg1", + description: "", + type: ArgumentType.STRING, + defaultValue: "", + }), + ], + }, + ], + }, + {}, + ), + + HardhatError.ERRORS.TASK_DEFINITIONS.TASK_OPTION_ALREADY_DEFINED, { actorFragment: "Plugin plugin1 is", task: "task1", @@ -457,24 +484,27 @@ describe("TaskManagerImplementation", () => { it("should throw if trying to add a task with an empty id", async () => { await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - // Manually creating a task as the builder doesn't allow empty ids - { - type: TaskDefinitionType.NEW_TASK, - id: [], // empty id - description: "", - action: () => {}, - options: {}, - positionalArguments: [], - }, - ], - }, - ], - }), + HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + // Manually creating a task as the builder doesn't allow empty ids + { + type: TaskDefinitionType.NEW_TASK, + id: [], // empty id + description: "", + action: () => {}, + options: {}, + positionalArguments: [], + }, + ], + }, + ], + }, + {}, + ), HardhatError.ERRORS.TASK_DEFINITIONS.EMPTY_TASK_ID, {}, ); @@ -482,22 +512,25 @@ describe("TaskManagerImplementation", () => { it("should throw if trying to add a subtask for a task that doesn't exist", async () => { await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation([ - "task1", - "subtask1", - ]) - .addOption({ name: "arg1", defaultValue: "default" }) - .setAction(() => {}) - .build(), - ], - }, - ], - }), + HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation([ + "task1", + "subtask1", + ]) + .addOption({ name: "arg1", defaultValue: "default" }) + .setAction(() => {}) + .build(), + ], + }, + ], + }, + {}, + ), HardhatError.ERRORS.TASK_DEFINITIONS.SUBTASK_WITHOUT_PARENT, { task: "task1", @@ -508,28 +541,31 @@ describe("TaskManagerImplementation", () => { it("should throw if trying to add a task that already exists", async () => { await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .addOption({ name: "arg1", defaultValue: "default" }) - .setAction(() => {}) - .build(), - ], - }, - { - id: "plugin2", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .addOption({ name: "arg2", defaultValue: "default" }) - .setAction(() => {}) - .build(), - ], - }, - ], - }), + HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .addOption({ name: "arg1", defaultValue: "default" }) + .setAction(() => {}) + .build(), + ], + }, + { + id: "plugin2", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .addOption({ name: "arg2", defaultValue: "default" }) + .setAction(() => {}) + .build(), + ], + }, + ], + }, + {}, + ), HardhatError.ERRORS.TASK_DEFINITIONS.TASK_ALREADY_DEFINED, { actorFragment: "Plugin plugin2 is", @@ -542,18 +578,21 @@ describe("TaskManagerImplementation", () => { it("should throw if trying to override a task that doesn't exist", async () => { // task1 will not be found as it's not defined await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new TaskOverrideDefinitionBuilderImplementation("task1") - .setAction(() => {}) - .build(), - ], - }, - ], - }), + HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new TaskOverrideDefinitionBuilderImplementation("task1") + .setAction(() => {}) + .build(), + ], + }, + ], + }, + {}, + ), HardhatError.ERRORS.TASK_DEFINITIONS.TASK_NOT_FOUND, { task: "task1", @@ -564,28 +603,31 @@ describe("TaskManagerImplementation", () => { it("should throw if trying to override a task and there is a name clash with an existing option", async () => { // added argument clash with an existing option await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .addOption({ name: "arg1", defaultValue: "default" }) - .setAction(() => {}) - .build(), - ], - }, - { - id: "plugin2", - tasks: [ - new TaskOverrideDefinitionBuilderImplementation("task1") - .addOption({ name: "arg1", defaultValue: "default" }) - .setAction(() => {}) - .build(), - ], - }, - ], - }), + HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .addOption({ name: "arg1", defaultValue: "default" }) + .setAction(() => {}) + .build(), + ], + }, + { + id: "plugin2", + tasks: [ + new TaskOverrideDefinitionBuilderImplementation("task1") + .addOption({ name: "arg1", defaultValue: "default" }) + .setAction(() => {}) + .build(), + ], + }, + ], + }, + {}, + ), HardhatError.ERRORS.TASK_DEFINITIONS .TASK_OVERRIDE_OPTION_ALREADY_DEFINED, { @@ -597,28 +639,31 @@ describe("TaskManagerImplementation", () => { // added flag clash with an existing option await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .addOption({ name: "arg1", defaultValue: "default" }) - .setAction(() => {}) - .build(), - ], - }, - { - id: "plugin2", - tasks: [ - new TaskOverrideDefinitionBuilderImplementation("task1") - .addFlag({ name: "arg1" }) - .setAction(() => {}) - .build(), - ], - }, - ], - }), + HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .addOption({ name: "arg1", defaultValue: "default" }) + .setAction(() => {}) + .build(), + ], + }, + { + id: "plugin2", + tasks: [ + new TaskOverrideDefinitionBuilderImplementation("task1") + .addFlag({ name: "arg1" }) + .setAction(() => {}) + .build(), + ], + }, + ], + }, + {}, + ), HardhatError.ERRORS.TASK_DEFINITIONS .TASK_OVERRIDE_OPTION_ALREADY_DEFINED, { @@ -632,28 +677,31 @@ describe("TaskManagerImplementation", () => { it("should throw if trying to override a task and there is a name clash with an exising flag argument", async () => { // added argument clash with an existing flag await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .addFlag({ name: "flag1" }) - .setAction(() => {}) - .build(), - ], - }, - { - id: "plugin2", - tasks: [ - new TaskOverrideDefinitionBuilderImplementation("task1") - .addOption({ name: "flag1", defaultValue: "default" }) - .setAction(() => {}) - .build(), - ], - }, - ], - }), + HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .addFlag({ name: "flag1" }) + .setAction(() => {}) + .build(), + ], + }, + { + id: "plugin2", + tasks: [ + new TaskOverrideDefinitionBuilderImplementation("task1") + .addOption({ name: "flag1", defaultValue: "default" }) + .setAction(() => {}) + .build(), + ], + }, + ], + }, + {}, + ), HardhatError.ERRORS.TASK_DEFINITIONS .TASK_OVERRIDE_OPTION_ALREADY_DEFINED, { @@ -665,28 +713,31 @@ describe("TaskManagerImplementation", () => { // added flag clash with an existing flag await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .addFlag({ name: "flag1" }) - .setAction(() => {}) - .build(), - ], - }, - { - id: "plugin2", - tasks: [ - new TaskOverrideDefinitionBuilderImplementation("task1") - .addFlag({ name: "flag1" }) - .setAction(() => {}) - .build(), - ], - }, - ], - }), + HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .addFlag({ name: "flag1" }) + .setAction(() => {}) + .build(), + ], + }, + { + id: "plugin2", + tasks: [ + new TaskOverrideDefinitionBuilderImplementation("task1") + .addFlag({ name: "flag1" }) + .setAction(() => {}) + .build(), + ], + }, + ], + }, + {}, + ), HardhatError.ERRORS.TASK_DEFINITIONS .TASK_OVERRIDE_OPTION_ALREADY_DEFINED, { @@ -700,28 +751,31 @@ describe("TaskManagerImplementation", () => { it("should throw if trying to override a task and there is a name clash with an exising positional argument", async () => { // added argument clash with an existing positional argument await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .addPositionalArgument({ name: "arg1" }) - .setAction(() => {}) - .build(), - ], - }, - { - id: "plugin2", - tasks: [ - new TaskOverrideDefinitionBuilderImplementation("task1") - .addOption({ name: "arg1", defaultValue: "default" }) - .setAction(() => {}) - .build(), - ], - }, - ], - }), + HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .addPositionalArgument({ name: "arg1" }) + .setAction(() => {}) + .build(), + ], + }, + { + id: "plugin2", + tasks: [ + new TaskOverrideDefinitionBuilderImplementation("task1") + .addOption({ name: "arg1", defaultValue: "default" }) + .setAction(() => {}) + .build(), + ], + }, + ], + }, + {}, + ), HardhatError.ERRORS.TASK_DEFINITIONS .TASK_OVERRIDE_OPTION_ALREADY_DEFINED, { @@ -733,28 +787,31 @@ describe("TaskManagerImplementation", () => { // added flag clash with an existing positional argument await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .addPositionalArgument({ name: "flag1" }) - .setAction(() => {}) - .build(), - ], - }, - { - id: "plugin2", - tasks: [ - new TaskOverrideDefinitionBuilderImplementation("task1") - .addFlag({ name: "flag1" }) - .setAction(() => {}) - .build(), - ], - }, - ], - }), + HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .addPositionalArgument({ name: "flag1" }) + .setAction(() => {}) + .build(), + ], + }, + { + id: "plugin2", + tasks: [ + new TaskOverrideDefinitionBuilderImplementation("task1") + .addFlag({ name: "flag1" }) + .setAction(() => {}) + .build(), + ], + }, + ], + }, + {}, + ), HardhatError.ERRORS.TASK_DEFINITIONS .TASK_OVERRIDE_OPTION_ALREADY_DEFINED, { @@ -768,28 +825,31 @@ describe("TaskManagerImplementation", () => { it("should throw if trying to override a task and there is a name clash with an exising variadic argument", async () => { // added argument clash with an existing variadic argument await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .addVariadicArgument({ name: "arg1" }) - .setAction(() => {}) - .build(), - ], - }, - { - id: "plugin2", - tasks: [ - new TaskOverrideDefinitionBuilderImplementation("task1") - .addOption({ name: "arg1", defaultValue: "default" }) - .setAction(() => {}) - .build(), - ], - }, - ], - }), + HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .addVariadicArgument({ name: "arg1" }) + .setAction(() => {}) + .build(), + ], + }, + { + id: "plugin2", + tasks: [ + new TaskOverrideDefinitionBuilderImplementation("task1") + .addOption({ name: "arg1", defaultValue: "default" }) + .setAction(() => {}) + .build(), + ], + }, + ], + }, + {}, + ), HardhatError.ERRORS.TASK_DEFINITIONS .TASK_OVERRIDE_OPTION_ALREADY_DEFINED, { @@ -801,28 +861,31 @@ describe("TaskManagerImplementation", () => { // added flag clash with an existing variadic argument await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .addVariadicArgument({ name: "flag1" }) - .setAction(() => {}) - .build(), - ], - }, - { - id: "plugin2", - tasks: [ - new TaskOverrideDefinitionBuilderImplementation("task1") - .addFlag({ name: "flag1" }) - .setAction(() => {}) - .build(), - ], - }, - ], - }), + HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .addVariadicArgument({ name: "flag1" }) + .setAction(() => {}) + .build(), + ], + }, + { + id: "plugin2", + tasks: [ + new TaskOverrideDefinitionBuilderImplementation("task1") + .addFlag({ name: "flag1" }) + .setAction(() => {}) + .build(), + ], + }, + ], + }, + {}, + ), HardhatError.ERRORS.TASK_DEFINITIONS .TASK_OVERRIDE_OPTION_ALREADY_DEFINED, { @@ -837,23 +900,26 @@ describe("TaskManagerImplementation", () => { // this will fail as the config tasks are processed after // the plugin tasks so the override logic will not find task1 await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .setAction(() => {}) - .build(), - ], - plugins: [ - { - id: "plugin1", - tasks: [ - new TaskOverrideDefinitionBuilderImplementation("task1") - .setAction(() => {}) - .build(), - ], - }, - ], - }), + HardhatRuntimeEnvironmentImplementation.create( + { + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .setAction(() => {}) + .build(), + ], + plugins: [ + { + id: "plugin1", + tasks: [ + new TaskOverrideDefinitionBuilderImplementation("task1") + .setAction(() => {}) + .build(), + ], + }, + ], + }, + {}, + ), HardhatError.ERRORS.TASK_DEFINITIONS.TASK_NOT_FOUND, { task: "task1", @@ -864,63 +930,72 @@ describe("TaskManagerImplementation", () => { describe("plain object validations", () => { it("should throw if the task definition object has an empty id", async () => { await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - { - type: TaskDefinitionType.EMPTY_TASK, - id: [], - description: "", - }, - ], - }, - ], - }), + HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + { + type: TaskDefinitionType.EMPTY_TASK, + id: [], + description: "", + }, + ], + }, + ], + }, + {}, + ), HardhatError.ERRORS.TASK_DEFINITIONS.EMPTY_TASK_ID, {}, ); await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - { - type: TaskDefinitionType.NEW_TASK, - id: [], - description: "", - action: () => {}, - options: {}, - positionalArguments: [], - }, - ], - }, - ], - }), + HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + { + type: TaskDefinitionType.NEW_TASK, + id: [], + description: "", + action: () => {}, + options: {}, + positionalArguments: [], + }, + ], + }, + ], + }, + {}, + ), HardhatError.ERRORS.TASK_DEFINITIONS.EMPTY_TASK_ID, {}, ); await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - { - type: TaskDefinitionType.TASK_OVERRIDE, - id: [], - description: "", - action: () => {}, - options: {}, - }, - ], - }, - ], - }), + HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + { + type: TaskDefinitionType.TASK_OVERRIDE, + id: [], + description: "", + action: () => {}, + options: {}, + }, + ], + }, + ], + }, + {}, + ), HardhatError.ERRORS.TASK_DEFINITIONS.EMPTY_TASK_ID, {}, ); @@ -929,23 +1004,26 @@ describe("TaskManagerImplementation", () => { it("should throw if the task definition object has an invalid action file URL", async () => { const invalidActionFileUrl = "not-a-valid-file-url"; await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - { - type: TaskDefinitionType.NEW_TASK, - id: ["task-id"], - description: "", - action: invalidActionFileUrl, - options: {}, - positionalArguments: [], - }, - ], - }, - ], - }), + HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + { + type: TaskDefinitionType.NEW_TASK, + id: ["task-id"], + description: "", + action: invalidActionFileUrl, + options: {}, + positionalArguments: [], + }, + ], + }, + ], + }, + {}, + ), HardhatError.ERRORS.TASK_DEFINITIONS.INVALID_FILE_ACTION, { action: invalidActionFileUrl, @@ -953,22 +1031,25 @@ describe("TaskManagerImplementation", () => { ); await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - { - type: TaskDefinitionType.TASK_OVERRIDE, - id: ["task-id"], - description: "", - action: invalidActionFileUrl, - options: {}, - }, - ], - }, - ], - }), + HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + { + type: TaskDefinitionType.TASK_OVERRIDE, + id: ["task-id"], + description: "", + action: invalidActionFileUrl, + options: {}, + }, + ], + }, + ], + }, + {}, + ), HardhatError.ERRORS.TASK_DEFINITIONS.INVALID_FILE_ACTION, { action: invalidActionFileUrl, @@ -979,30 +1060,33 @@ describe("TaskManagerImplementation", () => { it("should throw if the task definition object has an option with an invalid name", async () => { const invalidName = "invalid-name"; await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - { - type: TaskDefinitionType.NEW_TASK, - id: ["task-id"], - description: "", - action: () => {}, - options: { - [invalidName]: { - name: invalidName, - description: "A description", - type: ArgumentType.STRING, - defaultValue: "default", + HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + { + type: TaskDefinitionType.NEW_TASK, + id: ["task-id"], + description: "", + action: () => {}, + options: { + [invalidName]: { + name: invalidName, + description: "A description", + type: ArgumentType.STRING, + defaultValue: "default", + }, }, + positionalArguments: [], }, - positionalArguments: [], - }, - ], - }, - ], - }), + ], + }, + ], + }, + {}, + ), HardhatError.ERRORS.ARGUMENTS.INVALID_NAME, { name: invalidName, @@ -1010,29 +1094,32 @@ describe("TaskManagerImplementation", () => { ); await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - { - type: TaskDefinitionType.TASK_OVERRIDE, - id: ["task-id"], - description: "", - action: () => {}, - options: { - [invalidName]: { - name: invalidName, - description: "A description", - type: ArgumentType.STRING, - defaultValue: "default", + HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + { + type: TaskDefinitionType.TASK_OVERRIDE, + id: ["task-id"], + description: "", + action: () => {}, + options: { + [invalidName]: { + name: invalidName, + description: "A description", + type: ArgumentType.STRING, + defaultValue: "default", + }, }, }, - }, - ], - }, - ], - }), + ], + }, + ], + }, + {}, + ), HardhatError.ERRORS.ARGUMENTS.INVALID_NAME, { name: invalidName, @@ -1043,7 +1130,79 @@ describe("TaskManagerImplementation", () => { it("should throw if the task definition object has an option with an reserved name", async () => { RESERVED_ARGUMENT_NAMES.forEach(async (reservedName) => { await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ + HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + { + type: TaskDefinitionType.NEW_TASK, + id: ["task-id"], + description: "", + action: () => {}, + options: { + [reservedName]: { + name: reservedName, + description: "A description", + type: ArgumentType.STRING, + defaultValue: "default", + }, + }, + positionalArguments: [], + }, + ], + }, + ], + }, + {}, + ), + HardhatError.ERRORS.ARGUMENTS.RESERVED_NAME, + { + name: reservedName, + }, + ); + + await assertRejectsWithHardhatError( + HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + { + type: TaskDefinitionType.TASK_OVERRIDE, + id: ["task-id"], + description: "", + action: () => {}, + options: { + [reservedName]: { + name: reservedName, + description: "A description", + type: ArgumentType.STRING, + defaultValue: "default", + }, + }, + }, + ], + }, + ], + }, + {}, + ), + HardhatError.ERRORS.ARGUMENTS.RESERVED_NAME, + { + name: reservedName, + }, + ); + }); + }); + + it("should throw if the task definition object has arguments with an duplicated name", async () => { + const duplicatedName = "duplicatedName"; + await assertRejectsWithHardhatError( + HardhatRuntimeEnvironmentImplementation.create( + { plugins: [ { id: "plugin1", @@ -1054,173 +1213,154 @@ describe("TaskManagerImplementation", () => { description: "", action: () => {}, options: { - [reservedName]: { - name: reservedName, + [duplicatedName]: { + name: duplicatedName, description: "A description", type: ArgumentType.STRING, defaultValue: "default", }, }, - positionalArguments: [], + positionalArguments: [ + { + name: duplicatedName, + description: "A description", + type: ArgumentType.STRING, + isVariadic: false, + }, + ], }, ], }, ], - }), - HardhatError.ERRORS.ARGUMENTS.RESERVED_NAME, - { - name: reservedName, }, - ); + {}, + ), + HardhatError.ERRORS.ARGUMENTS.DUPLICATED_NAME, + { + name: duplicatedName, + }, + ); - await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ + await assertRejectsWithHardhatError( + HardhatRuntimeEnvironmentImplementation.create( + { plugins: [ { id: "plugin1", tasks: [ { - type: TaskDefinitionType.TASK_OVERRIDE, + type: TaskDefinitionType.NEW_TASK, id: ["task-id"], description: "", action: () => {}, - options: { - [reservedName]: { - name: reservedName, + options: {}, + positionalArguments: [ + { + name: duplicatedName, description: "A description", type: ArgumentType.STRING, - defaultValue: "default", + isVariadic: false, }, - }, + { + name: duplicatedName, + description: "A description", + type: ArgumentType.STRING, + isVariadic: false, + }, + ], }, ], }, ], - }), - HardhatError.ERRORS.ARGUMENTS.RESERVED_NAME, - { - name: reservedName, }, - ); - }); - }); - - it("should throw if the task definition object has arguments with an duplicated name", async () => { - const duplicatedName = "duplicatedName"; - await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - { - type: TaskDefinitionType.NEW_TASK, - id: ["task-id"], - description: "", - action: () => {}, - options: { - [duplicatedName]: { - name: duplicatedName, - description: "A description", - type: ArgumentType.STRING, - defaultValue: "default", - }, - }, - positionalArguments: [ - { - name: duplicatedName, - description: "A description", - type: ArgumentType.STRING, - isVariadic: false, - }, - ], - }, - ], - }, - ], - }), + {}, + ), HardhatError.ERRORS.ARGUMENTS.DUPLICATED_NAME, { name: duplicatedName, }, ); + }); + it("should throw if the task definition object has a positional argument with an invalid name", async () => { + const invalidName = "invalid-name"; await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - { - type: TaskDefinitionType.NEW_TASK, - id: ["task-id"], - description: "", - action: () => {}, - options: {}, - positionalArguments: [ - { - name: duplicatedName, - description: "A description", - type: ArgumentType.STRING, - isVariadic: false, - }, - { - name: duplicatedName, - description: "A description", - type: ArgumentType.STRING, - isVariadic: false, - }, - ], - }, - ], - }, - ], - }), - HardhatError.ERRORS.ARGUMENTS.DUPLICATED_NAME, + HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + { + type: TaskDefinitionType.NEW_TASK, + id: ["task-id"], + description: "", + action: () => {}, + options: {}, + positionalArguments: [ + { + name: invalidName, + description: "A description", + type: ArgumentType.STRING, + isVariadic: false, + }, + ], + }, + ], + }, + ], + }, + {}, + ), + HardhatError.ERRORS.ARGUMENTS.INVALID_NAME, { - name: duplicatedName, + name: invalidName, }, ); }); - it("should throw if the task definition object has a positional argument with an invalid name", async () => { - const invalidName = "invalid-name"; - await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ - plugins: [ + it("should throw if the task definition object has a positional argument with an reserved name", async () => { + RESERVED_ARGUMENT_NAMES.forEach(async (reservedName) => { + await assertRejectsWithHardhatError( + HardhatRuntimeEnvironmentImplementation.create( { - id: "plugin1", - tasks: [ + plugins: [ { - type: TaskDefinitionType.NEW_TASK, - id: ["task-id"], - description: "", - action: () => {}, - options: {}, - positionalArguments: [ + id: "plugin1", + tasks: [ { - name: invalidName, - description: "A description", - type: ArgumentType.STRING, - isVariadic: false, + type: TaskDefinitionType.NEW_TASK, + id: ["task-id"], + description: "", + action: () => {}, + options: {}, + positionalArguments: [ + { + name: reservedName, + description: "A description", + type: ArgumentType.STRING, + isVariadic: false, + }, + ], }, ], }, ], }, - ], - }), - HardhatError.ERRORS.ARGUMENTS.INVALID_NAME, - { - name: invalidName, - }, - ); + {}, + ), + HardhatError.ERRORS.ARGUMENTS.RESERVED_NAME, + { + name: reservedName, + }, + ); + }); }); - it("should throw if the task definition object has a positional argument with an reserved name", async () => { - RESERVED_ARGUMENT_NAMES.forEach(async (reservedName) => { - await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ + it("should throw if the task definition object has a required positional argument after an optional argument", async () => { + await assertRejectsWithHardhatError( + HardhatRuntimeEnvironmentImplementation.create( + { plugins: [ { id: "plugin1", @@ -1233,7 +1373,14 @@ describe("TaskManagerImplementation", () => { options: {}, positionalArguments: [ { - name: reservedName, + name: "posArg", + description: "A description", + type: ArgumentType.STRING, + isVariadic: false, + defaultValue: "default", + }, + { + name: "posArg2", description: "A description", type: ArgumentType.STRING, isVariadic: false, @@ -1243,48 +1390,9 @@ describe("TaskManagerImplementation", () => { ], }, ], - }), - HardhatError.ERRORS.ARGUMENTS.RESERVED_NAME, - { - name: reservedName, }, - ); - }); - }); - - it("should throw if the task definition object has a required positional argument after an optional argument", async () => { - await assertRejectsWithHardhatError( - createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - { - type: TaskDefinitionType.NEW_TASK, - id: ["task-id"], - description: "", - action: () => {}, - options: {}, - positionalArguments: [ - { - name: "posArg", - description: "A description", - type: ArgumentType.STRING, - isVariadic: false, - defaultValue: "default", - }, - { - name: "posArg2", - description: "A description", - type: ArgumentType.STRING, - isVariadic: false, - }, - ], - }, - ], - }, - ], - }), + {}, + ), HardhatError.ERRORS.TASK_DEFINITIONS.REQUIRED_ARG_AFTER_OPTIONAL, { name: "posArg2", @@ -1296,23 +1404,26 @@ describe("TaskManagerImplementation", () => { describe("getTask", () => { it("should return the task if it exists", async () => { - const hre = await createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .setAction(() => {}) - .build(), - ], - }, - ], - tasks: [ - new NewTaskDefinitionBuilderImplementation("task2") - .setAction(() => {}) - .build(), - ], - }); + const hre = await HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .setAction(() => {}) + .build(), + ], + }, + ], + tasks: [ + new NewTaskDefinitionBuilderImplementation("task2") + .setAction(() => {}) + .build(), + ], + }, + {}, + ); const task1 = hre.tasks.getTask("task1"); assert.deepEqual(task1.id, ["task1"]); @@ -1324,7 +1435,7 @@ describe("TaskManagerImplementation", () => { }); it("should throw if the task doesn't exist", async () => { - const hre = await createBaseHardhatRuntimeEnvironment({}); + const hre = await HardhatRuntimeEnvironmentImplementation.create({}, {}); // task1 will not be found as it's not defined assertThrowsHardhatError( () => hre.tasks.getTask("task1"), @@ -1343,40 +1454,46 @@ describe("TaskManagerImplementation", () => { describe("run", () => { it("should run a task without arguments", async () => { let taskRun = false; - const hre = await createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .setAction(() => { - taskRun = true; - }) - .build(), - ], - }, - ], - }); + const hre = await HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .setAction(() => { + taskRun = true; + }) + .build(), + ], + }, + ], + }, + {}, + ); const task1 = hre.tasks.getTask("task1"); assert.equal(taskRun, false); - await task1.run({}); - assert.equal(taskRun, true); - }); - - it("should return the result of the task action", async () => { - const hre = await createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .setAction(() => "task run successfully") - .build(), - ], - }, - ], - }); + await task1.run({}); + assert.equal(taskRun, true); + }); + + it("should return the result of the task action", async () => { + const hre = await HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .setAction(() => "task run successfully") + .build(), + ], + }, + ], + }, + {}, + ); const task1 = hre.tasks.getTask("task1"); const result = await task1.run({}); @@ -1386,26 +1503,29 @@ describe("TaskManagerImplementation", () => { it("should run a overridden task without arguments", async () => { let taskRun = false; let overrideTaskRun = false; - const hre = await createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .setAction(() => { - taskRun = true; - }) - .build(), - new TaskOverrideDefinitionBuilderImplementation("task1") - .setAction(async (args, _hre, runSuper) => { - await runSuper(args); - overrideTaskRun = true; - }) - .build(), - ], - }, - ], - }); + const hre = await HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .setAction(() => { + taskRun = true; + }) + .build(), + new TaskOverrideDefinitionBuilderImplementation("task1") + .setAction(async (args, _hre, runSuper) => { + await runSuper(args); + overrideTaskRun = true; + }) + .build(), + ], + }, + ], + }, + {}, + ); const task1 = hre.tasks.getTask("task1"); assert.equal(taskRun, false); @@ -1420,45 +1540,48 @@ describe("TaskManagerImplementation", () => { let override1TaskRun = false; let override2TaskRun = false; let override3TaskRun = false; - const hre = await createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .setAction(() => { - taskRun = true; - }) - .build(), - new TaskOverrideDefinitionBuilderImplementation("task1") - .setAction(async (args, _hre, runSuper) => { - await runSuper(args); - override1TaskRun = true; - }) - .build(), - ], - }, - { - id: "plugin2", - tasks: [ - new TaskOverrideDefinitionBuilderImplementation("task1") - .setAction(async (args, _hre, runSuper) => { - await runSuper(args); - override2TaskRun = true; - }) - .build(), - ], - }, - ], - tasks: [ - new TaskOverrideDefinitionBuilderImplementation("task1") - .setAction(async (args, _hre, runSuper) => { - await runSuper(args); - override3TaskRun = true; - }) - .build(), - ], - }); + const hre = await HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .setAction(() => { + taskRun = true; + }) + .build(), + new TaskOverrideDefinitionBuilderImplementation("task1") + .setAction(async (args, _hre, runSuper) => { + await runSuper(args); + override1TaskRun = true; + }) + .build(), + ], + }, + { + id: "plugin2", + tasks: [ + new TaskOverrideDefinitionBuilderImplementation("task1") + .setAction(async (args, _hre, runSuper) => { + await runSuper(args); + override2TaskRun = true; + }) + .build(), + ], + }, + ], + tasks: [ + new TaskOverrideDefinitionBuilderImplementation("task1") + .setAction(async (args, _hre, runSuper) => { + await runSuper(args); + override3TaskRun = true; + }) + .build(), + ], + }, + {}, + ); const task1 = hre.tasks.getTask("task1"); assert.equal(taskRun, false); @@ -1475,25 +1598,28 @@ describe("TaskManagerImplementation", () => { it("should not run the original task action if the override task action doesn't call runSuper", async () => { let taskRun = false; let overrideTaskRun = false; - const hre = await createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .setAction(() => { - taskRun = true; - }) - .build(), - new TaskOverrideDefinitionBuilderImplementation("task1") - .setAction(async (_args, _hre, _runSuper) => { - overrideTaskRun = true; - }) - .build(), - ], - }, - ], - }); + const hre = await HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .setAction(() => { + taskRun = true; + }) + .build(), + new TaskOverrideDefinitionBuilderImplementation("task1") + .setAction(async (_args, _hre, _runSuper) => { + overrideTaskRun = true; + }) + .build(), + ], + }, + ], + }, + {}, + ); const task1 = hre.tasks.getTask("task1"); assert.equal(taskRun, false); @@ -1504,45 +1630,50 @@ describe("TaskManagerImplementation", () => { }); it("should run a task with arguments", async () => { - const hre = await createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .addOption({ name: "arg1", defaultValue: "default" }) - .addOption({ name: "withDefault", defaultValue: "default" }) - .addFlag({ name: "flag1" }) - .addPositionalArgument({ name: "posArg" }) - .addVariadicArgument({ name: "varArg" }) - .setAction((args) => { - assert.deepEqual(args, { - arg1: "arg1Value", - flag1: true, - posArg: "posValue", - varArg: ["varValue1", "varValue2"], - withDefault: "default", - }); - }) - .build(), - new TaskOverrideDefinitionBuilderImplementation("task1") - .addOption({ name: "arg2", defaultValue: "default" }) - .addFlag({ name: "flag2" }) - .setAction(async ({ arg2, flag2, ...args }, _hre, runSuper) => { - await runSuper(args); - assert.deepEqual( - { arg2, flag2 }, - { - arg2: "arg2Value", - flag2: true, + const hre = await HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .addOption({ name: "arg1", defaultValue: "default" }) + .addOption({ name: "withDefault", defaultValue: "default" }) + .addFlag({ name: "flag1" }) + .addPositionalArgument({ name: "posArg" }) + .addVariadicArgument({ name: "varArg" }) + .setAction((args) => { + assert.deepEqual(args, { + arg1: "arg1Value", + flag1: true, + posArg: "posValue", + varArg: ["varValue1", "varValue2"], + withDefault: "default", + }); + }) + .build(), + new TaskOverrideDefinitionBuilderImplementation("task1") + .addOption({ name: "arg2", defaultValue: "default" }) + .addFlag({ name: "flag2" }) + .setAction( + async ({ arg2, flag2, ...args }, _hre, runSuper) => { + await runSuper(args); + assert.deepEqual( + { arg2, flag2 }, + { + arg2: "arg2Value", + flag2: true, + }, + ); }, - ); - }) - .build(), - ], - }, - ], - }); + ) + .build(), + ], + }, + ], + }, + {}, + ); // withDefault option is intentionally omitted const providedArgs = { arg1: "arg1Value", @@ -1571,38 +1702,41 @@ describe("TaskManagerImplementation", () => { }); it("should run a task with arguments and resolve their default values", async () => { - const hre = await createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .addOption({ - name: "arg1", - defaultValue: "arg1DefaultValue", - }) - .addFlag({ name: "flag1" }) - .addPositionalArgument({ - name: "posArg", - defaultValue: "posValue", - }) - .addVariadicArgument({ - name: "varArg", - defaultValue: ["varValue1", "varValue2"], - }) - .setAction((args) => { - assert.deepEqual(args, { - arg1: "arg1DefaultValue", - flag1: false, - posArg: "posValue", - varArg: ["varValue1", "varValue2"], - }); - }) - .build(), - ], - }, - ], - }); + const hre = await HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .addOption({ + name: "arg1", + defaultValue: "arg1DefaultValue", + }) + .addFlag({ name: "flag1" }) + .addPositionalArgument({ + name: "posArg", + defaultValue: "posValue", + }) + .addVariadicArgument({ + name: "varArg", + defaultValue: ["varValue1", "varValue2"], + }) + .setAction((args) => { + assert.deepEqual(args, { + arg1: "arg1DefaultValue", + flag1: false, + posArg: "posValue", + varArg: ["varValue1", "varValue2"], + }); + }) + .build(), + ], + }, + ], + }, + {}, + ); const task1 = hre.tasks.getTask("task1"); await task1.run({}); @@ -1610,25 +1744,28 @@ describe("TaskManagerImplementation", () => { it("should run an empty task that was overriden", async () => { let overrideTaskRun = false; - const hre = await createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new EmptyTaskDefinitionBuilderImplementation( - "task1", - "description1", - ).build(), - new TaskOverrideDefinitionBuilderImplementation("task1") - .setAction(async (args, _hre, runSuper) => { - await runSuper(args); - overrideTaskRun = true; - }) - .build(), - ], - }, - ], - }); + const hre = await HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new EmptyTaskDefinitionBuilderImplementation( + "task1", + "description1", + ).build(), + new TaskOverrideDefinitionBuilderImplementation("task1") + .setAction(async (args, _hre, runSuper) => { + await runSuper(args); + overrideTaskRun = true; + }) + .build(), + ], + }, + ], + }, + {}, + ); const task1 = hre.tasks.getTask("task1"); assert.equal(overrideTaskRun, false); @@ -1641,22 +1778,25 @@ describe("TaskManagerImplementation", () => { "./fixture-projects/file-actions/action-fn.js", ); - const hre = await createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .setAction((args) => args) - .build(), - new TaskOverrideDefinitionBuilderImplementation("task1") - .addOption({ name: "arg1", defaultValue: "default" }) - .setAction(actionUrl) - .build(), - ], - }, - ], - }); + const hre = await HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .setAction((args) => args) + .build(), + new TaskOverrideDefinitionBuilderImplementation("task1") + .addOption({ name: "arg1", defaultValue: "default" }) + .setAction(actionUrl) + .build(), + ], + }, + ], + }, + {}, + ); const task1 = hre.tasks.getTask("task1"); const response = await task1.run({ arg1: "arg1Value" }); @@ -1668,22 +1808,25 @@ describe("TaskManagerImplementation", () => { "./fixture-projects/file-actions/no-run-super.js", ); - const hre = await createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .setAction("file://not-a-module") - .build(), - new TaskOverrideDefinitionBuilderImplementation("task1") - .addOption({ name: "arg1", defaultValue: "default" }) - .setAction(validActionUrl) - .build(), - ], - }, - ], - }); + const hre = await HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .setAction("file://not-a-module") + .build(), + new TaskOverrideDefinitionBuilderImplementation("task1") + .addOption({ name: "arg1", defaultValue: "default" }) + .setAction(validActionUrl) + .build(), + ], + }, + ], + }, + {}, + ); const task1 = hre.tasks.getTask("task1"); const response = await task1.run({ arg1: "arg1Value" }); @@ -1695,19 +1838,22 @@ describe("TaskManagerImplementation", () => { describe("validations", () => { it("should throw if the task is empty", async () => { - const hre = await createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new EmptyTaskDefinitionBuilderImplementation( - "task1", - "description1", - ).build(), - ], - }, - ], - }); + const hre = await HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new EmptyTaskDefinitionBuilderImplementation( + "task1", + "description1", + ).build(), + ], + }, + ], + }, + {}, + ); const task1 = hre.tasks.getTask("task1"); await assertRejectsWithHardhatError( @@ -1720,18 +1866,21 @@ describe("TaskManagerImplementation", () => { }); it("should throw if the provided argument is not one of the task arguments", async () => { - const hre = await createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .setAction(() => {}) - .build(), - ], - }, - ], - }); + const hre = await HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .setAction(() => {}) + .build(), + ], + }, + ], + }, + {}, + ); const task1 = hre.tasks.getTask("task1"); await assertRejectsWithHardhatError( @@ -1745,20 +1894,23 @@ describe("TaskManagerImplementation", () => { }); it("should throw if a required argument is missing", async () => { - const hre = await createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .addPositionalArgument({ name: "posArg" }) - .addVariadicArgument({ name: "varArg" }) - .setAction(() => {}) - .build(), - ], - }, - ], - }); + const hre = await HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .addPositionalArgument({ name: "posArg" }) + .addVariadicArgument({ name: "varArg" }) + .setAction(() => {}) + .build(), + ], + }, + ], + }, + {}, + ); const task1 = hre.tasks.getTask("task1"); @@ -1790,31 +1942,34 @@ describe("TaskManagerImplementation", () => { }); it("should throw if the provided value for the argument is not of the correct type", async () => { - const hre = await createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .addOption({ - name: "option", - type: ArgumentType.BIGINT, - defaultValue: 1n, - }) - .addPositionalArgument({ - name: "posArg", - type: ArgumentType.INT, - }) - .addVariadicArgument({ - name: "varArg", - type: ArgumentType.FILE, - }) - .setAction(() => {}) - .build(), - ], - }, - ], - }); + const hre = await HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .addOption({ + name: "option", + type: ArgumentType.BIGINT, + defaultValue: 1n, + }) + .addPositionalArgument({ + name: "posArg", + type: ArgumentType.INT, + }) + .addVariadicArgument({ + name: "varArg", + type: ArgumentType.FILE, + }) + .setAction(() => {}) + .build(), + ], + }, + ], + }, + {}, + ); const task1 = hre.tasks.getTask("task1"); @@ -1884,18 +2039,21 @@ describe("TaskManagerImplementation", () => { }); it("should throw if an action url is provided but the corresponding module can't be resolved", async () => { - const hre = await createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .setAction("file://not-a-module") - .build(), - ], - }, - ], - }); + const hre = await HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .setAction("file://not-a-module") + .build(), + ], + }, + ], + }, + {}, + ); const task1 = hre.tasks.getTask("task1"); await assertRejectsWithHardhatError( @@ -1919,19 +2077,22 @@ describe("TaskManagerImplementation", () => { ); // the missing dependency is used in the NEW_TASK action - let hre = await createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - npmPackage: "non-installed-package", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .setAction(nonInstalledPackageActionUrl) - .build(), - ], - }, - ], - }); + let hre = await HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + npmPackage: "non-installed-package", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .setAction(nonInstalledPackageActionUrl) + .build(), + ], + }, + ], + }, + {}, + ); await assertRejectsWithHardhatError( hre.tasks.getTask("task1").run({}), @@ -1942,27 +2103,30 @@ describe("TaskManagerImplementation", () => { ); // the missing dependency is used in the TASK_OVERRIDE action - hre = await createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .setAction(() => {}) - .build(), - ], - }, - { - id: "plugin2", - npmPackage: "non-installed-package", - tasks: [ - new TaskOverrideDefinitionBuilderImplementation("task1") - .setAction(nonInstalledPackageActionUrl) - .build(), - ], - }, - ], - }); + hre = await HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .setAction(() => {}) + .build(), + ], + }, + { + id: "plugin2", + npmPackage: "non-installed-package", + tasks: [ + new TaskOverrideDefinitionBuilderImplementation("task1") + .setAction(nonInstalledPackageActionUrl) + .build(), + ], + }, + ], + }, + {}, + ); await assertRejectsWithHardhatError( hre.tasks.getTask("task1").run({}), @@ -1978,18 +2142,21 @@ describe("TaskManagerImplementation", () => { "./fixture-projects/file-actions/no-default.js", ); - const hre = await createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .setAction(actionUrl) - .build(), - ], - }, - ], - }); + const hre = await HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .setAction(actionUrl) + .build(), + ], + }, + ], + }, + {}, + ); const task1 = hre.tasks.getTask("task1"); await assertRejectsWithHardhatError( @@ -2007,18 +2174,21 @@ describe("TaskManagerImplementation", () => { "./fixture-projects/file-actions/no-default-fn.js", ); - const hre = await createBaseHardhatRuntimeEnvironment({ - plugins: [ - { - id: "plugin1", - tasks: [ - new NewTaskDefinitionBuilderImplementation("task1") - .setAction(actionUrl) - .build(), - ], - }, - ], - }); + const hre = await HardhatRuntimeEnvironmentImplementation.create( + { + plugins: [ + { + id: "plugin1", + tasks: [ + new NewTaskDefinitionBuilderImplementation("task1") + .setAction(actionUrl) + .build(), + ], + }, + ], + }, + {}, + ); const task1 = hre.tasks.getTask("task1"); await assertRejectsWithHardhatError( diff --git a/v-next/hardhat/test/internal/core/user-interruptions/user-interruptions-manager.ts b/v-next/hardhat/test/internal/core/user-interruptions/user-interruptions-manager.ts index 0c8382ed4c..e82bfb8722 100644 --- a/v-next/hardhat/test/internal/core/user-interruptions/user-interruptions-manager.ts +++ b/v-next/hardhat/test/internal/core/user-interruptions/user-interruptions-manager.ts @@ -4,7 +4,7 @@ import assert from "node:assert/strict"; import { before, describe, it } from "node:test"; import { HookManagerImplementation } from "../../../../src/internal/core/hook-manager.js"; -import { resolveProjectRoot } from "../../../../src/internal/core/index.js"; +import { resolveProjectRoot } from "../../../../src/internal/core/hre.js"; import { UserInterruptionManagerImplementation } from "../../../../src/internal/core/user-interruptions.js"; describe("UserInterruptionManager", () => { From 1cc9f27d65d0fa3864257f48acb14164e7fa8709 Mon Sep 17 00:00:00 2001 From: Patricio Palladino Date: Sat, 31 Aug 2024 15:59:30 +0000 Subject: [PATCH 05/14] Move config-loading out of helpers. This change is due to that module being more important now, and not just a helper. --- v-next/hardhat/src/hre.ts | 2 +- v-next/hardhat/src/index.ts | 8 ++++---- v-next/hardhat/src/internal/cli/init/init.ts | 2 +- v-next/hardhat/src/internal/cli/main.ts | 8 ++++---- .../hardhat/src/internal/{helpers => }/config-loading.ts | 2 +- v-next/hardhat/test/hre/index.ts | 2 +- v-next/hardhat/test/internal/cli/init/init.ts | 2 +- 7 files changed, 13 insertions(+), 13 deletions(-) rename v-next/hardhat/src/internal/{helpers => }/config-loading.ts (98%) diff --git a/v-next/hardhat/src/hre.ts b/v-next/hardhat/src/hre.ts index 8bc296ef95..355c91f744 100644 --- a/v-next/hardhat/src/hre.ts +++ b/v-next/hardhat/src/hre.ts @@ -12,7 +12,7 @@ import { } from "./internal/core/hre.js"; import { resolvePluginList } from "./internal/core/plugins/resolve-plugin-list.js"; -export { resolveHardhatConfigPath } from "./internal/helpers/config-loading.js"; +export { resolveHardhatConfigPath } from "./internal/config-loading.js"; /** * Creates an instances of the Hardhat Runtime Environment. diff --git a/v-next/hardhat/src/index.ts b/v-next/hardhat/src/index.ts index 4e19931670..894569a9d1 100644 --- a/v-next/hardhat/src/index.ts +++ b/v-next/hardhat/src/index.ts @@ -6,15 +6,15 @@ import type { TaskManager } from "./types/tasks.js"; import type { UserInterruptionManager } from "./types/user-interruptions.js"; import { createHardhatRuntimeEnvironment } from "./hre.js"; +import { + importUserConfig, + resolveHardhatConfigPath, +} from "./internal/config-loading.js"; import { resolveProjectRoot } from "./internal/core/hre.js"; import { getGlobalHardhatRuntimeEnvironment, setGlobalHardhatRuntimeEnvironment, } from "./internal/global-hre-instance.js"; -import { - importUserConfig, - resolveHardhatConfigPath, -} from "./internal/helpers/config-loading.js"; let maybeHre: HardhatRuntimeEnvironment | undefined = getGlobalHardhatRuntimeEnvironment(); diff --git a/v-next/hardhat/src/internal/cli/init/init.ts b/v-next/hardhat/src/internal/cli/init/init.ts index 601145ad45..1951f7ea98 100644 --- a/v-next/hardhat/src/internal/cli/init/init.ts +++ b/v-next/hardhat/src/internal/cli/init/init.ts @@ -1,6 +1,6 @@ import { HardhatError } from "@ignored/hardhat-vnext-errors"; -import { findClosestHardhatConfig } from "../../helpers/config-loading.js"; +import { findClosestHardhatConfig } from "../../config-loading.js"; import { createProject } from "./project-creation.js"; diff --git a/v-next/hardhat/src/internal/cli/main.ts b/v-next/hardhat/src/internal/cli/main.ts index 79acf5a789..b13293cf21 100644 --- a/v-next/hardhat/src/internal/cli/main.ts +++ b/v-next/hardhat/src/internal/cli/main.ts @@ -21,15 +21,15 @@ import { } from "../../types/arguments.js"; import { BUILTIN_GLOBAL_OPTIONS_DEFINITIONS } from "../builtin-global-options.js"; import { builtinPlugins } from "../builtin-plugins/index.js"; +import { + importUserConfig, + resolveHardhatConfigPath, +} from "../config-loading.js"; import { parseArgumentValue } from "../core/arguments.js"; import { buildGlobalOptionDefinitions } from "../core/global-options.js"; import { resolveProjectRoot } from "../core/hre.js"; import { resolvePluginList } from "../core/plugins/resolve-plugin-list.js"; import { setGlobalHardhatRuntimeEnvironment } from "../global-hre-instance.js"; -import { - importUserConfig, - resolveHardhatConfigPath, -} from "../helpers/config-loading.js"; import { printErrorMessages } from "./error-handler.js"; import { getGlobalHelpString } from "./helpers/getGlobalHelpString.js"; diff --git a/v-next/hardhat/src/internal/helpers/config-loading.ts b/v-next/hardhat/src/internal/config-loading.ts similarity index 98% rename from v-next/hardhat/src/internal/helpers/config-loading.ts rename to v-next/hardhat/src/internal/config-loading.ts index 8e9374c35e..fc008e2be2 100644 --- a/v-next/hardhat/src/internal/helpers/config-loading.ts +++ b/v-next/hardhat/src/internal/config-loading.ts @@ -1,4 +1,4 @@ -import type { HardhatUserConfig } from "../../types/config.js"; +import type { HardhatUserConfig } from "../types/config.js"; import { isAbsolute, resolve } from "node:path"; import { pathToFileURL } from "node:url"; diff --git a/v-next/hardhat/test/hre/index.ts b/v-next/hardhat/test/hre/index.ts index 8376ec55f3..81e79d12b8 100644 --- a/v-next/hardhat/test/hre/index.ts +++ b/v-next/hardhat/test/hre/index.ts @@ -10,12 +10,12 @@ import { import { createHardhatRuntimeEnvironment } from "../../src/hre.js"; import { builtinPlugins } from "../../src/internal/builtin-plugins/index.js"; +import { resolveHardhatConfigPath } from "../../src/internal/config-loading.js"; import { getGlobalHardhatRuntimeEnvironment, resetGlobalHardhatRuntimeEnvironment, setGlobalHardhatRuntimeEnvironment, } from "../../src/internal/global-hre-instance.js"; -import { resolveHardhatConfigPath } from "../../src/internal/helpers/config-loading.js"; describe("HRE", () => { afterEach(() => { diff --git a/v-next/hardhat/test/internal/cli/init/init.ts b/v-next/hardhat/test/internal/cli/init/init.ts index 4d860b4622..ad93fb445f 100644 --- a/v-next/hardhat/test/internal/cli/init/init.ts +++ b/v-next/hardhat/test/internal/cli/init/init.ts @@ -15,7 +15,7 @@ import { import { initHardhat } from "../../../../src/internal/cli/init/init.js"; import { EMPTY_HARDHAT_CONFIG } from "../../../../src/internal/cli/init/sample-config-file.js"; -import { findClosestHardhatConfig } from "../../../../src/internal/helpers/config-loading.js"; +import { findClosestHardhatConfig } from "../../../../src/internal/config-loading.js"; async function deleteHardhatConfigFile() { await remove(path.join(process.cwd(), "hardhat.config.ts")); From a8f9a9efc6d285f15bf483d78b7410c1299d103a Mon Sep 17 00:00:00 2001 From: Patricio Palladino Date: Sat, 31 Aug 2024 16:02:49 +0000 Subject: [PATCH 06/14] Move global-dir out of core --- .../hardhat/src/internal/builtin-plugins/clean/task-action.ts | 2 +- .../hardhat/src/internal/builtin-plugins/console/task-action.ts | 2 +- v-next/hardhat/src/internal/cli/telemetry/analytics/utils.ts | 2 +- .../hardhat/src/internal/cli/telemetry/telemetry-permissions.ts | 2 +- v-next/hardhat/src/internal/{core => }/global-dir.ts | 0 .../hardhat/test/internal/builtin-plugins/clean/task-action.ts | 2 +- .../test/internal/cli/telemetry/telemetry-permissions.ts | 2 +- v-next/hardhat/test/internal/{core => }/global-dir.ts | 2 +- 8 files changed, 7 insertions(+), 7 deletions(-) rename v-next/hardhat/src/internal/{core => }/global-dir.ts (100%) rename v-next/hardhat/test/internal/{core => }/global-dir.ts (86%) diff --git a/v-next/hardhat/src/internal/builtin-plugins/clean/task-action.ts b/v-next/hardhat/src/internal/builtin-plugins/clean/task-action.ts index f53e6dbfaf..d421c5a9bb 100644 --- a/v-next/hardhat/src/internal/builtin-plugins/clean/task-action.ts +++ b/v-next/hardhat/src/internal/builtin-plugins/clean/task-action.ts @@ -2,7 +2,7 @@ import type { NewTaskActionFunction } from "../../../types/tasks.js"; import { emptyDir, remove } from "@ignored/hardhat-vnext-utils/fs"; -import { getCacheDir } from "../../core/global-dir.js"; +import { getCacheDir } from "../../global-dir.js"; interface CleanActionArguments { global: boolean; diff --git a/v-next/hardhat/src/internal/builtin-plugins/console/task-action.ts b/v-next/hardhat/src/internal/builtin-plugins/console/task-action.ts index f610814150..86cdb1e0e3 100644 --- a/v-next/hardhat/src/internal/builtin-plugins/console/task-action.ts +++ b/v-next/hardhat/src/internal/builtin-plugins/console/task-action.ts @@ -6,7 +6,7 @@ import repl from "node:repl"; import debug from "debug"; -import { getCacheDir } from "../../core/global-dir.js"; +import { getCacheDir } from "../../global-dir.js"; const log = debug("hardhat:core:tasks:console"); diff --git a/v-next/hardhat/src/internal/cli/telemetry/analytics/utils.ts b/v-next/hardhat/src/internal/cli/telemetry/analytics/utils.ts index ca26e5c4bf..3409d52aac 100644 --- a/v-next/hardhat/src/internal/cli/telemetry/analytics/utils.ts +++ b/v-next/hardhat/src/internal/cli/telemetry/analytics/utils.ts @@ -9,7 +9,7 @@ import { } from "@ignored/hardhat-vnext-utils/fs"; import debug from "debug"; -import { getTelemetryDir } from "../../../core/global-dir.js"; +import { getTelemetryDir } from "../../../global-dir.js"; const log = debug("hardhat:cli:telemetry:analytics:utils"); diff --git a/v-next/hardhat/src/internal/cli/telemetry/telemetry-permissions.ts b/v-next/hardhat/src/internal/cli/telemetry/telemetry-permissions.ts index 62475f8a9d..4e7c570d44 100644 --- a/v-next/hardhat/src/internal/cli/telemetry/telemetry-permissions.ts +++ b/v-next/hardhat/src/internal/cli/telemetry/telemetry-permissions.ts @@ -8,7 +8,7 @@ import { } from "@ignored/hardhat-vnext-utils/fs"; import debug from "debug"; -import { getConfigDir } from "../../core/global-dir.js"; +import { getConfigDir } from "../../global-dir.js"; import { confirmationPromptWithTimeout } from "../prompt/prompt.js"; import { sendTelemetryConsentAnalytics } from "./analytics/analytics.js"; diff --git a/v-next/hardhat/src/internal/core/global-dir.ts b/v-next/hardhat/src/internal/global-dir.ts similarity index 100% rename from v-next/hardhat/src/internal/core/global-dir.ts rename to v-next/hardhat/src/internal/global-dir.ts diff --git a/v-next/hardhat/test/internal/builtin-plugins/clean/task-action.ts b/v-next/hardhat/test/internal/builtin-plugins/clean/task-action.ts index b28b5c6c51..d57a1ddef6 100644 --- a/v-next/hardhat/test/internal/builtin-plugins/clean/task-action.ts +++ b/v-next/hardhat/test/internal/builtin-plugins/clean/task-action.ts @@ -15,7 +15,7 @@ import { useFixtureProject } from "@nomicfoundation/hardhat-test-utils"; import { createHardhatRuntimeEnvironment } from "../../../../src/hre.js"; import cleanAction from "../../../../src/internal/builtin-plugins/clean/task-action.js"; -import { getCacheDir } from "../../../../src/internal/core/global-dir.js"; +import { getCacheDir } from "../../../../src/internal/global-dir.js"; let hre: HardhatRuntimeEnvironment; let globalCacheDir: string; diff --git a/v-next/hardhat/test/internal/cli/telemetry/telemetry-permissions.ts b/v-next/hardhat/test/internal/cli/telemetry/telemetry-permissions.ts index 6409c9be58..edad31b516 100644 --- a/v-next/hardhat/test/internal/cli/telemetry/telemetry-permissions.ts +++ b/v-next/hardhat/test/internal/cli/telemetry/telemetry-permissions.ts @@ -5,7 +5,7 @@ import { afterEach, beforeEach, describe, it } from "node:test"; import { remove, writeJsonFile } from "@ignored/hardhat-vnext-utils/fs"; import { isTelemetryAllowed } from "../../../../src/internal/cli/telemetry/telemetry-permissions.js"; -import { getConfigDir } from "../../../../src/internal/core/global-dir.js"; +import { getConfigDir } from "../../../../src/internal/global-dir.js"; async function setTelemetryConsentFile(consent: boolean) { const configDir = await getConfigDir(); diff --git a/v-next/hardhat/test/internal/core/global-dir.ts b/v-next/hardhat/test/internal/global-dir.ts similarity index 86% rename from v-next/hardhat/test/internal/core/global-dir.ts rename to v-next/hardhat/test/internal/global-dir.ts index a909267a33..0153190378 100644 --- a/v-next/hardhat/test/internal/core/global-dir.ts +++ b/v-next/hardhat/test/internal/global-dir.ts @@ -1,7 +1,7 @@ import assert from "node:assert/strict"; import { describe, it } from "node:test"; -import { getConfigDir } from "../../../src/internal/core/global-dir.js"; +import { getConfigDir } from "../../src/internal/global-dir.js"; describe("global-dir", () => { describe("getGlobalDir", () => { From 1f63bba0e8188939d888dbe3237f88e0d52c03b3 Mon Sep 17 00:00:00 2001 From: Patricio Palladino Date: Sat, 31 Aug 2024 16:33:23 +0000 Subject: [PATCH 07/14] Move the initialization of the HRE to its own module --- v-next/hardhat/src/cli.ts | 4 - v-next/hardhat/src/hre.ts | 66 +---------- v-next/hardhat/src/index.ts | 34 +----- v-next/hardhat/src/internal/cli/main.ts | 2 +- .../hardhat/src/internal/hre-intialization.ts | 103 ++++++++++++++++++ .../builtin-plugins/clean/task-action.ts | 2 +- .../builtin-plugins/console/task-action.ts | 2 +- .../builtin-plugins/run/task-action.ts | 2 +- v-next/hardhat/test/internal/cli/main.ts | 2 +- .../hre-intialization.ts} | 19 +++- 10 files changed, 127 insertions(+), 109 deletions(-) create mode 100644 v-next/hardhat/src/internal/hre-intialization.ts rename v-next/hardhat/test/{hre/index.ts => internal/hre-intialization.ts} (92%) diff --git a/v-next/hardhat/src/cli.ts b/v-next/hardhat/src/cli.ts index 8f76b6cf67..6a768cd033 100644 --- a/v-next/hardhat/src/cli.ts +++ b/v-next/hardhat/src/cli.ts @@ -1,9 +1,5 @@ import { register } from "tsx/esm/api"; -// Note: We import the builtin plugins' types here, so that any type extension -// they may have gets loaded. -import "./internal/builtin-plugins/index.js"; - // We enable the sourcemaps before loading main, so that everything except this // small file is loaded with sourcemaps enabled. process.setSourceMapsEnabled(true); diff --git a/v-next/hardhat/src/hre.ts b/v-next/hardhat/src/hre.ts index 355c91f744..d59fbdaa29 100644 --- a/v-next/hardhat/src/hre.ts +++ b/v-next/hardhat/src/hre.ts @@ -1,66 +1,2 @@ -import type { UnsafeHardhatRuntimeEnvironmentOptions } from "./internal/core/types.js"; -import type { HardhatUserConfig } from "./types/config.js"; -import type { GlobalOptions } from "./types/global-options.js"; -import type { HardhatRuntimeEnvironment } from "./types/hre.js"; - -import { BUILTIN_GLOBAL_OPTIONS_DEFINITIONS } from "./internal/builtin-global-options.js"; -import { builtinPlugins } from "./internal/builtin-plugins/index.js"; -import { buildGlobalOptionDefinitions } from "./internal/core/global-options.js"; -import { - HardhatRuntimeEnvironmentImplementation, - resolveProjectRoot, -} from "./internal/core/hre.js"; -import { resolvePluginList } from "./internal/core/plugins/resolve-plugin-list.js"; - export { resolveHardhatConfigPath } from "./internal/config-loading.js"; - -/** - * Creates an instances of the Hardhat Runtime Environment. - * - * @param config - The user's Hardhat configuration. - * @param userProvidedGlobalOptions - The global options provided by the - * user. - * @param projectRoot - The root of the Hardhat project. Hardhat expects this - * to be the root of the npm project containing your config file. If none is - * provided, it will use the root of the npm project that contains the CWD. - * @returns The Hardhat Runtime Environment. - */ -export async function createHardhatRuntimeEnvironment( - config: HardhatUserConfig, - userProvidedGlobalOptions: Partial = {}, - projectRoot?: string, - unsafeOptions: UnsafeHardhatRuntimeEnvironmentOptions = {}, -): Promise { - const resolvedProjectRoot = await resolveProjectRoot(projectRoot); - - if (unsafeOptions.resolvedPlugins === undefined) { - const plugins = [...builtinPlugins, ...(config.plugins ?? [])]; - - const resolvedPlugins = await resolvePluginList( - resolvedProjectRoot, - plugins, - ); - - unsafeOptions.resolvedPlugins = resolvedPlugins; - } - - if (unsafeOptions.globalOptionDefinitions === undefined) { - const pluginGlobalOptionDefinitions = buildGlobalOptionDefinitions( - unsafeOptions.resolvedPlugins, - ); - - const globalOptionDefinitions = new Map([ - ...BUILTIN_GLOBAL_OPTIONS_DEFINITIONS, - ...pluginGlobalOptionDefinitions, - ]); - - unsafeOptions.globalOptionDefinitions = globalOptionDefinitions; - } - - return HardhatRuntimeEnvironmentImplementation.create( - config, - userProvidedGlobalOptions, - resolvedProjectRoot, - unsafeOptions, - ); -} +export { createHardhatRuntimeEnvironment } from "./internal/hre-intialization.js"; diff --git a/v-next/hardhat/src/index.ts b/v-next/hardhat/src/index.ts index 894569a9d1..1048fa13fa 100644 --- a/v-next/hardhat/src/index.ts +++ b/v-next/hardhat/src/index.ts @@ -5,37 +5,11 @@ import type { HardhatRuntimeEnvironment } from "./types/hre.js"; import type { TaskManager } from "./types/tasks.js"; import type { UserInterruptionManager } from "./types/user-interruptions.js"; -import { createHardhatRuntimeEnvironment } from "./hre.js"; -import { - importUserConfig, - resolveHardhatConfigPath, -} from "./internal/config-loading.js"; -import { resolveProjectRoot } from "./internal/core/hre.js"; -import { - getGlobalHardhatRuntimeEnvironment, - setGlobalHardhatRuntimeEnvironment, -} from "./internal/global-hre-instance.js"; +import { getOrCreateGlobalHardhatRuntimeEnvironment } from "./internal/hre-intialization.js"; -let maybeHre: HardhatRuntimeEnvironment | undefined = - getGlobalHardhatRuntimeEnvironment(); - -// Note: We import the builtin plugins' types here, so that any type extension -// they may have gets loaded. -import "./internal/builtin-plugins/index.js"; - -if (maybeHre === undefined) { - /* eslint-disable no-restricted-syntax -- Allow top-level await here */ - const configPath = await resolveHardhatConfigPath(); - const projectRoot = await resolveProjectRoot(configPath); - const userConfig = await importUserConfig(configPath); - - maybeHre = await createHardhatRuntimeEnvironment(userConfig, {}, projectRoot); - /* eslint-enable no-restricted-syntax */ - - setGlobalHardhatRuntimeEnvironment(maybeHre); -} - -const hre: HardhatRuntimeEnvironment = maybeHre; +const hre: HardhatRuntimeEnvironment = + // eslint-disable-next-line no-restricted-syntax -- Allow top-level await here + await getOrCreateGlobalHardhatRuntimeEnvironment(); export const config: HardhatConfig = hre.config; export const tasks: TaskManager = hre.tasks; diff --git a/v-next/hardhat/src/internal/cli/main.ts b/v-next/hardhat/src/internal/cli/main.ts index b13293cf21..e269e606e0 100644 --- a/v-next/hardhat/src/internal/cli/main.ts +++ b/v-next/hardhat/src/internal/cli/main.ts @@ -13,7 +13,6 @@ import { isCi } from "@ignored/hardhat-vnext-utils/ci"; import { kebabToCamelCase } from "@ignored/hardhat-vnext-utils/string"; import debug from "debug"; -import { createHardhatRuntimeEnvironment } from "../../hre.js"; import { ArgumentType, type OptionDefinition, @@ -30,6 +29,7 @@ import { buildGlobalOptionDefinitions } from "../core/global-options.js"; import { resolveProjectRoot } from "../core/hre.js"; import { resolvePluginList } from "../core/plugins/resolve-plugin-list.js"; import { setGlobalHardhatRuntimeEnvironment } from "../global-hre-instance.js"; +import { createHardhatRuntimeEnvironment } from "../hre-intialization.js"; import { printErrorMessages } from "./error-handler.js"; import { getGlobalHelpString } from "./helpers/getGlobalHelpString.js"; diff --git a/v-next/hardhat/src/internal/hre-intialization.ts b/v-next/hardhat/src/internal/hre-intialization.ts new file mode 100644 index 0000000000..ab6a344ec8 --- /dev/null +++ b/v-next/hardhat/src/internal/hre-intialization.ts @@ -0,0 +1,103 @@ +import type { UnsafeHardhatRuntimeEnvironmentOptions } from "./core/types.js"; +import type { HardhatUserConfig } from "../types/config.js"; +import type { GlobalOptions } from "../types/global-options.js"; +import type { HardhatRuntimeEnvironment } from "../types/hre.js"; + +import { BUILTIN_GLOBAL_OPTIONS_DEFINITIONS } from "./builtin-global-options.js"; +// NOTE: As we import the builting plugins in this module, which is always used +// to initialize the HRE, all of the builtin plugins' type-extensions are loaded +import { builtinPlugins } from "./builtin-plugins/index.js"; +import { + importUserConfig, + resolveHardhatConfigPath, +} from "./config-loading.js"; +import { buildGlobalOptionDefinitions } from "./core/global-options.js"; +import { + resolveProjectRoot, + HardhatRuntimeEnvironmentImplementation, +} from "./core/hre.js"; +import { resolvePluginList } from "./core/plugins/resolve-plugin-list.js"; +import { + getGlobalHardhatRuntimeEnvironment, + setGlobalHardhatRuntimeEnvironment, +} from "./global-hre-instance.js"; + +/** + * Creates an instances of the Hardhat Runtime Environment. + * + * @param config - The user's Hardhat configuration. Note that the config + * doesn't have to come from a file, but if it does, you should provide its + * path as the `config` property in the `userProvidedGlobalOptions` object. + * @param userProvidedGlobalOptions - The global options provided by the + * user. + * @param projectRoot - The root of the Hardhat project. Hardhat expects this + * to be the root of the npm project containing your config file. If none is + * provided, it will use the root of the npm project that contains the CWD. + * @returns The Hardhat Runtime Environment. + */ +export async function createHardhatRuntimeEnvironment( + config: HardhatUserConfig, + userProvidedGlobalOptions: Partial = {}, + projectRoot?: string, + unsafeOptions: UnsafeHardhatRuntimeEnvironmentOptions = {}, +): Promise { + const resolvedProjectRoot = await resolveProjectRoot(projectRoot); + + if (unsafeOptions.resolvedPlugins === undefined) { + const plugins = [...builtinPlugins, ...(config.plugins ?? [])]; + + const resolvedPlugins = await resolvePluginList( + resolvedProjectRoot, + plugins, + ); + + unsafeOptions.resolvedPlugins = resolvedPlugins; + } + + if (unsafeOptions.globalOptionDefinitions === undefined) { + const pluginGlobalOptionDefinitions = buildGlobalOptionDefinitions( + unsafeOptions.resolvedPlugins, + ); + + const globalOptionDefinitions = new Map([ + ...BUILTIN_GLOBAL_OPTIONS_DEFINITIONS, + ...pluginGlobalOptionDefinitions, + ]); + + unsafeOptions.globalOptionDefinitions = globalOptionDefinitions; + } + + return HardhatRuntimeEnvironmentImplementation.create( + config, + userProvidedGlobalOptions, + resolvedProjectRoot, + unsafeOptions, + ); +} + +/** + * Gets the global Hardhat Runtime Environment, or creates it if it doesn't exist. + * + * This function is meant to be used when `hardhat` is imported as a library. + */ +export async function getOrCreateGlobalHardhatRuntimeEnvironment(): Promise { + let globalHre = getGlobalHardhatRuntimeEnvironment(); + + if (globalHre !== undefined) { + return globalHre; + } + + const configPath = await resolveHardhatConfigPath(); + const projectRoot = await resolveProjectRoot(configPath); + const userConfig = await importUserConfig(configPath); + + globalHre = await createHardhatRuntimeEnvironment( + userConfig, + { config: configPath }, + projectRoot, + ); + + setGlobalHardhatRuntimeEnvironment(globalHre); + + return globalHre; +} diff --git a/v-next/hardhat/test/internal/builtin-plugins/clean/task-action.ts b/v-next/hardhat/test/internal/builtin-plugins/clean/task-action.ts index d57a1ddef6..deff15bffa 100644 --- a/v-next/hardhat/test/internal/builtin-plugins/clean/task-action.ts +++ b/v-next/hardhat/test/internal/builtin-plugins/clean/task-action.ts @@ -13,9 +13,9 @@ import { } from "@ignored/hardhat-vnext-utils/fs"; import { useFixtureProject } from "@nomicfoundation/hardhat-test-utils"; -import { createHardhatRuntimeEnvironment } from "../../../../src/hre.js"; import cleanAction from "../../../../src/internal/builtin-plugins/clean/task-action.js"; import { getCacheDir } from "../../../../src/internal/global-dir.js"; +import { createHardhatRuntimeEnvironment } from "../../../../src/internal/hre-intialization.js"; let hre: HardhatRuntimeEnvironment; let globalCacheDir: string; diff --git a/v-next/hardhat/test/internal/builtin-plugins/console/task-action.ts b/v-next/hardhat/test/internal/builtin-plugins/console/task-action.ts index a06537778d..8ae1e4c6b3 100644 --- a/v-next/hardhat/test/internal/builtin-plugins/console/task-action.ts +++ b/v-next/hardhat/test/internal/builtin-plugins/console/task-action.ts @@ -13,8 +13,8 @@ import { exists, remove } from "@ignored/hardhat-vnext-utils/fs"; import { useFixtureProject } from "@nomicfoundation/hardhat-test-utils"; import debug from "debug"; -import { createHardhatRuntimeEnvironment } from "../../../../src/hre.js"; import consoleAction from "../../../../src/internal/builtin-plugins/console/task-action.js"; +import { createHardhatRuntimeEnvironment } from "../../../../src/internal/hre-intialization.js"; const log = debug("hardhat:test:console:task-action"); diff --git a/v-next/hardhat/test/internal/builtin-plugins/run/task-action.ts b/v-next/hardhat/test/internal/builtin-plugins/run/task-action.ts index 051efb0354..16b35cc0e6 100644 --- a/v-next/hardhat/test/internal/builtin-plugins/run/task-action.ts +++ b/v-next/hardhat/test/internal/builtin-plugins/run/task-action.ts @@ -8,8 +8,8 @@ import { useFixtureProject, } from "@nomicfoundation/hardhat-test-utils"; -import { createHardhatRuntimeEnvironment } from "../../../../src/hre.js"; import runScriptWithHardhat from "../../../../src/internal/builtin-plugins/run/task-action.js"; +import { createHardhatRuntimeEnvironment } from "../../../../src/internal/hre-intialization.js"; describe("run/task-action", function () { let hre: HardhatRuntimeEnvironment; diff --git a/v-next/hardhat/test/internal/cli/main.ts b/v-next/hardhat/test/internal/cli/main.ts index 76d7a980ce..5a464ac52a 100644 --- a/v-next/hardhat/test/internal/cli/main.ts +++ b/v-next/hardhat/test/internal/cli/main.ts @@ -23,7 +23,6 @@ import { } from "@nomicfoundation/hardhat-test-utils"; import chalk from "chalk"; -import { createHardhatRuntimeEnvironment } from "../../../src/hre.js"; import { main, parseGlobalOptions, @@ -37,6 +36,7 @@ import { task, } from "../../../src/internal/core/config.js"; import { resetGlobalHardhatRuntimeEnvironment } from "../../../src/internal/global-hre-instance.js"; +import { createHardhatRuntimeEnvironment } from "../../../src/internal/hre-intialization.js"; import { getHardhatVersion } from "../../../src/internal/utils/package.js"; import { ArgumentType } from "../../../src/types/arguments.js"; diff --git a/v-next/hardhat/test/hre/index.ts b/v-next/hardhat/test/internal/hre-intialization.ts similarity index 92% rename from v-next/hardhat/test/hre/index.ts rename to v-next/hardhat/test/internal/hre-intialization.ts index 81e79d12b8..1e934ec77d 100644 --- a/v-next/hardhat/test/hre/index.ts +++ b/v-next/hardhat/test/internal/hre-intialization.ts @@ -8,7 +8,6 @@ import { useFixtureProject, } from "@nomicfoundation/hardhat-test-utils"; -import { createHardhatRuntimeEnvironment } from "../../src/hre.js"; import { builtinPlugins } from "../../src/internal/builtin-plugins/index.js"; import { resolveHardhatConfigPath } from "../../src/internal/config-loading.js"; import { @@ -16,8 +15,12 @@ import { resetGlobalHardhatRuntimeEnvironment, setGlobalHardhatRuntimeEnvironment, } from "../../src/internal/global-hre-instance.js"; +import { + createHardhatRuntimeEnvironment, + getOrCreateGlobalHardhatRuntimeEnvironment, +} from "../../src/internal/hre-intialization.js"; -describe("HRE", () => { +describe("HRE intialization", () => { afterEach(() => { resetGlobalHardhatRuntimeEnvironment(); }); @@ -162,8 +165,12 @@ describe("HRE", () => { describe("programmatic API", () => { useFixtureProject("loaded-config"); + afterEach(() => { + resetGlobalHardhatRuntimeEnvironment(); + }); + it("should load the plugins from the config file", async () => { - const hre = await import("../../src/index.js"); + const hre = await getOrCreateGlobalHardhatRuntimeEnvironment(); const { testPlugin } = await import( "../fixture-projects/loaded-config/hardhat.config.js" ); @@ -172,10 +179,12 @@ describe("HRE", () => { }); it("should load the global options", async () => { - const hre = await import("../../src/index.js"); + const hre = await getOrCreateGlobalHardhatRuntimeEnvironment(); + + const configPath = await getRealPath("hardhat.config.ts"); assert.deepEqual(hre.globalOptions, { - config: "", + config: configPath, help: false, init: false, showStackTraces: false, From 14e8fdc36a463d6c98cd20b84ea6eaebdf6f4aff Mon Sep 17 00:00:00 2001 From: Patricio Palladino Date: Sat, 31 Aug 2024 16:47:16 +0000 Subject: [PATCH 08/14] Clean up the builtin plugins type-extensions story --- v-next/hardhat/src/config.ts | 4 ++++ v-next/hardhat/src/hre.ts | 4 ++++ v-next/hardhat/src/index.ts | 4 ++++ v-next/hardhat/src/internal/hre-intialization.ts | 2 -- 4 files changed, 12 insertions(+), 2 deletions(-) diff --git a/v-next/hardhat/src/config.ts b/v-next/hardhat/src/config.ts index 318ca01085..890b8c9453 100644 --- a/v-next/hardhat/src/config.ts +++ b/v-next/hardhat/src/config.ts @@ -2,3 +2,7 @@ export type * from "./internal/core/config.js"; export * from "./internal/core/config.js"; export type { HardhatUserConfig } from "./types/config.js"; + +// NOTE: We import the builtin plugins in this module, so that their +// type-extensions are loaded then the user imports `hardhat/config`. +import "./internal/builtin-plugins/index.js"; diff --git a/v-next/hardhat/src/hre.ts b/v-next/hardhat/src/hre.ts index d59fbdaa29..c923c5ddd8 100644 --- a/v-next/hardhat/src/hre.ts +++ b/v-next/hardhat/src/hre.ts @@ -1,2 +1,6 @@ +// NOTE: We import the builtin plugins in this module, so that their +// type-extensions are loaded then the user imports `hardhat/hre`. +import "./internal/builtin-plugins/index.js"; + export { resolveHardhatConfigPath } from "./internal/config-loading.js"; export { createHardhatRuntimeEnvironment } from "./internal/hre-intialization.js"; diff --git a/v-next/hardhat/src/index.ts b/v-next/hardhat/src/index.ts index 1048fa13fa..eeb3097ea3 100644 --- a/v-next/hardhat/src/index.ts +++ b/v-next/hardhat/src/index.ts @@ -7,6 +7,10 @@ import type { UserInterruptionManager } from "./types/user-interruptions.js"; import { getOrCreateGlobalHardhatRuntimeEnvironment } from "./internal/hre-intialization.js"; +// NOTE: We import the builtin plugins in this module, so that their +// type-extensions are loaded then the user imports `hardhat`. +import "./internal/builtin-plugins/index.js"; + const hre: HardhatRuntimeEnvironment = // eslint-disable-next-line no-restricted-syntax -- Allow top-level await here await getOrCreateGlobalHardhatRuntimeEnvironment(); diff --git a/v-next/hardhat/src/internal/hre-intialization.ts b/v-next/hardhat/src/internal/hre-intialization.ts index ab6a344ec8..3b3de8a77d 100644 --- a/v-next/hardhat/src/internal/hre-intialization.ts +++ b/v-next/hardhat/src/internal/hre-intialization.ts @@ -4,8 +4,6 @@ import type { GlobalOptions } from "../types/global-options.js"; import type { HardhatRuntimeEnvironment } from "../types/hre.js"; import { BUILTIN_GLOBAL_OPTIONS_DEFINITIONS } from "./builtin-global-options.js"; -// NOTE: As we import the builting plugins in this module, which is always used -// to initialize the HRE, all of the builtin plugins' type-extensions are loaded import { builtinPlugins } from "./builtin-plugins/index.js"; import { importUserConfig, From 42ef01a83fecf3f70478061a1fff598b50b5403a Mon Sep 17 00:00:00 2001 From: Patricio Palladino Date: Sat, 31 Aug 2024 18:06:38 +0000 Subject: [PATCH 09/14] Remove outdated linter rule --- v-next/hardhat/.eslintrc.cjs | 8 -------- 1 file changed, 8 deletions(-) diff --git a/v-next/hardhat/.eslintrc.cjs b/v-next/hardhat/.eslintrc.cjs index 70d742d97d..939317ab1c 100644 --- a/v-next/hardhat/.eslintrc.cjs +++ b/v-next/hardhat/.eslintrc.cjs @@ -1,11 +1,3 @@ const { createConfig } = require("../../config-v-next/eslint.cjs"); module.exports = createConfig(__filename); - -module.exports.rules["no-restricted-imports"][1].paths.push({ - // TODO: Rename this once we migrate to the official package names - name: "@ignored/hardhat-vnext-core", - importNames: ["createBaseHardhatRuntimeEnvironment"], - message: - "Use `createHardhatRuntimeEnvironment` from `hre.js` instead of this function.", -}); From 9edaa1f07f9f9d4540f9ef7b4c4ee342b5266d7a Mon Sep 17 00:00:00 2001 From: Patricio Palladino Date: Sat, 31 Aug 2024 18:16:34 +0000 Subject: [PATCH 10/14] Use the HRE to test the user interruptions manager --- .../user-interruptions-manager.ts | 106 ++---------------- 1 file changed, 11 insertions(+), 95 deletions(-) diff --git a/v-next/hardhat/test/internal/core/user-interruptions/user-interruptions-manager.ts b/v-next/hardhat/test/internal/core/user-interruptions/user-interruptions-manager.ts index e82bfb8722..ea35bc653e 100644 --- a/v-next/hardhat/test/internal/core/user-interruptions/user-interruptions-manager.ts +++ b/v-next/hardhat/test/internal/core/user-interruptions/user-interruptions-manager.ts @@ -1,50 +1,14 @@ import type { UserInterruptionHooks } from "../../../../src/types/hooks.js"; import assert from "node:assert/strict"; -import { before, describe, it } from "node:test"; +import { describe, it } from "node:test"; -import { HookManagerImplementation } from "../../../../src/internal/core/hook-manager.js"; -import { resolveProjectRoot } from "../../../../src/internal/core/hre.js"; -import { UserInterruptionManagerImplementation } from "../../../../src/internal/core/user-interruptions.js"; +import { HardhatRuntimeEnvironmentImplementation } from "../../../../src/internal/core/hre.js"; describe("UserInterruptionManager", () => { - let projectRoot: string; - - before(async () => { - projectRoot = await resolveProjectRoot(process.cwd()); - }); - describe("displayMessage", () => { it("Should call a dynamic handler with a given message from an interruptor", async () => { - const hookManager = new HookManagerImplementation(projectRoot, []); - const userInterruptionManager = new UserInterruptionManagerImplementation( - hookManager, - ); - - // TODO: Setting the context like this is a bit fragile. If this test - // breaks we should probably switch to initializing an entire HRE in these - // tests. - hookManager.setContext({ - /* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- - TODO: This is a temporary fix to land a refactor sooner without creating - more merge conflicts than needed. It will be fixed in a subsequent PR */ - config: { - tasks: [], - plugins: [], - paths: { - root: projectRoot, - cache: "", - artifacts: "", - tests: "", - }, - } as any, - /* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- - TODO: This is a temporary fix to land a refactor sooner without creating - more merge conflicts than needed. It will be fixed in a subsequent PR */ - globalOptions: {} as any, - hooks: hookManager, - interruptions: userInterruptionManager, - }); + const hre = await HardhatRuntimeEnvironmentImplementation.create({}, {}); let called = false; let givenInterruptor: string = ""; @@ -58,9 +22,9 @@ describe("UserInterruptionManager", () => { }, }; - hookManager.registerHandlers("userInterruptions", handlers); + hre.hooks.registerHandlers("userInterruptions", handlers); - await userInterruptionManager.displayMessage( + await hre.interruptions.displayMessage( "test-interruptor", "test-message", ); @@ -73,31 +37,7 @@ describe("UserInterruptionManager", () => { describe("requestInput", () => { it("Should call a dynamic handler with a given input description from an interruptor", async () => { - const hookManager = new HookManagerImplementation(projectRoot, []); - const userInterruptionManager = new UserInterruptionManagerImplementation( - hookManager, - ); - hookManager.setContext({ - /* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- - TODO: This is a temporary fix to land a refactor sooner without creating - more merge conflicts than needed. It will be fixed in a subsequent PR */ - config: { - tasks: [], - plugins: [], - paths: { - root: projectRoot, - cache: "", - artifacts: "", - tests: "", - }, - } as any, - /* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- - TODO: This is a temporary fix to land a refactor sooner without creating - more merge conflicts than needed. It will be fixed in a subsequent PR */ - globalOptions: {} as any, - hooks: hookManager, - interruptions: userInterruptionManager, - }); + const hre = await HardhatRuntimeEnvironmentImplementation.create({}, {}); let called = false; let givenInterruptor: string = ""; @@ -112,9 +52,9 @@ describe("UserInterruptionManager", () => { }, }; - hookManager.registerHandlers("userInterruptions", handlers); + hre.hooks.registerHandlers("userInterruptions", handlers); - const input = await userInterruptionManager.requestInput( + const input = await hre.interruptions.requestInput( "test-interruptor", "test-input-description", ); @@ -128,31 +68,7 @@ describe("UserInterruptionManager", () => { describe("requestSecretInput", () => { it("Should call a dynamic handler with a given input description from an interruptor", async () => { - const hookManager = new HookManagerImplementation(projectRoot, []); - const userInterruptionManager = new UserInterruptionManagerImplementation( - hookManager, - ); - hookManager.setContext({ - /* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- - TODO: This is a temporary fix to land a refactor sooner without creating - more merge conflicts than needed. It will be fixed in a subsequent PR */ - config: { - tasks: [], - plugins: [], - paths: { - root: projectRoot, - cache: "", - artifacts: "", - tests: "", - }, - } as any, - /* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- - TODO: This is a temporary fix to land a refactor sooner without creating - more merge conflicts than needed. It will be fixed in a subsequent PR */ - globalOptions: {} as any, - hooks: hookManager, - interruptions: userInterruptionManager, - }); + const hre = await HardhatRuntimeEnvironmentImplementation.create({}, {}); let called = false; let givenInterruptor: string = ""; @@ -167,9 +83,9 @@ describe("UserInterruptionManager", () => { }, }; - hookManager.registerHandlers("userInterruptions", handlers); + hre.hooks.registerHandlers("userInterruptions", handlers); - const input = await userInterruptionManager.requestSecretInput( + const input = await hre.interruptions.requestSecretInput( "test-interruptor", "test-input-description", ); From 56b44f4e933e734fc4637b99e8bc68bfdc2c55fc Mon Sep 17 00:00:00 2001 From: Patricio Palladino Date: Sat, 31 Aug 2024 18:20:51 +0000 Subject: [PATCH 11/14] Use the HRE to test the configuration variables --- .../configuration-variables.ts | 53 +++++-------------- 1 file changed, 13 insertions(+), 40 deletions(-) diff --git a/v-next/hardhat/test/internal/core/configuration-variables/configuration-variables.ts b/v-next/hardhat/test/internal/core/configuration-variables/configuration-variables.ts index e6cb595ff4..484e8a2a52 100644 --- a/v-next/hardhat/test/internal/core/configuration-variables/configuration-variables.ts +++ b/v-next/hardhat/test/internal/core/configuration-variables/configuration-variables.ts @@ -1,3 +1,5 @@ +import type { HardhatRuntimeEnvironment } from "../../../../src/types/hre.js"; + import assert from "node:assert/strict"; import { before, describe, it } from "node:test"; @@ -5,47 +7,18 @@ import { HardhatError } from "@ignored/hardhat-vnext-errors"; import { assertRejectsWithHardhatError } from "@nomicfoundation/hardhat-test-utils"; import { ResolvedConfigurationVariableImplementation } from "../../../../src/internal/core/configuration-variables.js"; -import { HookManagerImplementation } from "../../../../src/internal/core/hook-manager.js"; -import { resolveProjectRoot } from "../../../../src/internal/core/hre.js"; -import { UserInterruptionManagerImplementation } from "../../../../src/internal/core/user-interruptions.js"; +import { HardhatRuntimeEnvironmentImplementation } from "../../../../src/internal/core/hre.js"; describe("ResolvedConfigurationVariable", () => { - let hookManager: HookManagerImplementation; + let hre: HardhatRuntimeEnvironment; before(async () => { - const projectRoot = await resolveProjectRoot(process.cwd()); - - hookManager = new HookManagerImplementation(projectRoot, []); - const userInterruptionsManager = new UserInterruptionManagerImplementation( - hookManager, - ); - - hookManager.setContext({ - /* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- - TODO: This is a temporary fix to land a refactor sooner without creating - more merge conflicts than needed. It will be fixed in a subsequent PR */ - config: { - tasks: [], - plugins: [], - paths: { - root: projectRoot, - cache: "", - artifacts: "", - tests: "", - }, - } as any, - hooks: hookManager, - /* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- - TODO: This is a temporary fix to land a refactor sooner without creating - more merge conflicts than needed. It will be fixed in a subsequent PR */ - globalOptions: {} as any, - interruptions: userInterruptionsManager, - }); + hre = await HardhatRuntimeEnvironmentImplementation.create({}, {}); }); it("should return the value of a string variable", async () => { const variable = new ResolvedConfigurationVariableImplementation( - hookManager, + hre.hooks, "foo", ); @@ -54,7 +27,7 @@ describe("ResolvedConfigurationVariable", () => { it("should return the value of a configuration variable from an environment variable", async () => { const variable = new ResolvedConfigurationVariableImplementation( - hookManager, + hre.hooks, { name: "foo", _type: "ConfigurationVariable" }, ); @@ -67,7 +40,7 @@ describe("ResolvedConfigurationVariable", () => { it("should throw if the environment variable is not found", async () => { const variable = new ResolvedConfigurationVariableImplementation( - hookManager, + hre.hooks, { name: "foo", _type: "ConfigurationVariable" }, ); @@ -80,7 +53,7 @@ describe("ResolvedConfigurationVariable", () => { it("should return the cached value if called multiple times", async () => { const variable = new ResolvedConfigurationVariableImplementation( - hookManager, + hre.hooks, { name: "foo", _type: "ConfigurationVariable" }, ); @@ -97,7 +70,7 @@ describe("ResolvedConfigurationVariable", () => { it("should return the value of a configuration variable as a URL", async () => { const variable = new ResolvedConfigurationVariableImplementation( - hookManager, + hre.hooks, { name: "foo", _type: "ConfigurationVariable" }, ); @@ -110,7 +83,7 @@ describe("ResolvedConfigurationVariable", () => { it("should throw if the configuration variable is not a valid URL", async () => { const variable = new ResolvedConfigurationVariableImplementation( - hookManager, + hre.hooks, { name: "foo", _type: "ConfigurationVariable" }, ); @@ -129,7 +102,7 @@ describe("ResolvedConfigurationVariable", () => { it("should return the value of a configuration variable as a BigInt", async () => { const variable = new ResolvedConfigurationVariableImplementation( - hookManager, + hre.hooks, { name: "foo", _type: "ConfigurationVariable" }, ); @@ -142,7 +115,7 @@ describe("ResolvedConfigurationVariable", () => { it("should throw if the configuration variable is not a valid BigInt", async () => { const variable = new ResolvedConfigurationVariableImplementation( - hookManager, + hre.hooks, { name: "foo", _type: "ConfigurationVariable" }, ); From 3b73a77794552a9e67ef1e76448dd1bc3f1be5a9 Mon Sep 17 00:00:00 2001 From: Patricio Palladino Date: Sat, 31 Aug 2024 18:27:55 +0000 Subject: [PATCH 12/14] Use valid user configs to test the hook manager --- .../test/internal/core/hook-manager.ts | 112 ++---------------- 1 file changed, 8 insertions(+), 104 deletions(-) diff --git a/v-next/hardhat/test/internal/core/hook-manager.ts b/v-next/hardhat/test/internal/core/hook-manager.ts index 93b46fea47..c01f99a014 100644 --- a/v-next/hardhat/test/internal/core/hook-manager.ts +++ b/v-next/hardhat/test/internal/core/hook-manager.ts @@ -173,19 +173,7 @@ describe("HookManager", () => { }); it("should use plugins during parallel handlers runs", async () => { - /* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- - TODO: This is a temporary fix to land a refactor sooner without creating - more merge conflicts than needed. It will be fixed in a subsequent PR.*/ - const originalConfig: HardhatUserConfig = { - plugins: [], - tasks: [], - paths: { - root: projectRoot, - cache: "", - artifacts: "", - tests: "", - }, - } as any; + const originalConfig: HardhatUserConfig = {}; hookManager.registerHandlers("config", { validateUserConfig: async ( @@ -232,19 +220,7 @@ describe("HookManager", () => { }, }; - /* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- - TODO: This is a temporary fix to land a refactor sooner without creating - more merge conflicts than needed. It will be fixed in a subsequent PR. */ - const expectedConfig: HardhatUserConfig = { - plugins: [], - tasks: [], - paths: { - root: projectRoot, - cache: "", - artifacts: "", - tests: "", - }, - } as any; + const expectedConfig: HardhatUserConfig = {}; const manager = new HookManagerImplementation(projectRoot, [ examplePlugin, @@ -368,19 +344,7 @@ describe("HookManager", () => { it("should return the default implementation if no other handlers are provided", async () => { const notExpectedConfig = {}; - /* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- - TODO: This is a temporary fix to land a refactor sooner without creating - more merge conflicts than needed. It will be fixed in a subsequent PR */ - const defaultImplementationVersionOfConfig: HardhatUserConfig = { - plugins: [], - tasks: [], - paths: { - root: projectRoot, - cache: "", - artifacts: "", - tests: "", - }, - } as any; + const defaultImplementationVersionOfConfig: HardhatUserConfig = {}; const resultConfig = await hookManager.runHandlerChain( "config", @@ -458,19 +422,7 @@ describe("HookManager", () => { }); it("should pass the parameters directly for config hooks", async () => { - /* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- - TODO: This is a temporary fix to land a refactor sooner without creating - more merge conflicts than needed. It will be fixed in a subsequent PR */ - const expectedConfig: HardhatUserConfig = { - plugins: [], - tasks: [], - paths: { - root: projectRoot, - cache: "", - artifacts: "", - tests: "", - }, - } as any; + const expectedConfig: HardhatUserConfig = {}; hookManager.registerHandlers("config", { extendUserConfig: async ( @@ -686,19 +638,7 @@ describe("HookManager", () => { }); it("Should stop config handlers having access to the hook context", async () => { - /* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- - TODO: This is a temporary fix to land a refactor sooner without creating - more merge conflicts than needed. It will be fixed in a subsequent PR */ - const expectedConfig: HardhatUserConfig = { - plugins: [], - tasks: [], - paths: { - root: projectRoot, - cache: "", - artifacts: "", - tests: "", - }, - } as any; + const expectedConfig: HardhatUserConfig = {}; hookManager.registerHandlers("config", { validateUserConfig: async ( @@ -759,19 +699,7 @@ describe("HookManager", () => { }); it("Should return an empty result set if no handlers are provided", async () => { - /* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- - TODO: This is a temporary fix to land a refactor sooner without creating - more merge conflicts than needed. It will be fixed in a subsequent PR */ - const originalConfig: HardhatUserConfig = { - plugins: [], - tasks: [], - paths: { - root: projectRoot, - cache: "", - artifacts: "", - tests: "", - }, - } as any; + const originalConfig: HardhatUserConfig = {}; const results = await hookManager.runParallelHandlers( "config", @@ -783,19 +711,7 @@ describe("HookManager", () => { }); it("Should return a result per handler", async () => { - /* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- - TODO: This is a temporary fix to land a refactor sooner without creating - more merge conflicts than needed. It will be fixed in a subsequent PR */ - const originalConfig: HardhatUserConfig = { - plugins: [], - tasks: [], - paths: { - root: projectRoot, - cache: "", - artifacts: "", - tests: "", - }, - } as any; + const originalConfig: HardhatUserConfig = {}; hookManager.registerHandlers("config", { validateUserConfig: async ( @@ -872,19 +788,7 @@ describe("HookManager", () => { }); it("Should not pass the hook context for config", async () => { - /* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- - TODO: This is a temporary fix to land a refactor sooner without creating - more merge conflicts than needed. It will be fixed in a subsequent PR */ - const expectedConfig: HardhatUserConfig = { - plugins: [], - tasks: [], - paths: { - root: projectRoot, - cache: "", - artifacts: "", - tests: "", - }, - } as any; + const expectedConfig: HardhatUserConfig = {}; const validationError = { path: [], From e8aea82ef1e41a84cf09720456e4354bc866b2d6 Mon Sep 17 00:00:00 2001 From: Patricio Palladino Date: Sat, 31 Aug 2024 18:49:55 +0000 Subject: [PATCH 13/14] Use the HRE to test the hook manager --- .../test/internal/core/hook-manager.ts | 358 ++++-------------- 1 file changed, 79 insertions(+), 279 deletions(-) diff --git a/v-next/hardhat/test/internal/core/hook-manager.ts b/v-next/hardhat/test/internal/core/hook-manager.ts index c01f99a014..a8f274c93f 100644 --- a/v-next/hardhat/test/internal/core/hook-manager.ts +++ b/v-next/hardhat/test/internal/core/hook-manager.ts @@ -4,7 +4,6 @@ sequential tests require casting - see the `runSequentialHandlers` describe */ import type { HardhatUserConfig } from "../../../src/config.js"; import type { ConfigurationVariable } from "../../../src/types/config.js"; import type { - HookManager, ConfigHooks, HardhatUserConfigValidationError, HookContext, @@ -13,8 +12,6 @@ import type { } from "../../../src/types/hooks.js"; import type { HardhatRuntimeEnvironment } from "../../../src/types/hre.js"; import type { HardhatPlugin } from "../../../src/types/plugins.js"; -import type { TaskManager, Task } from "../../../src/types/tasks.js"; -import type { UserInterruptionManager } from "../../../src/types/user-interruptions.js"; import assert from "node:assert/strict"; import { describe, it, beforeEach, before } from "node:test"; @@ -24,8 +21,10 @@ import { ensureError } from "@ignored/hardhat-vnext-utils/error"; import { assertRejectsWithHardhatError } from "@nomicfoundation/hardhat-test-utils"; import { HookManagerImplementation } from "../../../src/internal/core/hook-manager.js"; -import { resolveProjectRoot } from "../../../src/internal/core/hre.js"; -import { UserInterruptionManagerImplementation } from "../../../src/internal/core/user-interruptions.js"; +import { + HardhatRuntimeEnvironmentImplementation, + resolveProjectRoot, +} from "../../../src/internal/core/hre.js"; describe("HookManager", () => { let projectRoot: string; @@ -36,11 +35,13 @@ describe("HookManager", () => { describe("plugin hooks", () => { describe("running", () => { - let hookManager: HookManager; + let hre: HardhatRuntimeEnvironment; + let foceConfigValidationErrorFromPlugin: boolean; let sequence: string[]; - beforeEach(() => { + beforeEach(async () => { sequence = []; + foceConfigValidationErrorFromPlugin = false; const examplePlugin: HardhatPlugin = { id: "example", @@ -50,12 +51,16 @@ describe("HookManager", () => { validateUserConfig: async ( _config: HardhatUserConfig, ): Promise => { - return [ - { - path: [], - message: "FromPlugin", - }, - ]; + if (foceConfigValidationErrorFromPlugin) { + return [ + { + path: [], + message: "FromPlugin", + }, + ]; + } + + return []; }, extendUserConfig: async ( config: HardhatUserConfig, @@ -88,40 +93,14 @@ describe("HookManager", () => { }, }; - const manager = new HookManagerImplementation(projectRoot, [ - examplePlugin, - ]); - - const userInterruptionsManager = - new UserInterruptionManagerImplementation(hookManager); - - manager.setContext({ - /* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- - TODO: This is a temporary fix to land a refactor sooner without creating - more merge conflicts than needed. It will be fixed in a subsequent PR */ - config: { - tasks: [], - plugins: [], - paths: { - root: projectRoot, - cache: "", - artifacts: "", - tests: "", - }, - } as any, - hooks: hookManager, - /* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- - TODO: This is a temporary fix to land a refactor sooner without creating - more merge conflicts than needed. It will be fixed in a subsequent PR */ - globalOptions: {} as any, - interruptions: userInterruptionsManager, - }); - - hookManager = manager; + hre = await HardhatRuntimeEnvironmentImplementation.create( + { plugins: [examplePlugin] }, + {}, + ); }); it("should use plugins during handler runs", async () => { - hookManager.registerHandlers("config", { + hre.hooks.registerHandlers("config", { extendUserConfig: async ( config: HardhatUserConfig, next: (nextConfig: HardhatUserConfig) => Promise, @@ -134,7 +113,11 @@ describe("HookManager", () => { }, }); - await hookManager.runHandlerChain( + // We clear the sequense here, as we used the plugin already during the + // initialization of the HRE + sequence = []; + + await hre.hooks.runHandlerChain( "config", "extendUserConfig", [{}], @@ -154,7 +137,7 @@ describe("HookManager", () => { }); it("Should use plugins during a sequential run", async () => { - hookManager.registerHandlers("hre", { + hre.hooks.registerHandlers("hre", { testExample: async ( _context: HookContext, _input: string, @@ -163,7 +146,7 @@ describe("HookManager", () => { }, } as Partial); - const result = await hookManager.runSequentialHandlers( + const result = await hre.hooks.runSequentialHandlers( "hre", "testExample" as any, ["input"], @@ -175,7 +158,7 @@ describe("HookManager", () => { it("should use plugins during parallel handlers runs", async () => { const originalConfig: HardhatUserConfig = {}; - hookManager.registerHandlers("config", { + hre.hooks.registerHandlers("config", { validateUserConfig: async ( _config: HardhatUserConfig, ): Promise => { @@ -188,7 +171,9 @@ describe("HookManager", () => { }, }); - const results = await hookManager.runParallelHandlers( + foceConfigValidationErrorFromPlugin = true; + + const results = await hre.hooks.runParallelHandlers( "config", "validateUserConfig", [originalConfig], @@ -307,46 +292,19 @@ describe("HookManager", () => { }); describe("dynamic hooks", () => { - describe("runHandlerChain", () => { - let hookManager: HookManager; - - beforeEach(() => { - const manager = new HookManagerImplementation(projectRoot, []); - - const userInterruptionsManager = - new UserInterruptionManagerImplementation(hookManager); - - manager.setContext({ - /* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- - TODO: This is a temporary fix to land a refactor sooner without creating - more merge conflicts than needed. It will be fixed in a subsequent PR */ - config: { - tasks: [], - plugins: [], - paths: { - root: projectRoot, - cache: "", - artifacts: "", - tests: "", - }, - } as any, - hooks: hookManager, - /* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- - TODO: This is a temporary fix to land a refactor sooner without creating - more merge conflicts than needed. It will be fixed in a subsequent PR */ - globalOptions: {} as any, - interruptions: userInterruptionsManager, - }); + let hre: HardhatRuntimeEnvironment; - hookManager = manager; - }); + beforeEach(async () => { + hre = await HardhatRuntimeEnvironmentImplementation.create({}, {}); + }); + describe("runHandlerChain", () => { it("should return the default implementation if no other handlers are provided", async () => { const notExpectedConfig = {}; const defaultImplementationVersionOfConfig: HardhatUserConfig = {}; - const resultConfig = await hookManager.runHandlerChain( + const resultConfig = await hre.hooks.runHandlerChain( "config", "extendUserConfig", [notExpectedConfig], @@ -361,7 +319,7 @@ describe("HookManager", () => { it("should run the handlers as a chain finishing with the default implementation", async () => { const sequence: string[] = []; - hookManager.registerHandlers("config", { + hre.hooks.registerHandlers("config", { extendUserConfig: async ( config: HardhatUserConfig, next: (nextConfig: HardhatUserConfig) => Promise, @@ -374,7 +332,7 @@ describe("HookManager", () => { }, }); - hookManager.registerHandlers("config", { + hre.hooks.registerHandlers("config", { extendUserConfig: async ( config: HardhatUserConfig, next: (nextConfig: HardhatUserConfig) => Promise, @@ -387,7 +345,7 @@ describe("HookManager", () => { }, }); - hookManager.registerHandlers("config", { + hre.hooks.registerHandlers("config", { extendUserConfig: async ( config: HardhatUserConfig, next: (nextConfig: HardhatUserConfig) => Promise, @@ -400,7 +358,7 @@ describe("HookManager", () => { }, }); - await hookManager.runHandlerChain( + await hre.hooks.runHandlerChain( "config", "extendUserConfig", [{}], @@ -424,7 +382,7 @@ describe("HookManager", () => { it("should pass the parameters directly for config hooks", async () => { const expectedConfig: HardhatUserConfig = {}; - hookManager.registerHandlers("config", { + hre.hooks.registerHandlers("config", { extendUserConfig: async ( config: HardhatUserConfig, next: (nextConfig: HardhatUserConfig) => Promise, @@ -441,7 +399,7 @@ describe("HookManager", () => { }, }); - const resultConfig = await hookManager.runHandlerChain( + const resultConfig = await hre.hooks.runHandlerChain( "config", "extendUserConfig", [expectedConfig], @@ -465,7 +423,7 @@ describe("HookManager", () => { name: "example", }; - hookManager.registerHandlers("configurationVariables", { + hre.hooks.registerHandlers("configurationVariables", { fetchValue: async ( context: HookContext, variable: ConfigurationVariable, @@ -491,7 +449,7 @@ describe("HookManager", () => { }, }); - const resultValue = await hookManager.runHandlerChain( + const resultValue = await hre.hooks.runHandlerChain( "configurationVariables", "fetchValue", [exampleConfigVar], @@ -528,56 +486,18 @@ describe("HookManager", () => { * } */ describe("runSequentialHandlers", () => { - let hookManager: HookManager; - - beforeEach(() => { - const manager = new HookManagerImplementation(projectRoot, []); - - const userInterruptionsManager = - new UserInterruptionManagerImplementation(hookManager); - - manager.setContext({ - /* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- - TODO: This is a temporary fix to land a refactor sooner without creating - more merge conflicts than needed. It will be fixed in a subsequent PR */ - config: { - tasks: [], - plugins: [], - paths: { - root: projectRoot, - cache: "", - artifacts: "", - tests: "", - }, - } as any, - hooks: hookManager, - /* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- - TODO: This is a temporary fix to land a refactor sooner without creating - more merge conflicts than needed. It will be fixed in a subsequent PR */ - globalOptions: {} as any, - interruptions: userInterruptionsManager, - }); - - hookManager = manager; - }); - it("Should return the empty set if no handlers are registered", async () => { - const mockHre = buildMockHardhatRuntimeEnvironment( - projectRoot, - hookManager, - ); - - const resultHre = await hookManager.runSequentialHandlers( + const resultHreCreated = await hre.hooks.runSequentialHandlers( "hre", "created", - [mockHre], + [hre], ); - assert.deepEqual(resultHre, []); + assert.deepEqual(resultHreCreated, []); }); it("Should return a return entry per handler", async () => { - hookManager.registerHandlers("hre", { + hre.hooks.registerHandlers("hre", { testExample: async ( _context: HookContext, _input: string, @@ -586,7 +506,7 @@ describe("HookManager", () => { }, } as Partial); - hookManager.registerHandlers("hre", { + hre.hooks.registerHandlers("hre", { testExample: async ( _context: HookContext, _input: string, @@ -595,7 +515,7 @@ describe("HookManager", () => { }, } as Partial); - hookManager.registerHandlers("hre", { + hre.hooks.registerHandlers("hre", { testExample: async ( _context: HookContext, _input: string, @@ -604,7 +524,7 @@ describe("HookManager", () => { }, } as Partial); - const result = await hookManager.runSequentialHandlers( + const result = await hre.hooks.runSequentialHandlers( "hre", "testExample" as keyof HardhatHooks["hre"], ["input"] as any, @@ -614,7 +534,7 @@ describe("HookManager", () => { }); it("Should let handlers access the passed context (for non-config hooks)", async () => { - hookManager.registerHandlers("hre", { + hre.hooks.registerHandlers("hre", { testExample: async ( context: HookContext, input: string, @@ -628,7 +548,7 @@ describe("HookManager", () => { }, } as Partial); - const result = await hookManager.runSequentialHandlers( + const result = await hre.hooks.runSequentialHandlers( "hre", "testExample" as any, ["input"], @@ -640,7 +560,7 @@ describe("HookManager", () => { it("Should stop config handlers having access to the hook context", async () => { const expectedConfig: HardhatUserConfig = {}; - hookManager.registerHandlers("config", { + hre.hooks.registerHandlers("config", { validateUserConfig: async ( config: HardhatUserConfig, ): Promise => { @@ -654,7 +574,7 @@ describe("HookManager", () => { }, }); - const validationResult = await hookManager.runSequentialHandlers( + const validationResult = await hre.hooks.runSequentialHandlers( "config", "validateUserConfig", [expectedConfig], @@ -665,43 +585,10 @@ describe("HookManager", () => { }); describe("runParallelHandlers", () => { - let hookManager: HookManager; - - beforeEach(() => { - const manager = new HookManagerImplementation(projectRoot, []); - - const userInterruptionsManager = - new UserInterruptionManagerImplementation(hookManager); - - manager.setContext({ - /* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- - TODO: This is a temporary fix to land a refactor sooner without creating - more merge conflicts than needed. It will be fixed in a subsequent PR */ - config: { - tasks: [], - plugins: [], - paths: { - root: projectRoot, - cache: "", - artifacts: "", - tests: "", - }, - } as any, - hooks: hookManager, - /* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- - TODO: This is a temporary fix to land a refactor sooner without creating - more merge conflicts than needed. It will be fixed in a subsequent PR */ - globalOptions: {} as any, - interruptions: userInterruptionsManager, - }); - - hookManager = manager; - }); - it("Should return an empty result set if no handlers are provided", async () => { const originalConfig: HardhatUserConfig = {}; - const results = await hookManager.runParallelHandlers( + const results = await hre.hooks.runParallelHandlers( "config", "validateUserConfig", [originalConfig], @@ -713,7 +600,7 @@ describe("HookManager", () => { it("Should return a result per handler", async () => { const originalConfig: HardhatUserConfig = {}; - hookManager.registerHandlers("config", { + hre.hooks.registerHandlers("config", { validateUserConfig: async ( _config: HardhatUserConfig, ): Promise => { @@ -726,7 +613,7 @@ describe("HookManager", () => { }, }); - hookManager.registerHandlers("config", { + hre.hooks.registerHandlers("config", { validateUserConfig: async ( _config: HardhatUserConfig, ): Promise => { @@ -739,7 +626,7 @@ describe("HookManager", () => { }, }); - const results = await hookManager.runParallelHandlers( + const results = await hre.hooks.runParallelHandlers( "config", "validateUserConfig", [originalConfig], @@ -762,26 +649,21 @@ describe("HookManager", () => { }); it("Should pass the context to the handler (for non-config)", async () => { - const mockHre = buildMockHardhatRuntimeEnvironment( - projectRoot, - hookManager, - ); - - hookManager.registerHandlers("hre", { + hre.hooks.registerHandlers("hre", { created: async ( context: HookContext, - hre: HardhatRuntimeEnvironment, + hreInHandler: HardhatRuntimeEnvironment, ): Promise => { assert( context !== null && typeof context === "object", "hook context should be passed", ); - assert.equal(hre, mockHre); + assert.equal(hreInHandler, hre); }, }); - const result = await hookManager.runParallelHandlers("hre", "created", [ - mockHre, + const result = await hre.hooks.runParallelHandlers("hre", "created", [ + hre, ]); assert.deepEqual(result, [undefined]); @@ -795,7 +677,7 @@ describe("HookManager", () => { message: "first", }; - hookManager.registerHandlers("config", { + hre.hooks.registerHandlers("config", { validateUserConfig: async ( config: HardhatUserConfig, ): Promise => { @@ -804,7 +686,7 @@ describe("HookManager", () => { }, }); - const results = await hookManager.runParallelHandlers( + const results = await hre.hooks.runParallelHandlers( "config", "validateUserConfig", [expectedConfig], @@ -815,39 +697,6 @@ describe("HookManager", () => { }); describe("unregisterHandlers", () => { - let hookManager: HookManager; - - beforeEach(() => { - const manager = new HookManagerImplementation(projectRoot, []); - - const userInterruptionsManager = - new UserInterruptionManagerImplementation(hookManager); - - manager.setContext({ - /* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- - TODO: This is a temporary fix to land a refactor sooner without creating - more merge conflicts than needed. It will be fixed in a subsequent PR */ - config: { - tasks: [], - plugins: [], - paths: { - root: projectRoot, - cache: "", - artifacts: "", - tests: "", - }, - } as any, - hooks: hookManager, - /* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- - TODO: This is a temporary fix to land a refactor sooner without creating - more merge conflicts than needed. It will be fixed in a subsequent PR */ - globalOptions: {} as any, - interruptions: userInterruptionsManager, - }); - - hookManager = manager; - }); - it("Should unhook a handler", async () => { const hookCategory = { validateUserConfig: async ( @@ -857,11 +706,11 @@ describe("HookManager", () => { }, }; - hookManager.registerHandlers("config", hookCategory); + hre.hooks.registerHandlers("config", hookCategory); - hookManager.unregisterHandlers("config", hookCategory); + hre.hooks.unregisterHandlers("config", hookCategory); - const results = await hookManager.runParallelHandlers( + const results = await hre.hooks.runParallelHandlers( "config", "validateUserConfig", [ @@ -917,14 +766,14 @@ describe("HookManager", () => { }; // Arrange - hookManager.registerHandlers("config", firstHook); - hookManager.registerHandlers("config", secondHook); - hookManager.registerHandlers("config", thirdHook); + hre.hooks.registerHandlers("config", firstHook); + hre.hooks.registerHandlers("config", secondHook); + hre.hooks.registerHandlers("config", thirdHook); // Act - hookManager.unregisterHandlers("config", secondHook); + hre.hooks.unregisterHandlers("config", secondHook); - const results = await hookManager.runParallelHandlers( + const results = await hre.hooks.runParallelHandlers( "config", "validateUserConfig", [ @@ -961,57 +810,8 @@ describe("HookManager", () => { }, }; - hookManager.unregisterHandlers("config", exampleHook); + hre.hooks.unregisterHandlers("config", exampleHook); }); }); }); }); - -function buildMockHardhatRuntimeEnvironment( - projectRoot: string, - hookManager: HookManager, -): HardhatRuntimeEnvironment { - const mockInteruptionManager: UserInterruptionManager = { - displayMessage: async () => {}, - requestInput: async () => "", - requestSecretInput: async () => "", - uninterrupted: async ( - f: () => ReturnT, - ): Promise> => { - /* eslint-disable-next-line @typescript-eslint/return-await, @typescript-eslint/await-thenable -- this is following the pattern in the real implementation */ - return await f(); - }, - }; - - const mockTaskManager: TaskManager = { - getTask: () => { - throw new Error("Method not implemented."); - }, - rootTasks: new Map(), - }; - - const mockHre: HardhatRuntimeEnvironment = { - hooks: hookManager, - /* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- - TODO: This is a temporary fix to land a refactor sooner without creating - more merge conflicts than needed. It will be fixed in a subsequent PR */ - config: { - tasks: [], - plugins: [], - paths: { - root: projectRoot, - cache: "", - artifacts: "", - tests: "", - }, - } as any, - tasks: mockTaskManager, - /* eslint-disable-next-line @typescript-eslint/consistent-type-assertions -- - TODO: This is a temporary fix to land a refactor sooner without creating - more merge conflicts than needed. It will be fixed in a subsequent PR */ - globalOptions: {} as any, - interruptions: mockInteruptionManager, - }; - - return mockHre; -} From f155c05f0ae7e029fd024d3eef1922da45a3ca4c Mon Sep 17 00:00:00 2001 From: Patricio Palladino Date: Mon, 2 Sep 2024 18:56:12 +0000 Subject: [PATCH 14/14] Fix typo --- v-next/hardhat/test/internal/core/hook-manager.ts | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/v-next/hardhat/test/internal/core/hook-manager.ts b/v-next/hardhat/test/internal/core/hook-manager.ts index a8f274c93f..10dc3b9056 100644 --- a/v-next/hardhat/test/internal/core/hook-manager.ts +++ b/v-next/hardhat/test/internal/core/hook-manager.ts @@ -36,12 +36,12 @@ describe("HookManager", () => { describe("plugin hooks", () => { describe("running", () => { let hre: HardhatRuntimeEnvironment; - let foceConfigValidationErrorFromPlugin: boolean; + let forceConfigValidationErrorFromPlugin: boolean; let sequence: string[]; beforeEach(async () => { sequence = []; - foceConfigValidationErrorFromPlugin = false; + forceConfigValidationErrorFromPlugin = false; const examplePlugin: HardhatPlugin = { id: "example", @@ -51,7 +51,7 @@ describe("HookManager", () => { validateUserConfig: async ( _config: HardhatUserConfig, ): Promise => { - if (foceConfigValidationErrorFromPlugin) { + if (forceConfigValidationErrorFromPlugin) { return [ { path: [], @@ -171,7 +171,7 @@ describe("HookManager", () => { }, }); - foceConfigValidationErrorFromPlugin = true; + forceConfigValidationErrorFromPlugin = true; const results = await hre.hooks.runParallelHandlers( "config",