@nomiclabs/buidler
Version:
Buidler is an extensible developer tool that helps smart contract developers increase productivity by reliably bringing together the tools they want.
90 lines (73 loc) • 2.4 kB
text/typescript
import debug from "debug";
import findUp from "find-up";
import path from "path";
const log = debug("buidler:core:execution-mode");
/**
* This module defines different Buidler execution modes and autodetects them.
*
* IMPORTANT: This will have to be revisited once Yarn PnP and npm's tink get
* widely adopted.
*/
export enum ExecutionMode {
EXECUTION_MODE_TS_NODE_TESTS,
EXECUTION_MODE_LINKED,
EXECUTION_MODE_GLOBAL_INSTALLATION,
EXECUTION_MODE_LOCAL_INSTALLATION,
}
const workingDirectoryOnLoad = process.cwd();
export function getExecutionMode(): ExecutionMode {
const isInstalled = __filename.includes("node_modules");
if (!isInstalled) {
// When running the tests with ts-node we set the CWD to the root of
// buidler-core. We could check if the __filename ends with .ts
if (__dirname.startsWith(workingDirectoryOnLoad)) {
return ExecutionMode.EXECUTION_MODE_TS_NODE_TESTS;
}
return ExecutionMode.EXECUTION_MODE_LINKED;
}
try {
if (require("is-installed-globally")) {
return ExecutionMode.EXECUTION_MODE_GLOBAL_INSTALLATION;
}
} catch (error) {
log(
"Failed to load is-installed-globally. Using alternative local installation detection\n",
error
);
if (!alternativeIsLocalInstallation()) {
return ExecutionMode.EXECUTION_MODE_GLOBAL_INSTALLATION;
}
}
return ExecutionMode.EXECUTION_MODE_LOCAL_INSTALLATION;
}
/**
* Checks whether we're using Buidler in development mode (that is, we're working _on_ Buidler).
*/
export function isLocalDev(): boolean {
const executionMode = getExecutionMode();
return (
executionMode === ExecutionMode.EXECUTION_MODE_LINKED ||
executionMode === ExecutionMode.EXECUTION_MODE_TS_NODE_TESTS
);
}
/**
* This is a somewhat more limited detection, but we use it if
* is-installed-globally fails.
*
* If a user installs buidler locally, and executes it from outside the
* directory that contains the `node_module` with the installation, this will
* fail and return `false`.
*/
function alternativeIsLocalInstallation(): boolean {
let cwd = workingDirectoryOnLoad;
while (true) {
const nodeModules = findUp.sync("node_modules", { cwd });
if (nodeModules === null) {
return false;
}
if (__dirname.startsWith(nodeModules)) {
return true;
}
cwd = path.join(nodeModules, "..", "..");
}
}