pulumi/sdk/nodejs/automation/stack.ts

1003 lines
33 KiB
TypeScript
Raw Permalink Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

// Copyright 2016-2022, Pulumi Corporation.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
import * as fs from "fs";
import * as os from "os";
import * as pathlib from "path";
import * as readline from "readline";
import * as upath from "upath";
import * as grpc from "@grpc/grpc-js";
import TailFile from "@logdna/tail-file";
import * as log from "../log";
import { CommandResult, runPulumiCmd } from "./cmd";
import { ConfigMap, ConfigValue } from "./config";
import { StackNotFoundError } from "./errors";
import { EngineEvent, SummaryEvent } from "./events";
import { LanguageServer, maxRPCMessageSize } from "./server";
import { Deployment, PulumiFn, Workspace } from "./workspace";
import { LocalWorkspace } from "./localWorkspace";
import { TagMap } from "./tag";
const langrpc = require("../proto/language_grpc_pb.js");
interface ReadlineResult {
tail: TailFile;
rl: readline.Interface;
}
/**
* Stack is an isolated, independently configurable instance of a Pulumi program.
* Stack exposes methods for the full pulumi lifecycle (up/preview/refresh/destroy), as well as managing configuration.
* Multiple Stacks are commonly used to denote different phases of development
* (such as development, staging and production) or feature branches (such as feature-x-dev, jane-feature-x-dev).
*
* @alpha
*/
export class Stack {
/**
* The name identifying the Stack.
*/
readonly name: string;
/**
* The Workspace the Stack was created from.
*/
readonly workspace: Workspace;
private ready: Promise<any>;
/**
* Creates a new stack using the given workspace, and stack name.
* It fails if a stack with that name already exists
*
* @param name The name identifying the Stack.
* @param workspace The Workspace the Stack was created from.
*/
static async create(name: string, workspace: Workspace): Promise<Stack> {
const stack = new Stack(name, workspace, "create");
await stack.ready;
return stack;
}
/**
* Selects stack using the given workspace, and stack name.
* It returns an error if the given Stack does not exist.
*
* @param name The name identifying the Stack.
* @param workspace The Workspace the Stack was created from.
*/
static async select(name: string, workspace: Workspace): Promise<Stack> {
const stack = new Stack(name, workspace, "select");
await stack.ready;
return stack;
}
/**
* Tries to create a new stack using the given workspace and
* stack name if the stack does not already exist,
* or falls back to selecting the existing stack. If the stack does not exist,
* it will be created and selected.
*
* @param name The name identifying the Stack.
* @param workspace The Workspace the Stack was created from.
*/
static async createOrSelect(name: string, workspace: Workspace): Promise<Stack> {
const stack = new Stack(name, workspace, "createOrSelect");
await stack.ready;
return stack;
}
private constructor(name: string, workspace: Workspace, mode: StackInitMode) {
this.name = name;
this.workspace = workspace;
switch (mode) {
case "create":
this.ready = workspace.createStack(name);
return this;
case "select":
this.ready = workspace.selectStack(name);
return this;
case "createOrSelect":
this.ready = workspace.selectStack(name).catch((err) => {
if (err instanceof StackNotFoundError) {
return workspace.createStack(name);
}
throw err;
});
return this;
default:
throw new Error(`unexpected Stack creation mode: ${mode}`);
}
}
private async readLines(logPath: string, callback: (event: EngineEvent) => void): Promise<ReadlineResult> {
const eventLogTail = new TailFile(logPath, { startPos: 0, pollFileIntervalMs: 200 }).on("tail_error", (err) => {
throw err;
});
await eventLogTail.start();
const lineSplitter = readline.createInterface({ input: eventLogTail });
lineSplitter.on("line", (line) => {
let event: EngineEvent;
try {
event = JSON.parse(line);
callback(event);
} catch (e) {
log.warn(`Failed to parse engine event
If you're seeing this warning, please comment on https://github.com/pulumi/pulumi/issues/6768 with the event and any
details about your environment.
Event: ${line}\n${e.toString()}`);
}
});
return {
tail: eventLogTail,
rl: lineSplitter,
};
}
/**
* Creates or updates the resources in a stack by executing the program in the Workspace.
* https://www.pulumi.com/docs/cli/commands/pulumi_up/
*
* @param opts Options to customize the behavior of the update.
*/
async up(opts?: UpOptions): Promise<UpResult> {
const args = ["up", "--yes", "--skip-preview"];
let kind = execKind.local;
let program = this.workspace.program;
args.push(...this.remoteArgs());
if (opts) {
if (opts.program) {
program = opts.program;
}
if (opts.message) {
args.push("--message", opts.message);
}
if (opts.expectNoChanges) {
args.push("--expect-no-changes");
}
if (opts.refresh) {
args.push("--refresh");
}
if (opts.diff) {
args.push("--diff");
}
if (opts.replace) {
for (const rURN of opts.replace) {
args.push("--replace", rURN);
}
}
if (opts.target) {
for (const tURN of opts.target) {
args.push("--target", tURN);
}
}
if (opts.policyPacks) {
for (const pack of opts.policyPacks) {
args.push("--policy-pack", pack);
}
}
if (opts.policyPackConfigs) {
for (const packConfig of opts.policyPackConfigs) {
args.push("--policy-pack-config", packConfig);
}
}
if (opts.targetDependents) {
args.push("--target-dependents");
}
if (opts.parallel) {
args.push("--parallel", opts.parallel.toString());
}
if (opts.userAgent) {
args.push("--exec-agent", opts.userAgent);
}
if (opts.plan) {
args.push("--plan", opts.plan);
}
applyGlobalOpts(opts, args);
}
let onExit = (hasError: boolean) => {
return;
};
let didError = false;
if (program) {
kind = execKind.inline;
const server = new grpc.Server({
"grpc.max_receive_message_length": maxRPCMessageSize,
});
const languageServer = new LanguageServer(program);
server.addService(langrpc.LanguageRuntimeService, languageServer);
const port: number = await new Promise<number>((resolve, reject) => {
server.bindAsync(`127.0.0.1:0`, grpc.ServerCredentials.createInsecure(), (err, p) => {
if (err) {
reject(err);
} else {
resolve(p);
}
});
});
server.start();
onExit = (hasError: boolean) => {
languageServer.onPulumiExit(hasError);
server.forceShutdown();
};
args.push(`--client=127.0.0.1:${port}`);
}
args.push("--exec-kind", kind);
let logPromise: Promise<ReadlineResult> | undefined;
let logFile: string | undefined;
// Set up event log tailing
if (opts?.onEvent) {
const onEvent = opts.onEvent;
logFile = createLogFile("up");
args.push("--event-log", logFile);
logPromise = this.readLines(logFile, (event) => {
onEvent(event);
});
}
let upResult: CommandResult;
try {
upResult = await this.runPulumiCmd(args, opts?.onOutput);
} catch (e) {
didError = true;
throw e;
} finally {
onExit(didError);
await cleanUp(logFile, await logPromise);
}
// TODO: do this in parallel after this is fixed https://github.com/pulumi/pulumi/issues/6050
const outputs = await this.outputs();
// If it's a remote workspace, explicitly set showSecrets to false to prevent attempting to
// load the project file.
const summary = await this.info(!this.isRemote && opts?.showSecrets);
return {
stdout: upResult.stdout,
stderr: upResult.stderr,
summary: summary!,
outputs: outputs!,
};
}
/**
* Performs a dry-run update to a stack, returning pending changes.
* https://www.pulumi.com/docs/cli/commands/pulumi_preview/
*
* @param opts Options to customize the behavior of the preview.
*/
async preview(opts?: PreviewOptions): Promise<PreviewResult> {
const args = ["preview"];
let kind = execKind.local;
let program = this.workspace.program;
args.push(...this.remoteArgs());
if (opts) {
if (opts.program) {
program = opts.program;
}
if (opts.message) {
args.push("--message", opts.message);
}
if (opts.expectNoChanges) {
args.push("--expect-no-changes");
}
if (opts.refresh) {
args.push("--refresh");
}
if (opts.diff) {
args.push("--diff");
}
if (opts.replace) {
for (const rURN of opts.replace) {
args.push("--replace", rURN);
}
}
if (opts.target) {
for (const tURN of opts.target) {
args.push("--target", tURN);
}
}
if (opts.policyPacks) {
for (const pack of opts.policyPacks) {
args.push("--policy-pack", pack);
}
}
if (opts.policyPackConfigs) {
for (const packConfig of opts.policyPackConfigs) {
args.push("--policy-pack-config", packConfig);
}
}
if (opts.targetDependents) {
args.push("--target-dependents");
}
if (opts.parallel) {
args.push("--parallel", opts.parallel.toString());
}
if (opts.userAgent) {
args.push("--exec-agent", opts.userAgent);
}
if (opts.plan) {
args.push("--save-plan", opts.plan);
}
applyGlobalOpts(opts, args);
}
let onExit = (hasError: boolean) => {
return;
};
let didError = false;
if (program) {
kind = execKind.inline;
const server = new grpc.Server({
"grpc.max_receive_message_length": maxRPCMessageSize,
});
const languageServer = new LanguageServer(program);
server.addService(langrpc.LanguageRuntimeService, languageServer);
const port: number = await new Promise<number>((resolve, reject) => {
server.bindAsync(`127.0.0.1:0`, grpc.ServerCredentials.createInsecure(), (err, p) => {
if (err) {
reject(err);
} else {
resolve(p);
}
});
});
server.start();
onExit = (hasError: boolean) => {
languageServer.onPulumiExit(hasError);
server.forceShutdown();
};
args.push(`--client=127.0.0.1:${port}`);
}
args.push("--exec-kind", kind);
// Set up event log tailing
const logFile = createLogFile("preview");
args.push("--event-log", logFile);
let summaryEvent: SummaryEvent | undefined;
const logPromise = this.readLines(logFile, (event) => {
if (event.summaryEvent) {
summaryEvent = event.summaryEvent;
}
if (opts?.onEvent) {
const onEvent = opts.onEvent;
onEvent(event);
}
});
let previewResult: CommandResult;
try {
previewResult = await this.runPulumiCmd(args, opts?.onOutput);
} catch (e) {
didError = true;
throw e;
} finally {
onExit(didError);
await cleanUp(logFile, await logPromise);
}
if (!summaryEvent) {
log.warn(
"Failed to parse summary event, but preview succeeded. PreviewResult `changeSummary` will be empty.",
);
}
return {
stdout: previewResult.stdout,
stderr: previewResult.stderr,
changeSummary: summaryEvent?.resourceChanges || {},
};
}
/**
* Compares the current stacks resource state with the state known to exist in the actual
* cloud provider. Any such changes are adopted into the current stack.
*
* @param opts Options to customize the behavior of the refresh.
*/
async refresh(opts?: RefreshOptions): Promise<RefreshResult> {
const args = ["refresh", "--yes", "--skip-preview"];
args.push(...this.remoteArgs());
if (opts) {
if (opts.message) {
args.push("--message", opts.message);
}
if (opts.expectNoChanges) {
args.push("--expect-no-changes");
}
if (opts.target) {
for (const tURN of opts.target) {
args.push("--target", tURN);
}
}
if (opts.parallel) {
args.push("--parallel", opts.parallel.toString());
}
if (opts.userAgent) {
args.push("--exec-agent", opts.userAgent);
}
applyGlobalOpts(opts, args);
}
let logPromise: Promise<ReadlineResult> | undefined;
let logFile: string | undefined;
// Set up event log tailing
if (opts?.onEvent) {
const onEvent = opts.onEvent;
logFile = createLogFile("refresh");
args.push("--event-log", logFile);
logPromise = this.readLines(logFile, (event) => {
onEvent(event);
});
}
const kind = this.workspace.program ? execKind.inline : execKind.local;
args.push("--exec-kind", kind);
const refPromise = this.runPulumiCmd(args, opts?.onOutput);
const [refResult, logResult] = await Promise.all([refPromise, logPromise]);
await cleanUp(logFile, logResult);
// If it's a remote workspace, explicitly set showSecrets to false to prevent attempting to
// load the project file.
const summary = await this.info(!this.isRemote && opts?.showSecrets);
return {
stdout: refResult.stdout,
stderr: refResult.stderr,
summary: summary!,
};
}
/**
* Destroy deletes all resources in a stack, leaving all history and configuration intact.
*
* @param opts Options to customize the behavior of the destroy.
*/
async destroy(opts?: DestroyOptions): Promise<DestroyResult> {
const args = ["destroy", "--yes", "--skip-preview"];
args.push(...this.remoteArgs());
if (opts) {
if (opts.message) {
args.push("--message", opts.message);
}
if (opts.target) {
for (const tURN of opts.target) {
args.push("--target", tURN);
}
}
if (opts.targetDependents) {
args.push("--target-dependents");
}
if (opts.excludeProtected) {
args.push("--exclude-protected");
}
if (opts.parallel) {
args.push("--parallel", opts.parallel.toString());
}
if (opts.userAgent) {
args.push("--exec-agent", opts.userAgent);
}
applyGlobalOpts(opts, args);
}
let logPromise: Promise<ReadlineResult> | undefined;
let logFile: string | undefined;
// Set up event log tailing
if (opts?.onEvent) {
const onEvent = opts.onEvent;
logFile = createLogFile("destroy");
args.push("--event-log", logFile);
logPromise = this.readLines(logFile, (event) => {
onEvent(event);
});
}
const kind = this.workspace.program ? execKind.inline : execKind.local;
args.push("--exec-kind", kind);
const desPromise = this.runPulumiCmd(args, opts?.onOutput);
const [desResult, logResult] = await Promise.all([desPromise, logPromise]);
await cleanUp(logFile, logResult);
// If it's a remote workspace, explicitly set showSecrets to false to prevent attempting to
// load the project file.
const summary = await this.info(!this.isRemote && opts?.showSecrets);
return {
stdout: desResult.stdout,
stderr: desResult.stderr,
summary: summary!,
};
}
/**
* Returns the config value associated with the specified key.
*
* @param key The key to use for the config lookup
* @param path The key contains a path to a property in a map or list to get
*/
async getConfig(key: string, path?: boolean): Promise<ConfigValue> {
return this.workspace.getConfig(this.name, key, path);
}
/**
* Returns the full config map associated with the stack in the Workspace.
*/
async getAllConfig(): Promise<ConfigMap> {
return this.workspace.getAllConfig(this.name);
}
/**
* Sets a config key-value pair on the Stack in the associated Workspace.
*
* @param key The key to set.
* @param value The config value to set.
* @param path The key contains a path to a property in a map or list to set.
*/
async setConfig(key: string, value: ConfigValue, path?: boolean): Promise<void> {
return this.workspace.setConfig(this.name, key, value, path);
}
/**
* Sets all specified config values on the stack in the associated Workspace.
*
* @param config The map of config key-value pairs to set.
* @param path The keys contain a path to a property in a map or list to set.
*/
async setAllConfig(config: ConfigMap, path?: boolean): Promise<void> {
return this.workspace.setAllConfig(this.name, config, path);
}
/**
* Removes the specified config key from the Stack in the associated Workspace.
*
* @param key The config key to remove.
* @param path The key contains a path to a property in a map or list to remove.
*/
async removeConfig(key: string, path?: boolean): Promise<void> {
return this.workspace.removeConfig(this.name, key, path);
}
/**
* Removes the specified config keys from the Stack in the associated Workspace.
*
* @param keys The config keys to remove.
* @param path The keys contain a path to a property in a map or list to remove.
*/
async removeAllConfig(keys: string[], path?: boolean): Promise<void> {
return this.workspace.removeAllConfig(this.name, keys, path);
}
/**
* Gets and sets the config map used with the last update.
*/
async refreshConfig(): Promise<ConfigMap> {
return this.workspace.refreshConfig(this.name);
}
/**
* Returns the tag value associated with specified key.
*
* @param key The key to use for the tag lookup.
*/
async getTag(key: string): Promise<string> {
return this.workspace.getTag(this.name, key);
}
/**
* Sets a tag key-value pair on the Stack in the associated Workspace.
*
* @param key The tag key to set.
* @param value The tag value to set.
*/
async setTag(key: string, value: string): Promise<void> {
await this.workspace.setTag(this.name, key, value);
}
/**
* Removes the specified tag key-value pair from the Stack in the associated Workspace.
*
* @param key The tag key to remove.
*/
async removeTag(key: string): Promise<void> {
await this.workspace.removeTag(this.name, key);
}
/**
* Returns the full tag map associated with the stack in the Workspace.
*/
async listTags(): Promise<TagMap> {
return this.workspace.listTags(this.name);
}
/**
* Gets the current set of Stack outputs from the last Stack.up().
*/
async outputs(): Promise<OutputMap> {
return this.workspace.stackOutputs(this.name);
}
/**
* Returns a list summarizing all previous and current results from Stack lifecycle operations
* (up/preview/refresh/destroy).
*/
async history(pageSize?: number, page?: number, showSecrets?: boolean): Promise<UpdateSummary[]> {
const args = ["stack", "history", "--json"];
if (showSecrets ?? true) {
args.push("--show-secrets");
}
if (pageSize) {
if (!page || page < 1) {
page = 1;
}
args.push("--page-size", Math.floor(pageSize).toString(), "--page", Math.floor(page).toString());
}
const result = await this.runPulumiCmd(args);
return JSON.parse(result.stdout, (key, value) => {
if (key === "startTime" || key === "endTime") {
return new Date(value);
}
return value;
});
}
async info(showSecrets?: boolean): Promise<UpdateSummary | undefined> {
const history = await this.history(1 /*pageSize*/, undefined, showSecrets);
if (!history || history.length === 0) {
return undefined;
}
return history[0];
}
/**
* Cancel stops a stack's currently running update. It returns an error if no update is currently running.
* Note that this operation is _very dangerous_, and may leave the stack in an inconsistent state
* if a resource operation was pending when the update was canceled.
* This command is not supported for local backends.
*/
async cancel(): Promise<void> {
await this.runPulumiCmd(["cancel", "--yes"]);
}
/**
* exportStack exports the deployment state of the stack.
* This can be combined with Stack.importStack to edit a stack's state (such as recovery from failed deployments).
*/
async exportStack(): Promise<Deployment> {
return this.workspace.exportStack(this.name);
}
/**
* importStack imports the specified deployment state into a pre-existing stack.
* This can be combined with Stack.exportStack to edit a stack's state (such as recovery from failed deployments).
*
* @param state the stack state to import.
*/
async importStack(state: Deployment): Promise<void> {
return this.workspace.importStack(this.name, state);
}
private async runPulumiCmd(args: string[], onOutput?: (out: string) => void): Promise<CommandResult> {
let envs: { [key: string]: string } = {
PULUMI_DEBUG_COMMANDS: "true",
};
if (this.isRemote) {
envs["PULUMI_EXPERIMENTAL"] = "true";
}
const pulumiHome = this.workspace.pulumiHome;
if (pulumiHome) {
envs["PULUMI_HOME"] = pulumiHome;
}
envs = { ...envs, ...this.workspace.envVars };
const additionalArgs = await this.workspace.serializeArgsForOp(this.name);
args = [...args, "--stack", this.name, ...additionalArgs];
const result = await runPulumiCmd(args, this.workspace.workDir, envs, onOutput);
await this.workspace.postCommandCallback(this.name);
return result;
}
private get isRemote(): boolean {
const ws = this.workspace;
return ws instanceof LocalWorkspace ? ws.isRemote : false;
}
private remoteArgs(): string[] {
const ws = this.workspace;
return ws instanceof LocalWorkspace ? ws.remoteArgs() : [];
}
}
function applyGlobalOpts(opts: GlobalOpts, args: string[]) {
if (opts.color) {
args.push("--color", opts.color);
}
if (opts.logFlow) {
args.push("--logflow");
}
if (opts.logVerbosity) {
args.push("--verbose", opts.logVerbosity.toString());
}
if (opts.logToStdErr) {
args.push("--logtostderr");
}
if (opts.tracing) {
args.push("--tracing", opts.tracing);
}
if (opts.debug) {
args.push("--debug");
}
}
/**
* Returns a stack name formatted with the greatest possible specificity:
* org/project/stack or user/project/stack
* Using this format avoids ambiguity in stack identity guards creating or selecting the wrong stack.
* Note that filestate backends (local file, S3, Azure Blob) do not support stack names in this
* format, and instead only use the stack name without an org/user or project to qualify it.
* See: https://github.com/pulumi/pulumi/issues/2522
*
* @param org The org (or user) that contains the Stack.
* @param project The project that parents the Stack.
* @param stack The name of the Stack.
*/
export function fullyQualifiedStackName(org: string, project: string, stack: string): string {
return `${org}/${project}/${stack}`;
}
export interface OutputValue {
value: any;
secret: boolean;
}
export type OutputMap = { [key: string]: OutputValue };
export interface UpdateSummary {
// pre-update info
kind: UpdateKind;
startTime: Date;
message: string;
environment: { [key: string]: string };
config: ConfigMap;
// post-update info
result: UpdateResult;
endTime: Date;
version: number;
Deployment?: RawJSON;
resourceChanges?: OpMap;
}
/**
* The kind of update that was performed on the stack.
*/
export type UpdateKind = "update" | "preview" | "refresh" | "rename" | "destroy" | "import";
/**
* Represents the current status of a given update.
*/
export type UpdateResult = "not-started" | "in-progress" | "succeeded" | "failed";
/**
* The granular CRUD operation performed on a particular resource during an update.
*/
export type OpType =
| "same"
| "create"
| "update"
| "delete"
| "replace"
| "create-replacement"
| "delete-replaced"
| "read"
| "read-replacement"
| "refresh"
| "discard"
| "discard-replaced"
| "remove-pending-replace"
| "import"
| "import-replacement";
/**
* A map of operation types and their corresponding counts.
*/
export type OpMap = {
[key in OpType]?: number;
};
/**
* An unstructured JSON string used for back-compat with versioned APIs (such as Deployment).
*/
export type RawJSON = string;
/**
* The deployment output from running a Pulumi program update.
*/
export interface UpResult {
stdout: string;
stderr: string;
outputs: OutputMap;
summary: UpdateSummary;
}
/**
* Output from running a Pulumi program preview.
*/
export interface PreviewResult {
stdout: string;
stderr: string;
changeSummary: OpMap;
}
/**
* Output from refreshing the resources in a given Stack.
*/
export interface RefreshResult {
stdout: string;
stderr: string;
summary: UpdateSummary;
}
/**
* Output from destroying all resources in a Stack.
*/
export interface DestroyResult {
stdout: string;
stderr: string;
summary: UpdateSummary;
}
export interface GlobalOpts {
/** Colorize output. */
color?: "always" | "never" | "raw" | "auto";
/** Flow log settings to child processes (like plugins) */
logFlow?: boolean;
/** Enable verbose logging (e.g., v=3); anything >3 is very verbose */
logVerbosity?: number;
/** Log to stderr instead of to files */
logToStdErr?: boolean;
/** Emit tracing to the specified endpoint. Use the file: scheme to write tracing data to a local file */
tracing?: string;
/** Print detailed debugging output during resource operations */
debug?: boolean;
}
/**
* Options controlling the behavior of a Stack.up() operation.
*/
export interface UpOptions extends GlobalOpts {
parallel?: number;
message?: string;
expectNoChanges?: boolean;
/**
* Refresh the state of the stack's resources before this update.
*/
refresh?: boolean;
diff?: boolean;
replace?: string[];
policyPacks?: string[];
policyPackConfigs?: string[];
target?: string[];
targetDependents?: boolean;
userAgent?: string;
onOutput?: (out: string) => void;
onEvent?: (event: EngineEvent) => void;
program?: PulumiFn;
/**
* Plan specifies the path to an update plan to use for the update.
*/
plan?: string;
/**
* Include secrets in the UpSummary.
*/
showSecrets?: boolean;
}
/**
* Options controlling the behavior of a Stack.preview() operation.
*/
export interface PreviewOptions extends GlobalOpts {
parallel?: number;
message?: string;
expectNoChanges?: boolean;
/**
* Refresh the state of the stack's resources against the cloud provider before running preview.
*/
refresh?: boolean;
diff?: boolean;
replace?: string[];
policyPacks?: string[];
policyPackConfigs?: string[];
target?: string[];
targetDependents?: boolean;
userAgent?: string;
program?: PulumiFn;
onOutput?: (out: string) => void;
onEvent?: (event: EngineEvent) => void;
color?: "always" | "never" | "raw" | "auto";
/**
* Plan specifies the path where the update plan should be saved.
*/
plan?: string;
}
/**
* Options controlling the behavior of a Stack.refresh() operation.
*/
export interface RefreshOptions extends GlobalOpts {
parallel?: number;
message?: string;
expectNoChanges?: boolean;
target?: string[];
userAgent?: string;
onOutput?: (out: string) => void;
onEvent?: (event: EngineEvent) => void;
color?: "always" | "never" | "raw" | "auto";
// Include secrets in the RefreshSummary
showSecrets?: boolean;
}
/**
* Options controlling the behavior of a Stack.destroy() operation.
*/
export interface DestroyOptions extends GlobalOpts {
parallel?: number;
message?: string;
target?: string[];
targetDependents?: boolean;
userAgent?: string;
onOutput?: (out: string) => void;
onEvent?: (event: EngineEvent) => void;
color?: "always" | "never" | "raw" | "auto";
// Include secrets in the DestroySummary
showSecrets?: boolean;
/**
* Do not destroy protected resources.
*/
excludeProtected?: boolean;
}
const execKind = {
local: "auto.local",
inline: "auto.inline",
};
type StackInitMode = "create" | "select" | "createOrSelect";
const createLogFile = (command: string) => {
const logDir = fs.mkdtempSync(upath.joinSafe(os.tmpdir(), `automation-logs-${command}-`));
const logFile = upath.joinSafe(logDir, "eventlog.txt");
// just open/close the file to make sure it exists when we start polling.
fs.closeSync(fs.openSync(logFile, "w"));
return logFile;
};
const cleanUp = async (logFile?: string, rl?: ReadlineResult) => {
if (rl) {
// stop tailing
await rl.tail.quit();
// close the readline interface
rl.rl.close();
}
if (logFile) {
// remove the logfile
if (fs.rm) {
// remove with Node JS 15.X+
fs.rm(pathlib.dirname(logFile), { recursive: true }, () => {
return;
});
} else {
// remove with Node JS 14.X
fs.rmdir(pathlib.dirname(logFile), { recursive: true }, () => {
return;
});
}
}
};