From 67d76fae49524e337f08dbbdcd6cf6b03189706c Mon Sep 17 00:00:00 2001
From: Paul Gschwendtner <paulgschwendtner@gmail.com>
Date: Fri, 15 Oct 2021 19:23:43 +0200
Subject: [PATCH] feat(bazel): add support for custom environment variables in
 integration tests

Adds support for custom environment variables in integration tests. The
environment variable dictionary can use location expansion. Also there
is a special placeholder called `<TMP>` that can be used to acquire a
temporary directory. This is useful when setting up tools like BAZELISK,
which requires a `HOME` environment variable as an example.
---
 bazel/integration/index.bzl                   | 38 ++++++++-
 bazel/integration/test_runner/bazel.ts        | 12 +++
 bazel/integration/test_runner/constants.ts    | 14 ++++
 bazel/integration/test_runner/main.ts         |  8 +-
 bazel/integration/test_runner/runner.ts       | 78 +++++++++++++------
 .../tests/custom_env_variables/BUILD.bazel    | 22 ++++++
 .../tests/custom_env_variables/test.js        | 39 ++++++++++
 7 files changed, 185 insertions(+), 26 deletions(-)
 create mode 100644 bazel/integration/test_runner/constants.ts
 create mode 100644 bazel/integration/tests/custom_env_variables/BUILD.bazel
 create mode 100644 bazel/integration/tests/custom_env_variables/test.js

diff --git a/bazel/integration/index.bzl b/bazel/integration/index.bzl
index 9910be90a..582083fa6 100644
--- a/bazel/integration/index.bzl
+++ b/bazel/integration/index.bzl
@@ -6,10 +6,30 @@ def _serialize_file(file):
 
     return struct(path = file.path, shortPath = file.short_path)
 
+def _serialize_and_expand_location(ctx, value):
+    """Expands Bazel make location expressions for the given value. Returns a JSON
+      serializable dictionary matching the `BazelExpandedValue` type in the test runner."""
+    new_value = ctx.expand_location(value, targets = ctx.attr.data)
+
+    return {
+        "value": new_value,
+        "containsExpandedValue": new_value != value,
+    }
+
 def _split_and_expand_command(ctx, command):
     """Splits a command into the binary and its arguments. Also Bazel locations are expanded."""
-    expanded_command = ctx.expand_location(command, targets = ctx.attr.data)
-    return expanded_command.split(" ", 1)
+    return [_serialize_and_expand_location(ctx, v) for v in command.split(" ", 1)]
+
+def _serialize_and_expand_environment(ctx, environment_dict):
+    """Converts the given environment dictionary into a JSON-serializable dictionary
+      that will work with the test runner."""
+    result = {}
+
+    for variable_name in environment_dict:
+        value = environment_dict[variable_name]
+        result[variable_name] = _serialize_and_expand_location(ctx, value)
+
+    return result
 
 def _unwrap_label_keyed_mappings(dict, description):
     """Unwraps a label-keyed dictionary used for expressing mappings into a JSON-serializable
@@ -51,6 +71,7 @@ def _integration_test_config_impl(ctx):
         testPackage = ctx.label.package,
         testFiles = [_serialize_file(f) for f in ctx.files.srcs],
         commands = [_split_and_expand_command(ctx, c) for c in ctx.attr.commands],
+        environment = _serialize_and_expand_environment(ctx, ctx.attr.environment),
         npmPackageMappings = npmPackageMappings,
         toolMappings = toolMappings,
     )
@@ -105,6 +126,17 @@ _integration_test_config = rule(
               This allows for binaries like `node` to be made available to the integration test
               using the `PATH` environment variable.""",
         ),
+        "environment": attr.string_dict(
+            doc = """
+              Dictionary of environment variables and their values. This allows for custom
+              environment variables to be set when integration commands are invoked.
+
+              The environment variable values can use Bazel make location expansion similar
+              to the `commands` attribute. Additionally, values of `<TMP>` are replaced with
+              a unique temporary directory. This can be useful when providing `HOME` for
+              bazelisk or puppeteer as as an example.
+            """,
+        ),
     },
 )
 
