UNPKG

aws-cdk

Version:

AWS CDK CLI, the command line tool for CDK apps

133 lines 18.7 kB
"use strict"; 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"]}