aws-cdk
Version:
AWS CDK CLI, the command line tool for CDK apps
133 lines • 18.7 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
exports.execProgram = execProgram;
exports.createAssembly = createAssembly;
const path = require("path");
const util_1 = require("util");
const cloud_assembly_api_1 = require("@aws-cdk/cloud-assembly-api");
const cxschema = require("@aws-cdk/cloud-assembly-schema");
const cxapi = require("@aws-cdk/cx-api");
const toolkit_lib_1 = require("@aws-cdk/toolkit-lib");
const fs = require("fs-extra");
const api_1 = require("../api");
const user_configuration_1 = require("../cli/user-configuration");
const version_1 = require("../cli/version");
/** Invokes the cloud executable and returns JSON output */
async function execProgram(aws, ioHelper, config) {
const debugFn = (msg) => ioHelper.defaults.debug(msg);
let errorFile;
const params = (0, api_1.synthParametersFromSettings)(config.settings);
const context = {
...config.context.all,
...params.context,
};
await debugFn((0, util_1.format)('context:', context));
const env = noUndefined({
// Versioning, outdir, default account and region
...await (0, api_1.prepareDefaultEnvironment)(aws, debugFn),
// Environment variables derived from settings
...params.env,
});
const build = config.settings.get(['build']);
if (build) {
await exec(build);
}
let app = config.settings.get(['app']);
if (!app) {
throw new toolkit_lib_1.ToolkitError('AppRequired', `--app is required either in command-line, in ${user_configuration_1.PROJECT_CONFIG} or in ${user_configuration_1.USER_DEFAULTS}`);
}
// bypass "synth" if app points to a cloud assembly
if (await fs.pathExists(app) && (await fs.stat(app)).isDirectory()) {
await debugFn('--app points to a cloud assembly, so we bypass synth');
// Acquire a read lock on this directory
const lock = await new api_1.RWLock(app).acquireRead();
return { assembly: createAssembly(app), lock };
}
// Traditionally it has been possible, though not widely advertised, to put a string[] into `cdk.json`.
// However, we would just quickly join this array back up to string with spaces (unquoted even!) and proceed as usual,
// thereby losing all the benefits of a pre-segmented command line. This coercion is just here for backwards
// compatibility with existing configurations. An upcoming PR might retain the benefit of the string[].
if (Array.isArray(app)) {
app = app.join(' ');
}
const commandLine = await (0, api_1.guessExecutable)(app, debugFn);
const outdir = config.settings.get(['output']);
if (!outdir) {
throw new toolkit_lib_1.ToolkitError('OutputRequired', 'unexpected: --output is required');
}
if (typeof outdir !== 'string') {
throw new toolkit_lib_1.ToolkitError('OutputNotString', `--output takes a string, got ${JSON.stringify(outdir)}`);
}
try {
await fs.mkdirp(outdir);
}
catch (error) {
throw new toolkit_lib_1.ToolkitError('OutputDirCreateFailed', `Could not create output directory ${outdir} (${error.message})`);
}
await debugFn(`outdir: ${outdir}`);
env[cxapi.OUTDIR_ENV] = outdir;
// Send version information
env[cxapi.CLI_ASM_VERSION_ENV] = cxschema.Manifest.version();
env[cxapi.CLI_VERSION_ENV] = (0, version_1.versionNumber)();
// Acquire a lock on the output directory
const writerLock = await new api_1.RWLock(outdir).acquireWrite();
// Prepare an errorFile location
errorFile = path.join(outdir, 'error.txt');
await fs.promises.rm(errorFile, { force: true });
env.CDK_ERROR_FILE = errorFile;
await debugFn((0, util_1.format)('env:', env));
const cleanupTemp = (0, api_1.writeContextToEnv)(env, context, 'add-process-env-later');
try {
await exec(commandLine);
const assembly = createAssembly(outdir);
return { assembly, lock: await writerLock.convertToReaderLock() };
}
catch (e) {
await writerLock.release();
throw e;
}
finally {
await cleanupTemp();
}
async function exec(commandAndArgs) {
try {
return await (0, api_1.execInChildProcess)(commandAndArgs, {
env: {
...process.env,
...env,
},
errorCodeFile: errorFile,
// Not capturing the stdout/stderr of the CDK app. It must remain attached to a terminal
// if the parent process is attached to a terminal.
captureOutput: false,
});
}
catch (e) {
await debugFn(`failed command: ${commandAndArgs}`);
throw e;
}
}
}
/**
* Creates an assembly with error handling
*/
function createAssembly(appDir) {
try {
return new cloud_assembly_api_1.CloudAssembly(appDir, {
// We sort as we deploy
topoSort: false,
});
}
catch (error) {
if (error.message.includes(cxschema.VERSION_MISMATCH)) {
// this means the CLI version is too old.
// we instruct the user to upgrade.
throw new toolkit_lib_1.ToolkitError('CliVersionMismatch', `This CDK CLI is not compatible with the CDK library used by your application. Please upgrade the CLI to the latest version.\n(${error.message})`);
}
throw error;
}
}
function noUndefined(xs) {
return Object.fromEntries(Object.entries(xs).filter(([_, v]) => v !== undefined));
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"exec.js","sourceRoot":"","sources":["exec.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport { format } from 'util';\nimport { CloudAssembly } from '@aws-cdk/cloud-assembly-api';\nimport * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport { ToolkitError } from '@aws-cdk/toolkit-lib';\nimport * as fs from 'fs-extra';\nimport type { IoHelper } from '../../lib/api-private';\nimport type { SdkProvider, IReadLock } from '../api';\nimport { RWLock, guessExecutable, prepareDefaultEnvironment, writeContextToEnv, synthParametersFromSettings, execInChildProcess } from '../api';\nimport type { Configuration } from '../cli/user-configuration';\nimport { PROJECT_CONFIG, USER_DEFAULTS } from '../cli/user-configuration';\nimport { versionNumber } from '../cli/version';\n\nexport interface ExecProgramResult {\n  readonly assembly: CloudAssembly;\n  readonly lock: IReadLock;\n}\n\n/** Invokes the cloud executable and returns JSON output */\nexport async function execProgram(aws: SdkProvider, ioHelper: IoHelper, config: Configuration): Promise<ExecProgramResult> {\n  const debugFn = (msg: string) => ioHelper.defaults.debug(msg);\n  let errorFile: string | undefined;\n\n  const params = synthParametersFromSettings(config.settings);\n\n  const context = {\n    ...config.context.all,\n    ...params.context,\n  };\n  await debugFn(format('context:', context));\n\n  const env: Record<string, string> = noUndefined({\n    // Versioning, outdir, default account and region\n    ...await prepareDefaultEnvironment(aws, debugFn),\n    // Environment variables derived from settings\n    ...params.env,\n  });\n\n  const build = config.settings.get(['build']);\n  if (build) {\n    await exec(build);\n  }\n\n  let app = config.settings.get(['app']);\n  if (!app) {\n    throw new ToolkitError('AppRequired', `--app is required either in command-line, in ${PROJECT_CONFIG} or in ${USER_DEFAULTS}`);\n  }\n\n  // bypass \"synth\" if app points to a cloud assembly\n  if (await fs.pathExists(app) && (await fs.stat(app)).isDirectory()) {\n    await debugFn('--app points to a cloud assembly, so we bypass synth');\n\n    // Acquire a read lock on this directory\n    const lock = await new RWLock(app).acquireRead();\n\n    return { assembly: createAssembly(app), lock };\n  }\n\n  // Traditionally it has been possible, though not widely advertised, to put a string[] into `cdk.json`.\n  // However, we would just quickly join this array back up to string with spaces (unquoted even!) and proceed as usual,\n  // thereby losing all the benefits of a pre-segmented command line. This coercion is just here for backwards\n  // compatibility with existing configurations. An upcoming PR might retain the benefit of the string[].\n  if (Array.isArray(app)) {\n    app = app.join(' ');\n  }\n  const commandLine = await guessExecutable(app, debugFn);\n\n  const outdir = config.settings.get(['output']);\n  if (!outdir) {\n    throw new ToolkitError('OutputRequired', 'unexpected: --output is required');\n  }\n  if (typeof outdir !== 'string') {\n    throw new ToolkitError('OutputNotString', `--output takes a string, got ${JSON.stringify(outdir)}`);\n  }\n  try {\n    await fs.mkdirp(outdir);\n  } catch (error: any) {\n    throw new ToolkitError('OutputDirCreateFailed', `Could not create output directory ${outdir} (${error.message})`);\n  }\n\n  await debugFn(`outdir: ${outdir}`);\n\n  env[cxapi.OUTDIR_ENV] = outdir;\n\n  // Send version information\n  env[cxapi.CLI_ASM_VERSION_ENV] = cxschema.Manifest.version();\n  env[cxapi.CLI_VERSION_ENV] = versionNumber();\n\n  // Acquire a lock on the output directory\n  const writerLock = await new RWLock(outdir).acquireWrite();\n\n  // Prepare an errorFile location\n  errorFile = path.join(outdir, 'error.txt');\n  await fs.promises.rm(errorFile, { force: true });\n  env.CDK_ERROR_FILE = errorFile;\n\n  await debugFn(format('env:', env));\n\n  const cleanupTemp = writeContextToEnv(env, context, 'add-process-env-later');\n  try {\n    await exec(commandLine);\n\n    const assembly = createAssembly(outdir);\n\n    return { assembly, lock: await writerLock.convertToReaderLock() };\n  } catch (e) {\n    await writerLock.release();\n    throw e;\n  } finally {\n    await cleanupTemp();\n  }\n\n  async function exec(commandAndArgs: string) {\n    try {\n      return await execInChildProcess(commandAndArgs, {\n        env: {\n          ...process.env,\n          ...env,\n        },\n        errorCodeFile: errorFile,\n\n        // Not capturing the stdout/stderr of the CDK app. It must remain attached to a terminal\n        // if the parent process is attached to a terminal.\n        captureOutput: false,\n      });\n    } catch (e: any) {\n      await debugFn(`failed command: ${commandAndArgs}`);\n      throw e;\n    }\n  }\n}\n\n/**\n * Creates an assembly with error handling\n */\nexport function createAssembly(appDir: string) {\n  try {\n    return new CloudAssembly(appDir, {\n      // We sort as we deploy\n      topoSort: false,\n    });\n  } catch (error: any) {\n    if (error.message.includes(cxschema.VERSION_MISMATCH)) {\n      // this means the CLI version is too old.\n      // we instruct the user to upgrade.\n      throw new ToolkitError('CliVersionMismatch', `This CDK CLI is not compatible with the CDK library used by your application. Please upgrade the CLI to the latest version.\\n(${error.message})`);\n    }\n    throw error;\n  }\n}\n\nfunction noUndefined<A>(xs: Record<string, A>): Record<string, NonNullable<A>> {\n  return Object.fromEntries(Object.entries(xs).filter(([_, v]) => v !== undefined)) as any;\n}\n"]}