@@ -114,6 +146,7 @@ def integration_test(
         commands,
         npm_packages = {},
         tool_mappings = {},
+        environment = {},
         data = [],
         tags = [],
         **kwargs):
@@ -129,6 +162,7 @@ def integration_test(
         commands = commands,
         npm_packages = npm_packages,
         tool_mappings = tool_mappings,
+        environment = environment,
         tags = tags,
     )
 
diff --git a/bazel/integration/test_runner/bazel.ts b/bazel/integration/test_runner/bazel.ts
index 0673754da..e83c92336 100644
--- a/bazel/integration/test_runner/bazel.ts
+++ b/bazel/integration/test_runner/bazel.ts
@@ -23,6 +23,18 @@ export interface BazelFileInfo {
   shortPath: string;
 }
 
+/**
+ * Interface describing a Bazel-expanded value. A integration command for example could
+ * use a Bazel location expansion to resolve a binary. Such resolved values are captured in
+ * a structure like this.
+ */
+export interface BazelExpandedValue {
+  /** Actual value, with expanded Make expressions if it contained any. */
+  value: string;
+  /** Whether the value contains an expanded value. */
+  containsExpandedValue: boolean;
+}
+
 /** Resolves the specified Bazel file to an absolute disk path. */
 export function resolveBazelFile(file: BazelFileInfo): string {
   return runfiles.resolveWorkspaceRelative(file.shortPath);
diff --git a/bazel/integration/test_runner/constants.ts b/bazel/integration/test_runner/constants.ts
new file mode 100644
index 000000000..01ad6c2e5
--- /dev/null
+++ b/bazel/integration/test_runner/constants.ts
@@ -0,0 +1,14 @@
+/**
+ * @license
+ * Copyright Google LLC All Rights Reserved.
+ *
+ * Use of this source code is governed by an MIT-style license that can be
+ * found in the LICENSE file at https://angular.io/license
+ */
+
+/**
+ * Placeholder that can be used in the test environment dictionary to reserve a
+ * temporary directory. A temporary directory can be useful when running tools
+ * like Bazelisk which need a `HOME` directory for example.
+ */
+export const ENVIRONMENT_TMP_PLACEHOLDER = '<TMP>';
diff --git a/bazel/integration/test_runner/main.ts b/bazel/integration/test_runner/main.ts
index e7d042e7d..354a3ccd5 100644
--- a/bazel/integration/test_runner/main.ts
+++ b/bazel/integration/test_runner/main.ts
@@ -7,7 +7,7 @@
  */
 
 import {TestRunner} from './runner';
-import {BazelFileInfo, runfiles} from './bazel';
+import {BazelExpandedValue, BazelFileInfo, runfiles} from './bazel';
 import * as fs from 'fs';
 import {debug} from './debug';
 
@@ -20,7 +20,8 @@ interface TestConfig {
   testFiles: BazelFileInfo[];
   npmPackageMappings: Record<string, BazelFileInfo>;
   toolMappings: Record<string, BazelFileInfo>;
-  commands: [[binary: string, ...args: string[]]];
+  commands: [[binary: BazelExpandedValue, ...args: BazelExpandedValue[]]];
+  environment: Record<string, BazelExpandedValue>;
 }
 
 /** Main command line entry-point for the integration test runner. */
@@ -32,12 +33,15 @@ async function main(): Promise<void> {
   const configContent = await fs.promises.readFile(configPath, 'utf8');
   const config = JSON.parse(configContent) as TestConfig;
 
+  debug('Fetched test config:', config);
+
   const runner = new TestRunner(
     config.testFiles,
     config.testPackage,
     config.toolMappings,
     config.npmPackageMappings,
     config.commands,
+    config.environment,
   );
 
   await runner.run();
diff --git a/bazel/integration/test_runner/runner.ts b/bazel/integration/test_runner/runner.ts
index ed082a50a..63a0adc77 100644
--- a/bazel/integration/test_runner/runner.ts
+++ b/bazel/integration/test_runner/runner.ts
@@ -15,7 +15,12 @@ import {
   readPackageJsonContents,
   updateMappingsForPackageJson,
 } from './package_json';
-import {BazelFileInfo, resolveBazelFile, resolveBinaryWithRunfiles} from './bazel';
+import {
+  BazelExpandedValue,
+  BazelFileInfo,
+  resolveBazelFile,
+  resolveBinaryWithRunfiles,
+} from './bazel';
 import {debug} from './debug';
 import {
   expandEnvironmentVariableSubstitutions,
@@ -23,6 +28,7 @@ import {
   prependToPathVariable,
   runCommandInChildProcess,
 } from './process_utils';
+import {ENVIRONMENT_TMP_PLACEHOLDER} from './constants';
 
 /**
  * Test runner that takes a set of files within a Bazel package and copies the files
@@ -38,18 +44,20 @@ export class TestRunner {
     private readonly testPackage: string,
     private readonly toolMappings: Record<string, BazelFileInfo>,
     private readonly npmPackageMappings: Record<string, BazelFileInfo>,
-    private readonly commands: [[binary: string, ...args: string[]]],
+    private readonly commands: [[binary: BazelExpandedValue, ...args: BazelExpandedValue[]]],
+    private readonly environment: Record<string, BazelExpandedValue>,
   ) {}
 
   async run() {
-    const tmpDir = await this._getTmpDirectoryPath();
-    const toolMappings = await this._setupToolMappingsForTest(tmpDir);
+    const testDir = await this._getTestTmpDirectoryPath();
+    const toolMappings = await this._setupToolMappingsForTest(testDir);
+    const testEnv = await this._createTestProcessEnvironment(testDir, toolMappings.binDir);
 
-    console.info(`Running test in: ${path.normalize(tmpDir)}`);
+    console.info(`Running test in: ${path.normalize(testDir)}`);
 
-    await this._copyTestFilesToDirectory(tmpDir);
-    await this._patchPackageJsonIfNeeded(tmpDir);
-    await this._runTestCommands(tmpDir, toolMappings.binDir);
+    await this._copyTestFilesToDirectory(testDir);
+    await this._patchPackageJsonIfNeeded(testDir);
+    await this._runTestCommands(testDir, testEnv);
   }
 
   /**
@@ -59,7 +67,7 @@ export class TestRunner {
    * In case this test does not run as part of `bazel test`, a system-temporary directory
    * is being created, although not being cleaned up to allow for debugging.
    */
-  private async _getTmpDirectoryPath(): Promise<string> {
+  private async _getTestTmpDirectoryPath(): Promise<string> {
     // Bazel provides a temporary test directory itself when it executes a test. We prefer
     // this when the integration test runs with `bazel test`. In other cases we want to
     // provide a temporary directory that can be used for manually jumping into the
@@ -167,24 +175,50 @@ export class TestRunner {
   }
 
   /**
-   * Runs the test commands sequentially in the test directory. An additional directory
-   * that is added to the command process `$PATH` environment variables can be specified.
+   * Creates the test process environment. The specified tools bin directory
+   * will be added to the environment `PATH` variable.
    *
-   * @throws An error if any of the configured commands did not complete successfully.
+   * User-specified environment variables can use Bazel location expansion as well
+   * as a special placeholder for acquiring a temporary directory for the test.
    */
-  private async _runTestCommands(
+  private async _createTestProcessEnvironment(
     testDir: string,
-    additionalPathDirectory: string | null,
-  ): Promise<void> {
-    const commandPath =
-      additionalPathDirectory === null
-        ? process.env.PATH
-        : prependToPathVariable(additionalPathDirectory, process.env.PATH ?? '');
-    const commandEnv = {...process.env, PATH: commandPath};
+    toolsBinDir: string,
+  ): Promise<NodeJS.ProcessEnv> {
+    const testEnv: NodeJS.ProcessEnv = {...process.env};
+    let i = 0;
+
+    for (let [variableName, value] of Object.entries(this.environment)) {
+      let envValue: string = value.value;
+
+      if (value.containsExpandedValue) {
+        envValue = await resolveBinaryWithRunfiles(envValue);
+      } else if (envValue === ENVIRONMENT_TMP_PLACEHOLDER) {
+        envValue = path.join(testDir, `.tmp-env-${i++}`);
+        await fs.promises.mkdir(envValue);
+      }
 
+      testEnv[variableName] = envValue;
+    }
+
+    const commandPath = prependToPathVariable(toolsBinDir, testEnv.PATH ?? '');
+    return {...testEnv, PATH: commandPath};
+  }
+
+  /**
+   * Runs the test commands sequentially in the test directory with the given test
+   * environment applied to the child process executing the command.
+   *
+   * @throws An error if any of the configured commands did not complete successfully.
+   */
+  private async _runTestCommands(testDir: string, commandEnv: NodeJS.ProcessEnv): Promise<void> {
     for (const [binary, ...args] of this.commands) {
-      const resolvedBinary = await resolveBinaryWithRunfiles(binary);
-      const evaluatedArgs = expandEnvironmentVariableSubstitutions(args);
+      // Only resolve the binary if it contains an expanded value. In other cases we would
+      // not want to resolve through runfiles to avoid accidentally unexpected resolution.
+      const resolvedBinary = binary.containsExpandedValue
+        ? await resolveBinaryWithRunfiles(binary.value)
+        : binary.value;
+      const evaluatedArgs = expandEnvironmentVariableSubstitutions(args.map((v) => v.value));
       const success = await runCommandInChildProcess(
         resolvedBinary,
         evaluatedArgs,
diff --git a/bazel/integration/tests/custom_env_variables/BUILD.bazel b/bazel/integration/tests/custom_env_variables/BUILD.bazel
new file mode 100644
index 000000000..ddce66bcb
--- /dev/null
+++ b/bazel/integration/tests/custom_env_variables/BUILD.bazel
@@ -0,0 +1,22 @@
+load("//bazel/integration:index.bzl", "integration_test")
+
+integration_test(
+    name = "test",
+    srcs = [
+        "test.js",
+    ],
+    commands = [
+        "node ./test.js",
+    ],
+    data = ["@npm//:node_modules/semver/package.json"],
+    environment = {
+        "CUSTOM_VAR": "yes!",
+        "RESOLVED_BIN": "$(rootpath @npm//:node_modules/semver/package.json)",
+        "MANIFEST_PATH": "external/npm/node_modules/semver/package.json",
+        "BAZELISK_HOME": "<TMP>",
+        "BAZELISK_HOME_2": "<TMP>",
+    },
+    tool_mappings = {
+        "@nodejs//:node_bin": "node",
+    },
+)
diff --git a/bazel/integration/tests/custom_env_variables/test.js b/bazel/integration/tests/custom_env_variables/test.js
new file mode 100644
index 000000000..7df2d600d
--- /dev/null
+++ b/bazel/integration/tests/custom_env_variables/test.js
@@ -0,0 +1,39 @@
+const fs = require('fs');
+
+if (process.env.CUSTOM_VAR !== 'yes!') {
+  console.error('The expected `CUSTOM_VAR` environment variable is not set.');
+  process.exit(1);
+}
+
+if (process.env.RESOLVED_BIN === undefined) {
+  console.error('The expected `RESOLVED_BIN` variable is not set.');
+  process.exit(1);
+}
+
+if (require(process.env.RESOLVED_BIN).name !== 'semver') {
+  console.error('The `RESOLVED_BIN` file did not resolve to the "package.json" of "semver".');
+  process.exit(1);
+}
+
+if (process.env.MANIFEST_PATH !== 'external/npm/node_modules/semver/package.json') {
+  console.error('Expected `MANIFEST_PATH` to remain untouched as it has not not been expanded.');
+  process.exit(1);
+}
+
+const bazeliskHome = process.env.BAZELISK_HOME;
+const bazeliskHome_2 = process.env.BAZELISK_HOME_2;
+
+if (!fs.statSync(bazeliskHome).isDirectory()) {
+  console.error('Expected `BAZELISK_HOME` environment variable to point to a temp directory.');
+  process.exit(1);
+}
+
+if (!fs.statSync(bazeliskHome_2).isDirectory()) {
+  console.error('Expected `BAZELISK_HOME_2` environment variable to point to a temp directory.');
+  process.exit(1);
+}
+
+if (bazeliskHome === bazeliskHome_2) {
+  console.error('Expected the bazelisk home variables to point to different temp directories.');
+  process.exit(1);
+}