aws-cdk
Version:
AWS CDK CLI, the command line tool for CDK apps
334 lines • 41.8 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
exports.Configuration = exports.Command = exports.USER_DEFAULTS = exports.PROJECT_CONTEXT = exports.PROJECT_CONFIG = void 0;
exports.commandLineArgumentsToSettings = commandLineArgumentsToSettings;
const os = require("os");
const fs_path = require("path");
const toolkit_lib_1 = require("@aws-cdk/toolkit-lib");
const fs = require("fs-extra");
const context_1 = require("../api/context");
const settings_1 = require("../api/settings");
exports.PROJECT_CONFIG = 'cdk.json';
var context_2 = require("../api/context");
Object.defineProperty(exports, "PROJECT_CONTEXT", { enumerable: true, get: function () { return context_2.PROJECT_CONTEXT; } });
exports.USER_DEFAULTS = '~/.cdk.json';
const CONTEXT_KEY = 'context';
var Command;
(function (Command) {
Command["LS"] = "ls";
Command["LIST"] = "list";
Command["DIFF"] = "diff";
Command["BOOTSTRAP"] = "bootstrap";
Command["DEPLOY"] = "deploy";
Command["DESTROY"] = "destroy";
Command["SYNTHESIZE"] = "synthesize";
Command["SYNTH"] = "synth";
Command["METADATA"] = "metadata";
Command["INIT"] = "init";
Command["VERSION"] = "version";
Command["WATCH"] = "watch";
Command["GC"] = "gc";
Command["FLAGS"] = "flags";
Command["ROLLBACK"] = "rollback";
Command["PUBLISH_ASSETS"] = "publish-assets";
Command["IMPORT"] = "import";
Command["ACKNOWLEDGE"] = "acknowledge";
Command["ACK"] = "ack";
Command["NOTICES"] = "notices";
Command["MIGRATE"] = "migrate";
Command["CONTEXT"] = "context";
Command["DOCS"] = "docs";
Command["DOC"] = "doc";
Command["DOCTOR"] = "doctor";
Command["REFACTOR"] = "refactor";
Command["DRIFT"] = "drift";
Command["CLI_TELEMETRY"] = "cli-telemetry";
})(Command || (exports.Command = Command = {}));
const BUNDLING_COMMANDS = [
Command.DEPLOY,
Command.DIFF,
Command.SYNTH,
Command.SYNTHESIZE,
Command.WATCH,
Command.IMPORT,
Command.PUBLISH_ASSETS,
];
/**
* All sources of settings combined
*/
class Configuration {
/**
* Creates user configuration from commandLineArguments
*/
static async fromArgs(ioHelper, args) {
const commandLineArguments = args
? await commandLineArgumentsToSettings(ioHelper, args)
: new settings_1.Settings();
return new Configuration(ioHelper, commandLineArguments);
}
/**
* Creates user configuration from commandLineArguments and loads
*/
static async fromArgsAndFiles(ioHelper, props = {}) {
const configuration = await Configuration.fromArgs(ioHelper, props.commandLineArguments);
return configuration.loadConfigFiles(props.readUserContext ?? true);
}
constructor(ioHelper, commandLineArguments) {
this.settings = new settings_1.Settings();
this.context = new context_1.Context();
this.defaultConfig = new settings_1.Settings({
versionReporting: true,
assetMetadata: true,
pathMetadata: true,
output: 'cdk.out',
});
this.loaded = false;
this.ioHelper = ioHelper;
this.commandLineArguments = commandLineArguments;
this.commandLineContext = this.commandLineArguments
.subSettings([CONTEXT_KEY])
.makeReadOnly();
}
get projectConfig() {
if (!this._projectConfig) {
throw new toolkit_lib_1.ToolkitError('ConfigNotLoaded', '#load has not been called yet!');
}
return this._projectConfig;
}
get projectContext() {
if (!this._projectContext) {
throw new toolkit_lib_1.ToolkitError('ConfigNotLoaded', '#load has not been called yet!');
}
return this._projectContext;
}
/**
* Load all config
*/
async loadConfigFiles(readUserContext) {
const userConfig = await loadAndLog(this.ioHelper, exports.USER_DEFAULTS);
this._projectConfig = await loadAndLog(this.ioHelper, exports.PROJECT_CONFIG);
this._projectContext = await loadAndLog(this.ioHelper, context_1.PROJECT_CONTEXT);
if (userConfig.get(['build'])) {
throw new toolkit_lib_1.ToolkitError('BuildKeyInUserConfig', 'The `build` key cannot be specified in the user config (~/.cdk.json), specify it in the project config (cdk.json) instead');
}
const contextSources = [
{ bag: this.commandLineContext },
{
fileName: exports.PROJECT_CONFIG,
bag: this.projectConfig.subSettings([CONTEXT_KEY]).makeReadOnly(),
},
{ fileName: context_1.PROJECT_CONTEXT, bag: this.projectContext },
];
if (readUserContext) {
contextSources.push({
fileName: exports.USER_DEFAULTS,
bag: userConfig.subSettings([CONTEXT_KEY]).makeReadOnly(),
});
}
this.context = new context_1.Context(...contextSources);
// Build settings from what's left
this.settings = this.defaultConfig
.merge(userConfig)
.merge(this.projectConfig)
.merge(this.commandLineArguments)
.makeReadOnly();
await this.ioHelper.defaults.debug('merged settings:', this.settings.all);
this.loaded = true;
return this;
}
/**
* Save the project context
*/
async saveContext() {
if (!this.loaded) {
return this;
} // Avoid overwriting files with nothing
await this.projectContext.save(context_1.PROJECT_CONTEXT);
return this;
}
}
exports.Configuration = Configuration;
async function loadAndLog(ioHelper, fileName) {
const ret = await settingsFromFile(ioHelper, fileName);
if (!ret.empty) {
await ioHelper.defaults.debug(fileName + ':', JSON.stringify(ret.all, undefined, 2));
}
return ret;
}
async function settingsFromFile(ioHelper, fileName) {
let settings;
const expanded = expandHomeDir(fileName);
if (await fs.pathExists(expanded)) {
const data = await fs.readJson(expanded);
settings = new settings_1.Settings(data);
}
else {
settings = new settings_1.Settings();
}
// See https://github.com/aws/aws-cdk/issues/59
prohibitContextKeys(settings, ['default-account', 'default-region'], fileName);
await warnAboutContextKey(ioHelper, settings, 'aws:', fileName);
return settings;
}
function prohibitContextKeys(settings, keys, fileName) {
const context = settings.get(['context']);
if (!context || typeof context !== 'object') {
return;
}
for (const key of keys) {
if (key in context) {
throw new toolkit_lib_1.ToolkitError('UnsupportedContextKey', `The 'context.${key}' key was found in ${fs_path.resolve(fileName)}, but it is no longer supported. Please remove it.`);
}
}
}
async function warnAboutContextKey(ioHelper, settings, prefix, fileName) {
const context = settings.get(['context']);
if (!context || typeof context !== 'object') {
return;
}
for (const contextKey of Object.keys(context)) {
if (contextKey.startsWith(prefix)) {
await ioHelper.defaults.warning(`A reserved context key ('context.${prefix}') key was found in ${fs_path.resolve(fileName)}, it might cause surprising behavior and should be removed.`);
}
}
}
function expandHomeDir(x) {
if (x.startsWith('~')) {
return fs_path.join(os.homedir(), x.slice(1));
}
return x;
}
/**
* Parse CLI arguments into Settings
*
* CLI arguments in must be accessed in the CLI code via
* `configuration.settings.get(['argName'])` instead of via `args.argName`.
*
* The advantage is that they can be configured via `cdk.json` and
* `$HOME/.cdk.json`. Arguments not listed below and accessed via this object
* can only be specified on the command line.
*
* @param argv - the received CLI arguments.
* @returns a new Settings object.
*/
async function commandLineArgumentsToSettings(ioHelper, argv) {
const context = await parseStringContextListToObject(ioHelper, argv);
const tags = await parseStringTagsListToObject(ioHelper, expectStringList(argv.tags));
// Determine bundling stacks
let bundlingStacks;
if (BUNDLING_COMMANDS.includes(argv._[0])) {
// If we deploy, diff, synth or watch a list of stacks exclusively we skip
// bundling for all other stacks.
bundlingStacks = argv.exclusively ? argv.STACKS ?? ['**'] : ['**'];
}
else {
// Skip bundling for all stacks
bundlingStacks = [];
}
return new settings_1.Settings({
app: argv.app,
browser: argv.browser,
build: argv.build,
caBundlePath: argv.caBundlePath,
context,
debug: argv.debug,
tags,
language: argv.language,
pathMetadata: argv.pathMetadata,
assetMetadata: argv.assetMetadata,
profile: argv.profile,
plugin: argv.plugin,
requireApproval: argv.requireApproval,
toolkitStackName: argv.toolkitStackName,
toolkitBucket: {
bucketName: argv.bootstrapBucketName,
kmsKeyId: argv.bootstrapKmsKeyId,
},
versionReporting: argv.versionReporting,
staging: argv.staging,
output: argv.output,
outputsFile: argv.outputsFile,
progress: argv.progress,
proxy: argv.proxy,
bundlingStacks,
lookups: argv.lookups,
rollback: argv.rollback,
notices: argv.notices,
assetParallelism: argv['asset-parallelism'],
assetBuildConcurrency: argv['asset-build-concurrency'],
assetPrebuild: argv['asset-prebuild'],
ignoreNoStacks: argv['ignore-no-stacks'],
hotswap: {
ecs: {
minimumHealthyPercent: argv.hotswapEcsMinimumHealthyPercent,
maximumHealthyPercent: argv.hotswapEcsMaximumHealthyPercent,
stabilizationTimeoutSeconds: argv.hotswapEcsStabilizationTimeoutSeconds,
},
},
unstable: argv.unstable,
});
}
function expectStringList(x) {
if (x === undefined) {
return undefined;
}
if (!Array.isArray(x)) {
throw new toolkit_lib_1.ToolkitError('ExpectedArray', `Expected array, got '${x}'`);
}
const nonStrings = x.filter((e) => typeof e !== 'string');
if (nonStrings.length > 0) {
throw new toolkit_lib_1.ToolkitError('ExpectedStringList', `Expected list of strings, found ${nonStrings}`);
}
return x;
}
async function parseStringContextListToObject(ioHelper, argv) {
const context = {};
for (const assignment of argv.context || []) {
const parts = assignment.split(/=(.*)/, 2);
if (parts.length === 2) {
await ioHelper.defaults.debug('CLI argument context: %s=%s', parts[0], parts[1]);
if (parts[0].match(/^aws:.+/)) {
throw new toolkit_lib_1.ToolkitError('ReservedContextPrefix', `User-provided context cannot use keys prefixed with 'aws:', but ${parts[0]} was provided.`);
}
context[parts[0]] = parts[1];
}
else {
await ioHelper.defaults.warn('Context argument is not an assignment (key=value): %s', assignment);
}
}
return context;
}
/**
* Parse tags out of arguments
*
* Return undefined if no tags were provided, return an empty array if only empty
* strings were provided
*/
async function parseStringTagsListToObject(ioHelper, argTags) {
if (argTags === undefined) {
return undefined;
}
if (argTags.length === 0) {
return undefined;
}
const nonEmptyTags = argTags.filter((t) => t !== '');
if (nonEmptyTags.length === 0) {
return [];
}
const tags = [];
for (const assignment of nonEmptyTags) {
const parts = assignment.split(/=(.*)/, 2);
if (parts.length === 2) {
await ioHelper.defaults.debug('CLI argument tags: %s=%s', parts[0], parts[1]);
tags.push({
Key: parts[0],
Value: parts[1],
});
}
else {
await ioHelper.defaults.warn('Tags argument is not an assignment (key=value): %s', assignment);
}
}
return tags.length > 0 ? tags : undefined;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"user-configuration.js","sourceRoot":"","sources":["user-configuration.ts"],"names":[],"mappings":"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","sourcesContent":["import * as os from 'os';\nimport * as fs_path from 'path';\nimport { ToolkitError } from '@aws-cdk/toolkit-lib';\nimport * as fs from 'fs-extra';\nimport { Context, PROJECT_CONTEXT } from '../api/context';\nimport { Settings } from '../api/settings';\nimport type { Tag } from '../api/tags';\nimport type { IoHelper } from '../api-private';\n\nexport const PROJECT_CONFIG = 'cdk.json';\nexport { PROJECT_CONTEXT } from '../api/context';\nexport const USER_DEFAULTS = '~/.cdk.json';\nconst CONTEXT_KEY = 'context';\n\nexport enum Command {\n  LS = 'ls',\n  LIST = 'list',\n  DIFF = 'diff',\n  BOOTSTRAP = 'bootstrap',\n  DEPLOY = 'deploy',\n  DESTROY = 'destroy',\n  SYNTHESIZE = 'synthesize',\n  SYNTH = 'synth',\n  METADATA = 'metadata',\n  INIT = 'init',\n  VERSION = 'version',\n  WATCH = 'watch',\n  GC = 'gc',\n  FLAGS = 'flags',\n  ROLLBACK = 'rollback',\n  PUBLISH_ASSETS = 'publish-assets',\n  IMPORT = 'import',\n  ACKNOWLEDGE = 'acknowledge',\n  ACK = 'ack',\n  NOTICES = 'notices',\n  MIGRATE = 'migrate',\n  CONTEXT = 'context',\n  DOCS = 'docs',\n  DOC = 'doc',\n  DOCTOR = 'doctor',\n  REFACTOR = 'refactor',\n  DRIFT = 'drift',\n  CLI_TELEMETRY = 'cli-telemetry',\n}\n\nconst BUNDLING_COMMANDS = [\n  Command.DEPLOY,\n  Command.DIFF,\n  Command.SYNTH,\n  Command.SYNTHESIZE,\n  Command.WATCH,\n  Command.IMPORT,\n  Command.PUBLISH_ASSETS,\n];\n\nexport type Arguments = {\n  readonly _: [Command, ...string[]];\n  readonly exclusively?: boolean;\n  readonly STACKS?: string[];\n  readonly lookups?: boolean;\n  readonly [name: string]: unknown;\n};\n\nexport interface ConfigurationProps {\n  /**\n   * Configuration passed via command line arguments\n   *\n   * @default - Nothing passed\n   */\n  readonly commandLineArguments?: Arguments;\n\n  /**\n   * Whether or not to use context from `.cdk.json` in user home directory\n   *\n   * @default true\n   */\n  readonly readUserContext?: boolean;\n}\n\n/**\n * All sources of settings combined\n */\nexport class Configuration {\n  /**\n   * Creates user configuration from commandLineArguments\n   */\n  public static async fromArgs(ioHelper: IoHelper, args?: Arguments) {\n    const commandLineArguments = args\n      ? await commandLineArgumentsToSettings(ioHelper, args)\n      : new Settings();\n\n    return new Configuration(ioHelper, commandLineArguments);\n  }\n\n  /**\n   * Creates user configuration from commandLineArguments and loads\n   */\n  public static async fromArgsAndFiles(ioHelper: IoHelper, props: ConfigurationProps = {}) {\n    const configuration = await Configuration.fromArgs(ioHelper, props.commandLineArguments);\n    return configuration.loadConfigFiles(props.readUserContext ?? true);\n  }\n\n  public settings = new Settings();\n  public context = new Context();\n\n  public readonly defaultConfig = new Settings({\n    versionReporting: true,\n    assetMetadata: true,\n    pathMetadata: true,\n    output: 'cdk.out',\n  });\n\n  private readonly commandLineArguments: Settings;\n  private readonly commandLineContext: Settings;\n  private _projectConfig?: Settings;\n  private _projectContext?: Settings;\n  private loaded = false;\n\n  private ioHelper: IoHelper;\n\n  private constructor(ioHelper: IoHelper, commandLineArguments: Settings) {\n    this.ioHelper = ioHelper;\n    this.commandLineArguments = commandLineArguments;\n    this.commandLineContext = this.commandLineArguments\n      .subSettings([CONTEXT_KEY])\n      .makeReadOnly();\n  }\n\n  private get projectConfig() {\n    if (!this._projectConfig) {\n      throw new ToolkitError('ConfigNotLoaded', '#load has not been called yet!');\n    }\n    return this._projectConfig;\n  }\n\n  public get projectContext() {\n    if (!this._projectContext) {\n      throw new ToolkitError('ConfigNotLoaded', '#load has not been called yet!');\n    }\n    return this._projectContext;\n  }\n\n  /**\n   * Load all config\n   */\n  private async loadConfigFiles(readUserContext: boolean): Promise<this> {\n    const userConfig = await loadAndLog(this.ioHelper, USER_DEFAULTS);\n    this._projectConfig = await loadAndLog(this.ioHelper, PROJECT_CONFIG);\n    this._projectContext = await loadAndLog(this.ioHelper, PROJECT_CONTEXT);\n\n    if (userConfig.get(['build'])) {\n      throw new ToolkitError(\n        'BuildKeyInUserConfig',\n        'The `build` key cannot be specified in the user config (~/.cdk.json), specify it in the project config (cdk.json) instead',\n      );\n    }\n\n    const contextSources = [\n      { bag: this.commandLineContext },\n      {\n        fileName: PROJECT_CONFIG,\n        bag: this.projectConfig.subSettings([CONTEXT_KEY]).makeReadOnly(),\n      },\n      { fileName: PROJECT_CONTEXT, bag: this.projectContext },\n    ];\n    if (readUserContext) {\n      contextSources.push({\n        fileName: USER_DEFAULTS,\n        bag: userConfig.subSettings([CONTEXT_KEY]).makeReadOnly(),\n      });\n    }\n\n    this.context = new Context(...contextSources);\n\n    // Build settings from what's left\n    this.settings = this.defaultConfig\n      .merge(userConfig)\n      .merge(this.projectConfig)\n      .merge(this.commandLineArguments)\n      .makeReadOnly();\n\n    await this.ioHelper.defaults.debug('merged settings:', this.settings.all);\n\n    this.loaded = true;\n\n    return this;\n  }\n\n  /**\n   * Save the project context\n   */\n  public async saveContext(): Promise<this> {\n    if (!this.loaded) {\n      return this;\n    } // Avoid overwriting files with nothing\n\n    await this.projectContext.save(PROJECT_CONTEXT);\n\n    return this;\n  }\n}\n\nasync function loadAndLog(ioHelper: IoHelper, fileName: string): Promise<Settings> {\n  const ret = await settingsFromFile(ioHelper, fileName);\n  if (!ret.empty) {\n    await ioHelper.defaults.debug(fileName + ':', JSON.stringify(ret.all, undefined, 2));\n  }\n  return ret;\n}\n\nasync function settingsFromFile(ioHelper: IoHelper, fileName: string): Promise<Settings> {\n  let settings;\n  const expanded = expandHomeDir(fileName);\n  if (await fs.pathExists(expanded)) {\n    const data = await fs.readJson(expanded);\n    settings = new Settings(data);\n  } else {\n    settings = new Settings();\n  }\n\n  // See https://github.com/aws/aws-cdk/issues/59\n  prohibitContextKeys(settings, ['default-account', 'default-region'], fileName);\n  await warnAboutContextKey(ioHelper, settings, 'aws:', fileName);\n\n  return settings;\n}\n\nfunction prohibitContextKeys(settings: Settings, keys: string[], fileName: string) {\n  const context = settings.get(['context']);\n  if (!context || typeof context !== 'object') {\n    return;\n  }\n\n  for (const key of keys) {\n    if (key in context) {\n      throw new ToolkitError(\n        'UnsupportedContextKey',\n        `The 'context.${key}' key was found in ${fs_path.resolve(\n          fileName,\n        )}, but it is no longer supported. Please remove it.`,\n      );\n    }\n  }\n}\n\nasync function warnAboutContextKey(ioHelper: IoHelper, settings: Settings, prefix: string, fileName: string) {\n  const context = settings.get(['context']);\n  if (!context || typeof context !== 'object') {\n    return;\n  }\n\n  for (const contextKey of Object.keys(context)) {\n    if (contextKey.startsWith(prefix)) {\n      await ioHelper.defaults.warning(\n        `A reserved context key ('context.${prefix}') key was found in ${fs_path.resolve(\n          fileName,\n        )}, it might cause surprising behavior and should be removed.`,\n      );\n    }\n  }\n}\n\nfunction expandHomeDir(x: string) {\n  if (x.startsWith('~')) {\n    return fs_path.join(os.homedir(), x.slice(1));\n  }\n  return x;\n}\n\n/**\n * Parse CLI arguments into Settings\n *\n * CLI arguments in must be accessed in the CLI code via\n * `configuration.settings.get(['argName'])` instead of via `args.argName`.\n *\n * The advantage is that they can be configured via `cdk.json` and\n * `$HOME/.cdk.json`. Arguments not listed below and accessed via this object\n * can only be specified on the command line.\n *\n * @param argv - the received CLI arguments.\n * @returns a new Settings object.\n */\nexport async function commandLineArgumentsToSettings(ioHelper: IoHelper, argv: Arguments): Promise<Settings> {\n  const context = await parseStringContextListToObject(ioHelper, argv);\n  const tags = await parseStringTagsListToObject(ioHelper, expectStringList(argv.tags));\n\n  // Determine bundling stacks\n  let bundlingStacks: string[];\n  if (BUNDLING_COMMANDS.includes(argv._[0])) {\n    // If we deploy, diff, synth or watch a list of stacks exclusively we skip\n    // bundling for all other stacks.\n    bundlingStacks = argv.exclusively ? argv.STACKS ?? ['**'] : ['**'];\n  } else {\n    // Skip bundling for all stacks\n    bundlingStacks = [];\n  }\n\n  return new Settings({\n    app: argv.app,\n    browser: argv.browser,\n    build: argv.build,\n    caBundlePath: argv.caBundlePath,\n    context,\n    debug: argv.debug,\n    tags,\n    language: argv.language,\n    pathMetadata: argv.pathMetadata,\n    assetMetadata: argv.assetMetadata,\n    profile: argv.profile,\n    plugin: argv.plugin,\n    requireApproval: argv.requireApproval,\n    toolkitStackName: argv.toolkitStackName,\n    toolkitBucket: {\n      bucketName: argv.bootstrapBucketName,\n      kmsKeyId: argv.bootstrapKmsKeyId,\n    },\n    versionReporting: argv.versionReporting,\n    staging: argv.staging,\n    output: argv.output,\n    outputsFile: argv.outputsFile,\n    progress: argv.progress,\n    proxy: argv.proxy,\n    bundlingStacks,\n    lookups: argv.lookups,\n    rollback: argv.rollback,\n    notices: argv.notices,\n    assetParallelism: argv['asset-parallelism'],\n    assetBuildConcurrency: argv['asset-build-concurrency'],\n    assetPrebuild: argv['asset-prebuild'],\n    ignoreNoStacks: argv['ignore-no-stacks'],\n    hotswap: {\n      ecs: {\n        minimumHealthyPercent: argv.hotswapEcsMinimumHealthyPercent,\n        maximumHealthyPercent: argv.hotswapEcsMaximumHealthyPercent,\n        stabilizationTimeoutSeconds: argv.hotswapEcsStabilizationTimeoutSeconds,\n      },\n    },\n    unstable: argv.unstable,\n  });\n}\n\nfunction expectStringList(x: unknown): string[] | undefined {\n  if (x === undefined) {\n    return undefined;\n  }\n  if (!Array.isArray(x)) {\n    throw new ToolkitError('ExpectedArray', `Expected array, got '${x}'`);\n  }\n  const nonStrings = x.filter((e) => typeof e !== 'string');\n  if (nonStrings.length > 0) {\n    throw new ToolkitError('ExpectedStringList', `Expected list of strings, found ${nonStrings}`);\n  }\n  return x;\n}\n\nasync function parseStringContextListToObject(ioHelper: IoHelper, argv: Arguments): Promise<any> {\n  const context: any = {};\n\n  for (const assignment of (argv as any).context || []) {\n    const parts = assignment.split(/=(.*)/, 2);\n    if (parts.length === 2) {\n      await ioHelper.defaults.debug('CLI argument context: %s=%s', parts[0], parts[1]);\n      if (parts[0].match(/^aws:.+/)) {\n        throw new ToolkitError(\n          'ReservedContextPrefix',\n          `User-provided context cannot use keys prefixed with 'aws:', but ${parts[0]} was provided.`,\n        );\n      }\n      context[parts[0]] = parts[1];\n    } else {\n      await ioHelper.defaults.warn(\n        'Context argument is not an assignment (key=value): %s',\n        assignment,\n      );\n    }\n  }\n  return context;\n}\n\n/**\n * Parse tags out of arguments\n *\n * Return undefined if no tags were provided, return an empty array if only empty\n * strings were provided\n */\nasync function parseStringTagsListToObject(\n  ioHelper: IoHelper,\n  argTags: string[] | undefined,\n): Promise<Tag[] | undefined> {\n  if (argTags === undefined) {\n    return undefined;\n  }\n  if (argTags.length === 0) {\n    return undefined;\n  }\n  const nonEmptyTags = argTags.filter((t) => t !== '');\n  if (nonEmptyTags.length === 0) {\n    return [];\n  }\n\n  const tags: Tag[] = [];\n\n  for (const assignment of nonEmptyTags) {\n    const parts = assignment.split(/=(.*)/, 2);\n    if (parts.length === 2) {\n      await ioHelper.defaults.debug('CLI argument tags: %s=%s', parts[0], parts[1]);\n      tags.push({\n        Key: parts[0],\n        Value: parts[1],\n      });\n    } else {\n      await ioHelper.defaults.warn('Tags argument is not an assignment (key=value): %s', assignment);\n    }\n  }\n  return tags.length > 0 ? tags : undefined;\n}\n"]}