2022-09-01 18:39:09 +00:00
|
|
|
// 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 { AsyncLocalStorage } from "async_hooks";
|
2024-03-07 08:52:34 +00:00
|
|
|
import { ICallbackServer } from "./callbacks";
|
2022-09-01 18:39:09 +00:00
|
|
|
import * as config from "./config";
|
2023-12-04 15:22:44 +00:00
|
|
|
import { Stack } from "./stack";
|
|
|
|
|
|
|
|
import * as engrpc from "../proto/engine_grpc_pb";
|
|
|
|
import * as resrpc from "../proto/resource_grpc_pb";
|
2024-08-02 07:37:06 +00:00
|
|
|
import type { ResourceModule, ResourcePackage } from "./rpc";
|
2022-09-01 18:39:09 +00:00
|
|
|
|
|
|
|
const nodeEnvKeys = {
|
|
|
|
project: "PULUMI_NODEJS_PROJECT",
|
|
|
|
stack: "PULUMI_NODEJS_STACK",
|
|
|
|
dryRun: "PULUMI_NODEJS_DRY_RUN",
|
|
|
|
queryMode: "PULUMI_NODEJS_QUERY_MODE",
|
|
|
|
parallel: "PULUMI_NODEJS_PARALLEL",
|
|
|
|
monitorAddr: "PULUMI_NODEJS_MONITOR",
|
|
|
|
engineAddr: "PULUMI_NODEJS_ENGINE",
|
|
|
|
syncDir: "PULUMI_NODEJS_SYNC",
|
2024-07-15 11:27:47 +00:00
|
|
|
|
|
|
|
// Unlike the values above, this value is not set by the CLI and is
|
|
|
|
// controlled via a user-set environment variable.
|
2022-09-01 18:39:09 +00:00
|
|
|
cacheDynamicProviders: "PULUMI_NODEJS_CACHE_DYNAMIC_PROVIDERS",
|
2024-07-15 11:27:47 +00:00
|
|
|
|
2022-09-01 18:39:09 +00:00
|
|
|
organization: "PULUMI_NODEJS_ORGANIZATION",
|
|
|
|
};
|
|
|
|
|
|
|
|
const pulumiEnvKeys = {
|
|
|
|
legacyApply: "PULUMI_ENABLE_LEGACY_APPLY",
|
|
|
|
};
|
|
|
|
|
2024-07-15 11:27:47 +00:00
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
2022-09-01 18:39:09 +00:00
|
|
|
export const asyncLocalStorage = new AsyncLocalStorage<Store>();
|
|
|
|
|
2024-07-15 11:27:47 +00:00
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
2022-09-01 18:39:09 +00:00
|
|
|
export interface WriteableOptions {
|
2024-07-15 11:27:47 +00:00
|
|
|
/**
|
|
|
|
* The name of the current project.
|
|
|
|
*/
|
|
|
|
project?: string;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The name of the current stack being deployed into.
|
|
|
|
*/
|
|
|
|
stack?: string;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The degree of parallelism for resource operations (default is serial).
|
|
|
|
*/
|
|
|
|
parallel?: number;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A connection string to the engine's RPC, in case we need to reestablish.
|
|
|
|
*/
|
|
|
|
engineAddr?: string;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A connection string to the monitor's RPC, in case we need to reestablish.
|
|
|
|
*/
|
|
|
|
monitorAddr?: string;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether we are performing a preview (true) or a real deployment (false).
|
|
|
|
*/
|
|
|
|
dryRun?: boolean;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* True if we're in testing mode (allows execution without the CLI).
|
|
|
|
*/
|
|
|
|
testModeEnabled?: boolean;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* True if we're in query mode (does not allow resource registration).
|
|
|
|
*/
|
|
|
|
queryMode?: boolean;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* True if we will resolve missing outputs to inputs during preview.
|
|
|
|
*/
|
|
|
|
legacyApply?: boolean;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* True if we will cache serialized dynamic providers on the program side.
|
|
|
|
*/
|
|
|
|
cacheDynamicProviders?: boolean;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The name of the current organization (if available).
|
|
|
|
*/
|
|
|
|
organization?: string;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The number of process listeners which can be registered before writing a
|
|
|
|
* warning.
|
|
|
|
*/
|
|
|
|
maximumProcessListeners: number;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A directory containing the send/receive files for making synchronous
|
|
|
|
* invokes to the engine.
|
2022-09-01 18:39:09 +00:00
|
|
|
*/
|
|
|
|
syncDir?: string;
|
|
|
|
}
|
|
|
|
|
2024-07-15 11:27:47 +00:00
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
2022-09-01 18:39:09 +00:00
|
|
|
export interface Store {
|
|
|
|
settings: {
|
|
|
|
options: WriteableOptions;
|
2023-12-19 14:35:23 +00:00
|
|
|
monitor?: resrpc.IResourceMonitorClient;
|
|
|
|
engine?: engrpc.IEngineClient;
|
2022-09-01 18:39:09 +00:00
|
|
|
rpcDone: Promise<any>;
|
2023-12-19 14:35:23 +00:00
|
|
|
// Needed for legacy @pulumi/pulumi packages doing async feature checks.
|
2022-09-01 18:39:09 +00:00
|
|
|
featureSupport: Record<string, boolean>;
|
|
|
|
};
|
|
|
|
config: Record<string, string>;
|
|
|
|
stackResource?: Stack;
|
|
|
|
leakCandidates: Set<Promise<any>>;
|
2023-12-01 17:35:07 +00:00
|
|
|
logErrorCount: number;
|
2023-12-19 14:35:23 +00:00
|
|
|
|
|
|
|
/**
|
2024-07-15 11:27:47 +00:00
|
|
|
* Tells us if the resource monitor we are connected to is able to support
|
|
|
|
* secrets across its RPC interface. When it does, we marshal outputs marked
|
|
|
|
* with the secret bit in a special way.
|
2023-12-19 14:35:23 +00:00
|
|
|
*/
|
|
|
|
supportsSecrets: boolean;
|
|
|
|
|
|
|
|
/**
|
2024-07-15 11:27:47 +00:00
|
|
|
* Tells us if the resource monitor we are connected to is able to support
|
|
|
|
* resource references across its RPC interface. When it does, we marshal
|
|
|
|
* resources in a special way.
|
2023-12-19 14:35:23 +00:00
|
|
|
*/
|
|
|
|
supportsResourceReferences: boolean;
|
|
|
|
|
|
|
|
/**
|
2024-07-15 11:27:47 +00:00
|
|
|
* Tells u if the resource monitor we are connected to is able to support
|
|
|
|
* output values across its RPC interface. When it does, we marshal outputs
|
2023-12-19 14:35:23 +00:00
|
|
|
* in a special way.
|
|
|
|
*/
|
|
|
|
supportsOutputValues: boolean;
|
|
|
|
|
|
|
|
/**
|
2024-07-15 11:27:47 +00:00
|
|
|
* Tells us if the resource monitor we are connected to is able to support
|
|
|
|
* the `deletedWith` resource option across its RPC interface.
|
2023-12-19 14:35:23 +00:00
|
|
|
*/
|
|
|
|
supportsDeletedWith: boolean;
|
|
|
|
|
|
|
|
/**
|
2024-07-15 11:27:47 +00:00
|
|
|
* Tells us if the resource monitor we are connected to is able to support
|
|
|
|
* alias specs across its RPC interface. When it does, we marshal aliases in
|
|
|
|
* a special way.
|
2023-12-19 14:35:23 +00:00
|
|
|
*/
|
|
|
|
supportsAliasSpecs: boolean;
|
2024-03-07 08:52:34 +00:00
|
|
|
|
|
|
|
/**
|
2024-07-15 11:27:47 +00:00
|
|
|
* Tells us if the resource monitor we are connected to is able to support
|
|
|
|
* remote transforms across its RPC interface. When it does, we marshal
|
2024-03-07 08:52:34 +00:00
|
|
|
* transforms to the monitor instead of running them locally.
|
|
|
|
*/
|
|
|
|
supportsTransforms: boolean;
|
|
|
|
|
2024-07-18 08:41:39 +00:00
|
|
|
/**
|
|
|
|
* Tells us if the resource monitor we are connected to is able to support
|
|
|
|
* remote invoke transforms across its RPC interface. When it does, we marshal
|
|
|
|
* transforms to the monitor instead of running them locally.
|
|
|
|
*/
|
|
|
|
supportsInvokeTransforms: boolean;
|
|
|
|
|
2024-03-07 08:52:34 +00:00
|
|
|
/**
|
2024-07-15 11:27:47 +00:00
|
|
|
* The callback service running for this deployment. This registers
|
|
|
|
* callbacks and forwards them to the engine.
|
2024-03-07 08:52:34 +00:00
|
|
|
*/
|
|
|
|
callbacks?: ICallbackServer;
|
2024-08-02 07:37:06 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Tracks the list of resource packages that have been registered.
|
|
|
|
*/
|
|
|
|
resourcePackages: Map<string, ResourcePackage[]>;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Tracks the list of resource modules that have been registered.
|
|
|
|
*/
|
|
|
|
resourceModules: Map<string, ResourceModule[]>;
|
2022-09-01 18:39:09 +00:00
|
|
|
}
|
|
|
|
|
2024-07-15 11:27:47 +00:00
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
2022-09-01 18:39:09 +00:00
|
|
|
export class LocalStore implements Store {
|
|
|
|
settings = {
|
|
|
|
options: {
|
|
|
|
organization: process.env[nodeEnvKeys.organization],
|
|
|
|
project: process.env[nodeEnvKeys.project] || "project",
|
|
|
|
stack: process.env[nodeEnvKeys.stack] || "stack",
|
|
|
|
dryRun: process.env[nodeEnvKeys.dryRun] === "true",
|
|
|
|
queryMode: process.env[nodeEnvKeys.queryMode] === "true",
|
|
|
|
monitorAddr: process.env[nodeEnvKeys.monitorAddr],
|
|
|
|
engineAddr: process.env[nodeEnvKeys.engineAddr],
|
|
|
|
syncDir: process.env[nodeEnvKeys.syncDir],
|
|
|
|
cacheDynamicProviders: process.env[nodeEnvKeys.cacheDynamicProviders] !== "false",
|
|
|
|
legacyApply: process.env[pulumiEnvKeys.legacyApply] === "true",
|
|
|
|
maximumProcessListeners: 30,
|
|
|
|
},
|
|
|
|
rpcDone: Promise.resolve(),
|
|
|
|
featureSupport: {},
|
|
|
|
};
|
|
|
|
config = {
|
|
|
|
[config.configEnvKey]: process.env[config.configEnvKey] || "",
|
|
|
|
[config.configSecretKeysEnvKey]: process.env[config.configSecretKeysEnvKey] || "",
|
|
|
|
};
|
|
|
|
stackResource = undefined;
|
|
|
|
|
|
|
|
/**
|
2024-07-15 11:27:47 +00:00
|
|
|
* Tracks the list of potential leak candidates.
|
2023-04-28 22:27:10 +00:00
|
|
|
*/
|
2022-09-01 18:39:09 +00:00
|
|
|
leakCandidates = new Set<Promise<any>>();
|
2023-12-01 17:35:07 +00:00
|
|
|
|
|
|
|
logErrorCount = 0;
|
2023-12-19 14:35:23 +00:00
|
|
|
|
|
|
|
supportsSecrets = false;
|
|
|
|
supportsResourceReferences = false;
|
|
|
|
supportsOutputValues = false;
|
|
|
|
supportsDeletedWith = false;
|
|
|
|
supportsAliasSpecs = false;
|
2024-03-07 08:52:34 +00:00
|
|
|
supportsTransforms = false;
|
2024-07-18 08:41:39 +00:00
|
|
|
supportsInvokeTransforms = false;
|
2024-08-02 07:37:06 +00:00
|
|
|
resourcePackages = new Map<string, ResourcePackage[]>();
|
|
|
|
resourceModules = new Map<string, ResourceModule[]>();
|
2022-09-01 18:39:09 +00:00
|
|
|
}
|
|
|
|
|
2024-07-15 11:27:47 +00:00
|
|
|
/**
|
|
|
|
* Get the root stack resource for the current stack deployment.
|
|
|
|
*
|
2022-09-01 18:39:09 +00:00
|
|
|
* @internal
|
|
|
|
*/
|
|
|
|
export function getStackResource(): Stack | undefined {
|
|
|
|
const { stackResource } = getStore();
|
|
|
|
return stackResource;
|
|
|
|
}
|
|
|
|
|
2024-08-02 07:37:06 +00:00
|
|
|
/**
|
|
|
|
* Get the resource package map for the current stack deployment.
|
|
|
|
*
|
|
|
|
* @internal
|
|
|
|
*/
|
|
|
|
export function getResourcePackages(): Map<string, ResourcePackage[]> {
|
|
|
|
const store = getGlobalStore();
|
|
|
|
if (store.resourcePackages === undefined) {
|
|
|
|
// resourcePackages can be undefined if an older SDK where it was not defined is created it.
|
|
|
|
// In this case, we should initialize it to an empty map.
|
|
|
|
store.resourcePackages = new Map<string, ResourcePackage[]>();
|
|
|
|
}
|
|
|
|
return store.resourcePackages;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the resource module map for the current stack deployment.
|
|
|
|
*
|
|
|
|
* @internal
|
|
|
|
*/
|
|
|
|
export function getResourceModules(): Map<string, ResourceModule[]> {
|
|
|
|
const store = getGlobalStore();
|
|
|
|
if (store.resourceModules === undefined) {
|
|
|
|
// resourceModules can be undefined if an older SDK where it was not defined is created it.
|
|
|
|
// In this case, we should initialize it to an empty map.
|
|
|
|
store.resourceModules = new Map<string, ResourceModule[]>();
|
|
|
|
}
|
|
|
|
return store.resourceModules;
|
|
|
|
}
|
|
|
|
|
2024-07-15 11:27:47 +00:00
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
2022-09-01 18:39:09 +00:00
|
|
|
export function setStackResource(newStackResource?: Stack) {
|
|
|
|
const localStore = getStore();
|
|
|
|
globalThis.stackResource = newStackResource;
|
|
|
|
localStore.stackResource = newStackResource;
|
2023-04-28 22:27:10 +00:00
|
|
|
}
|
2022-09-01 18:39:09 +00:00
|
|
|
|
|
|
|
declare global {
|
|
|
|
/* eslint-disable no-var */
|
|
|
|
var globalStore: Store;
|
|
|
|
var stackResource: Stack | undefined;
|
|
|
|
}
|
|
|
|
|
2024-07-15 11:27:47 +00:00
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
2022-09-01 18:39:09 +00:00
|
|
|
export function getLocalStore(): Store | undefined {
|
|
|
|
return asyncLocalStorage.getStore();
|
2023-04-28 22:27:10 +00:00
|
|
|
}
|
2022-09-01 18:39:09 +00:00
|
|
|
|
|
|
|
(<any>getLocalStore).captureReplacement = () => {
|
|
|
|
const returnFunc = () => {
|
|
|
|
if (global.globalStore === undefined) {
|
|
|
|
global.globalStore = new LocalStore();
|
|
|
|
}
|
|
|
|
return global.globalStore;
|
|
|
|
};
|
|
|
|
return returnFunc;
|
|
|
|
};
|
|
|
|
|
2024-07-15 11:27:47 +00:00
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
2022-09-01 18:39:09 +00:00
|
|
|
export const getStore = () => {
|
|
|
|
const localStore = getLocalStore();
|
|
|
|
if (localStore === undefined) {
|
|
|
|
if (global.globalStore === undefined) {
|
|
|
|
global.globalStore = new LocalStore();
|
|
|
|
}
|
|
|
|
return global.globalStore;
|
|
|
|
}
|
|
|
|
return localStore;
|
|
|
|
};
|
|
|
|
|
|
|
|
(<any>getStore).captureReplacement = () => {
|
|
|
|
const returnFunc = () => {
|
|
|
|
if (global.globalStore === undefined) {
|
|
|
|
global.globalStore = new LocalStore();
|
|
|
|
}
|
|
|
|
return global.globalStore;
|
|
|
|
};
|
|
|
|
return returnFunc;
|
|
|
|
};
|
2024-08-02 07:37:06 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
|
|
|
export const getGlobalStore = () => {
|
|
|
|
if (global.globalStore === undefined) {
|
|
|
|
global.globalStore = new LocalStore();
|
|
|
|
}
|
|
|
|
return global.globalStore;
|
|
|
|
};
|