aws-cdk
Version:
AWS CDK CLI, the command line tool for CDK apps
156 lines • 22.9 kB
JavaScript
Object.defineProperty(exports, "__esModule", { value: true });
exports.contextHandler = contextHandler;
const toolkit_lib_1 = require("@aws-cdk/toolkit-lib");
const chalk = require("chalk");
const minimatch_1 = require("minimatch");
const display_version_1 = require("../cli/display-version");
const tables_1 = require("../cli/tables");
const user_configuration_1 = require("../cli/user-configuration");
async function contextHandler(options) {
const ioHelper = options.ioHelper;
if (options.clear) {
options.context.clear();
await options.context.save(user_configuration_1.PROJECT_CONTEXT);
await ioHelper.defaults.info('All context values cleared.');
}
else if (options.reset) {
await invalidateContext(ioHelper, options.context, options.reset, options.force ?? false);
await options.context.save(user_configuration_1.PROJECT_CONTEXT);
}
else {
// List -- support '--json' flag
if (options.json) {
/* c8 ignore start */
const contextValues = options.context.all;
await ioHelper.defaults.result(JSON.stringify(contextValues, undefined, 2));
/* c8 ignore stop */
}
else {
await listContext(ioHelper, options.context);
}
}
await (0, display_version_1.displayVersionMessage)(ioHelper);
return 0;
}
async function listContext(ioHelper, context) {
const keys = contextKeys(context);
if (keys.length === 0) {
await ioHelper.defaults.info('This CDK application does not have any saved context values yet.');
await ioHelper.defaults.info('');
await ioHelper.defaults.info('Context will automatically be saved when you synthesize CDK apps');
await ioHelper.defaults.info('that use environment context information like AZ information, VPCs,');
await ioHelper.defaults.info('SSM parameters, and so on.');
return;
}
// Print config by default
const data_out = [[chalk.green('#'), chalk.green('Key'), chalk.green('Value')]];
for (const [i, key] of keys) {
const jsonWithoutNewlines = JSON.stringify(context.all[key], undefined, 2).replace(/\s+/g, ' ');
data_out.push([i, key, jsonWithoutNewlines]);
}
await ioHelper.defaults.info('Context found in %s:', chalk.blue(user_configuration_1.PROJECT_CONFIG));
await ioHelper.defaults.info('');
await ioHelper.defaults.info((0, tables_1.renderTable)(data_out, process.stdout.columns));
// eslint-disable-next-line @stylistic/max-len
await ioHelper.defaults.info(`Run ${chalk.blue('cdk context --reset KEY_OR_NUMBER')} to remove a context key. It will be refreshed on the next CDK synthesis run.`);
}
async function invalidateContext(ioHelper, context, key, force) {
const i = parseInt(key, 10);
if (`${i}` === key) {
// was a number and we fully parsed it.
key = keyByNumber(context, i);
}
// Unset!
if (context.has(key)) {
context.unset(key);
// check if the value was actually unset.
if (!context.has(key)) {
await ioHelper.defaults.info('Context value %s reset. It will be refreshed on next synthesis', chalk.blue(key));
return;
}
// Value must be in readonly bag
await ioHelper.defaults.error('Only context values specified in %s can be reset through the CLI', chalk.blue(user_configuration_1.PROJECT_CONTEXT));
if (!force) {
throw new toolkit_lib_1.ToolkitError(`Cannot reset readonly context value with key: ${key}`);
}
}
// check if value is expression matching keys
const matches = keysByExpression(context, key);
if (matches.length > 0) {
matches.forEach((match) => {
context.unset(match);
});
const { unset, readonly } = getUnsetAndReadonly(context, matches);
// output the reset values
await printUnset(ioHelper, unset);
// warn about values not reset
await printReadonly(ioHelper, readonly);
// throw when none of the matches were reset
if (!force && unset.length === 0) {
throw new toolkit_lib_1.ToolkitError('None of the matched context values could be reset');
}
return;
}
if (!force) {
throw new toolkit_lib_1.ToolkitError(`No context value matching key: ${key}`);
}
}
async function printUnset(ioHelper, unset) {
if (unset.length === 0)
return;
await ioHelper.defaults.info('The following matched context values reset. They will be refreshed on next synthesis');
for (const match of unset) {
await ioHelper.defaults.info(' %s', match);
}
}
async function printReadonly(ioHelper, readonly) {
if (readonly.length === 0)
return;
await ioHelper.defaults.warn('The following matched context values could not be reset through the CLI');
for (const match of readonly) {
await ioHelper.defaults.info(' %s', match);
}
await ioHelper.defaults.info('');
await ioHelper.defaults.info('This usually means they are configured in %s or %s', chalk.blue(user_configuration_1.PROJECT_CONFIG), chalk.blue(user_configuration_1.USER_DEFAULTS));
}
function keysByExpression(context, expression) {
return context.keys.filter(minimatch_1.minimatch.filter(expression));
}
function getUnsetAndReadonly(context, matches) {
return matches.reduce((acc, match) => {
if (context.has(match)) {
acc.readonly.push(match);
}
else {
acc.unset.push(match);
}
return acc;
}, { unset: [], readonly: [] });
}
function keyByNumber(context, n) {
for (const [i, key] of contextKeys(context)) {
if (n === i) {
return key;
}
}
throw new toolkit_lib_1.ToolkitError(`No context key with number: ${n}`);
}
/**
* Return enumerated keys in a definitive order
*/
function contextKeys(context) {
const keys = context.keys;
keys.sort();
return enumerate1(keys);
}
function enumerate1(xs) {
const ret = new Array();
let i = 1;
for (const x of xs) {
ret.push([i, x]);
i += 1;
}
return ret;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"context.js","sourceRoot":"","sources":["context.ts"],"names":[],"mappings":";;AAoDA,wCAwBC;AA5ED,sDAAoD;AACpD,+BAA+B;AAC/B,yCAAsC;AAGtC,4DAA+D;AAC/D,0CAA4C;AAC5C,kEAA2F;AA6CpF,KAAK,UAAU,cAAc,CAAC,OAAuB;IAC1D,MAAM,QAAQ,GAAG,OAAO,CAAC,QAAQ,CAAC;IAElC,IAAI,OAAO,CAAC,KAAK,EAAE,CAAC;QAClB,OAAO,CAAC,OAAO,CAAC,KAAK,EAAE,CAAC;QACxB,MAAM,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,oCAAe,CAAC,CAAC;QAC5C,MAAM,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,6BAA6B,CAAC,CAAC;IAC9D,CAAC;SAAM,IAAI,OAAO,CAAC,KAAK,EAAE,CAAC;QACzB,MAAM,iBAAiB,CAAC,QAAQ,EAAE,OAAO,CAAC,OAAO,EAAE,OAAO,CAAC,KAAK,EAAE,OAAO,CAAC,KAAK,IAAI,KAAK,CAAC,CAAC;QAC1F,MAAM,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,oCAAe,CAAC,CAAC;IAC9C,CAAC;SAAM,CAAC;QACN,gCAAgC;QAChC,IAAI,OAAO,CAAC,IAAI,EAAE,CAAC;YACjB,qBAAqB;YACrB,MAAM,aAAa,GAAG,OAAO,CAAC,OAAO,CAAC,GAAG,CAAC;YAC1C,MAAM,QAAQ,CAAC,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,aAAa,EAAE,SAAS,EAAE,CAAC,CAAC,CAAC,CAAC;YAC5E,oBAAoB;QACtB,CAAC;aAAM,CAAC;YACN,MAAM,WAAW,CAAC,QAAQ,EAAE,OAAO,CAAC,OAAO,CAAC,CAAC;QAC/C,CAAC;IACH,CAAC;IACD,MAAM,IAAA,uCAAqB,EAAC,QAAQ,CAAC,CAAC;IAEtC,OAAO,CAAC,CAAC;AACX,CAAC;AAED,KAAK,UAAU,WAAW,CAAC,QAAkB,EAAE,OAAgB;IAC7D,MAAM,IAAI,GAAG,WAAW,CAAC,OAAO,CAAC,CAAC;IAElC,IAAI,IAAI,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACtB,MAAM,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,kEAAkE,CAAC,CAAC;QACjG,MAAM,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjC,MAAM,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,kEAAkE,CAAC,CAAC;QACjG,MAAM,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,qEAAqE,CAAC,CAAC;QACpG,MAAM,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,4BAA4B,CAAC,CAAC;QAE3D,OAAO;IACT,CAAC;IAED,0BAA0B;IAC1B,MAAM,QAAQ,GAAU,CAAC,CAAC,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE,KAAK,CAAC,KAAK,CAAC,KAAK,CAAC,EAAE,KAAK,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;IACvF,KAAK,MAAM,CAAC,CAAC,EAAE,GAAG,CAAC,IAAI,IAAI,EAAE,CAAC;QAC5B,MAAM,mBAAmB,GAAG,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,SAAS,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;QAChG,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,GAAG,EAAE,mBAAmB,CAAC,CAAC,CAAC;IAC/C,CAAC;IACD,MAAM,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,sBAAsB,EAAE,KAAK,CAAC,IAAI,CAAC,mCAAc,CAAC,CAAC,CAAC;IACjF,MAAM,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IACjC,MAAM,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAA,oBAAW,EAAC,QAAQ,EAAE,OAAO,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC;IAE5E,8CAA8C;IAC9C,MAAM,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,OAAO,KAAK,CAAC,IAAI,CAAC,mCAAmC,CAAC,+EAA+E,CAAC,CAAC;AACtK,CAAC;AAED,KAAK,UAAU,iBAAiB,CAAC,QAAkB,EAAE,OAAgB,EAAE,GAAW,EAAE,KAAc;IAChG,MAAM,CAAC,GAAG,QAAQ,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;IAC5B,IAAI,GAAG,CAAC,EAAE,KAAK,GAAG,EAAE,CAAC;QACnB,uCAAuC;QACvC,GAAG,GAAG,WAAW,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC;IAChC,CAAC;IACD,SAAS;IACT,IAAI,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC;QACrB,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QACnB,yCAAyC;QACzC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC;YACtB,MAAM,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,gEAAgE,EAAE,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC;YAChH,OAAO;QACT,CAAC;QAED,gCAAgC;QAChC,MAAM,QAAQ,CAAC,QAAQ,CAAC,KAAK,CAAC,kEAAkE,EAAE,KAAK,CAAC,IAAI,CAAC,oCAAe,CAAC,CAAC,CAAC;QAC/H,IAAI,CAAC,KAAK,EAAE,CAAC;YACX,MAAM,IAAI,0BAAY,CAAC,iDAAiD,GAAG,EAAE,CAAC,CAAC;QACjF,CAAC;IACH,CAAC;IAED,6CAA6C;IAC7C,MAAM,OAAO,GAAG,gBAAgB,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC;IAE/C,IAAI,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QACvB,OAAO,CAAC,OAAO,CAAC,CAAC,KAAK,EAAE,EAAE;YACxB,OAAO,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;QACvB,CAAC,CAAC,CAAC;QAEH,MAAM,EAAE,KAAK,EAAE,QAAQ,EAAE,GAAG,mBAAmB,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;QAElE,0BAA0B;QAC1B,MAAM,UAAU,CAAC,QAAQ,EAAE,KAAK,CAAC,CAAC;QAElC,8BAA8B;QAC9B,MAAM,aAAa,CAAC,QAAQ,EAAE,QAAQ,CAAC,CAAC;QAExC,4CAA4C;QAC5C,IAAI,CAAC,KAAK,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACjC,MAAM,IAAI,0BAAY,CAAC,mDAAmD,CAAC,CAAC;QAC9E,CAAC;QACD,OAAO;IACT,CAAC;IACD,IAAI,CAAC,KAAK,EAAE,CAAC;QACX,MAAM,IAAI,0BAAY,CAAC,kCAAkC,GAAG,EAAE,CAAC,CAAC;IAClE,CAAC;AACH,CAAC;AAED,KAAK,UAAU,UAAU,CAAC,QAAkB,EAAE,KAAe;IAC3D,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC;QAAE,OAAO;IAC/B,MAAM,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,sFAAsF,CAAC,CAAC;IACrH,KAAK,MAAM,KAAK,IAAI,KAAK,EAAE,CAAC;QAC1B,MAAM,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC;IAC9C,CAAC;AACH,CAAC;AAED,KAAK,UAAU,aAAa,CAAC,QAAkB,EAAE,QAAkB;IACjE,IAAI,QAAQ,CAAC,MAAM,KAAK,CAAC;QAAE,OAAO;IAClC,MAAM,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,yEAAyE,CAAC,CAAC;IACxG,KAAK,MAAM,KAAK,IAAI,QAAQ,EAAE,CAAC;QAC7B,MAAM,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC;IAC9C,CAAC;IACD,MAAM,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IACjC,MAAM,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,oDAAoD,EAAE,KAAK,CAAC,IAAI,CAAC,mCAAc,CAAC,EAAE,KAAK,CAAC,IAAI,CAAC,kCAAa,CAAC,CAAC,CAAC;AAC5I,CAAC;AAED,SAAS,gBAAgB,CAAC,OAAgB,EAAE,UAAkB;IAC5D,OAAO,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,qBAAS,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC,CAAC;AAC3D,CAAC;AAED,SAAS,mBAAmB,CAAC,OAAgB,EAAE,OAAiB;IAC9D,OAAO,OAAO,CAAC,MAAM,CAA0C,CAAC,GAAG,EAAE,KAAK,EAAE,EAAE;QAC5E,IAAI,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC;YACvB,GAAG,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAC3B,CAAC;aAAM,CAAC;YACN,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACxB,CAAC;QACD,OAAO,GAAG,CAAC;IACb,CAAC,EAAE,EAAE,KAAK,EAAE,EAAE,EAAE,QAAQ,EAAE,EAAE,EAAE,CAAC,CAAC;AAClC,CAAC;AAED,SAAS,WAAW,CAAC,OAAgB,EAAE,CAAS;IAC9C,KAAK,MAAM,CAAC,CAAC,EAAE,GAAG,CAAC,IAAI,WAAW,CAAC,OAAO,CAAC,EAAE,CAAC;QAC5C,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC;YACZ,OAAO,GAAG,CAAC;QACb,CAAC;IACH,CAAC;IACD,MAAM,IAAI,0BAAY,CAAC,+BAA+B,CAAC,EAAE,CAAC,CAAC;AAC7D,CAAC;AAED;;GAEG;AACH,SAAS,WAAW,CAAC,OAAgB;IACnC,MAAM,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC;IAC1B,IAAI,CAAC,IAAI,EAAE,CAAC;IACZ,OAAO,UAAU,CAAC,IAAI,CAAC,CAAC;AAC1B,CAAC;AAED,SAAS,UAAU,CAAI,EAAO;IAC5B,MAAM,GAAG,GAAG,IAAI,KAAK,EAAe,CAAC;IACrC,IAAI,CAAC,GAAG,CAAC,CAAC;IACV,KAAK,MAAM,CAAC,IAAI,EAAE,EAAE,CAAC;QACnB,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;QACjB,CAAC,IAAI,CAAC,CAAC;IACT,CAAC;IACD,OAAO,GAAG,CAAC;AACb,CAAC","sourcesContent":["import { ToolkitError } from '@aws-cdk/toolkit-lib';\nimport * as chalk from 'chalk';\nimport { minimatch } from 'minimatch';\nimport type { Context } from '../api/context';\nimport type { IoHelper } from '../api-private';\nimport { displayVersionMessage } from '../cli/display-version';\nimport { renderTable } from '../cli/tables';\nimport { PROJECT_CONFIG, PROJECT_CONTEXT, USER_DEFAULTS } from '../cli/user-configuration';\n\n/**\n * Options for the context command\n */\nexport interface ContextOptions {\n  /**\n   * The context object sourced from all context locations\n   */\n  readonly context: Context;\n\n  /**\n   * The context key (or its index) to reset\n   *\n   * @default undefined\n   */\n  readonly reset?: string;\n\n  /**\n   * Ignore missing key error\n   *\n   * @default false\n   */\n  readonly force?: boolean;\n\n  /**\n   * Clear all context\n   *\n   * @default false\n   */\n  readonly clear?: boolean;\n\n  /**\n   * Use JSON output instead of YAML when templates are printed to STDOUT\n   *\n   * @default false\n   */\n  readonly json?: boolean;\n\n  /**\n   * IoHelper for messaging.\n   */\n  readonly ioHelper: IoHelper;\n}\n\nexport async function contextHandler(options: ContextOptions): Promise<number> {\n  const ioHelper = options.ioHelper;\n\n  if (options.clear) {\n    options.context.clear();\n    await options.context.save(PROJECT_CONTEXT);\n    await ioHelper.defaults.info('All context values cleared.');\n  } else if (options.reset) {\n    await invalidateContext(ioHelper, options.context, options.reset, options.force ?? false);\n    await options.context.save(PROJECT_CONTEXT);\n  } else {\n    // List -- support '--json' flag\n    if (options.json) {\n      /* c8 ignore start */\n      const contextValues = options.context.all;\n      await ioHelper.defaults.result(JSON.stringify(contextValues, undefined, 2));\n      /* c8 ignore stop */\n    } else {\n      await listContext(ioHelper, options.context);\n    }\n  }\n  await displayVersionMessage(ioHelper);\n\n  return 0;\n}\n\nasync function listContext(ioHelper: IoHelper, context: Context) {\n  const keys = contextKeys(context);\n\n  if (keys.length === 0) {\n    await ioHelper.defaults.info('This CDK application does not have any saved context values yet.');\n    await ioHelper.defaults.info('');\n    await ioHelper.defaults.info('Context will automatically be saved when you synthesize CDK apps');\n    await ioHelper.defaults.info('that use environment context information like AZ information, VPCs,');\n    await ioHelper.defaults.info('SSM parameters, and so on.');\n\n    return;\n  }\n\n  // Print config by default\n  const data_out: any[] = [[chalk.green('#'), chalk.green('Key'), chalk.green('Value')]];\n  for (const [i, key] of keys) {\n    const jsonWithoutNewlines = JSON.stringify(context.all[key], undefined, 2).replace(/\\s+/g, ' ');\n    data_out.push([i, key, jsonWithoutNewlines]);\n  }\n  await ioHelper.defaults.info('Context found in %s:', chalk.blue(PROJECT_CONFIG));\n  await ioHelper.defaults.info('');\n  await ioHelper.defaults.info(renderTable(data_out, process.stdout.columns));\n\n  // eslint-disable-next-line @stylistic/max-len\n  await ioHelper.defaults.info(`Run ${chalk.blue('cdk context --reset KEY_OR_NUMBER')} to remove a context key. It will be refreshed on the next CDK synthesis run.`);\n}\n\nasync function invalidateContext(ioHelper: IoHelper, context: Context, key: string, force: boolean) {\n  const i = parseInt(key, 10);\n  if (`${i}` === key) {\n    // was a number and we fully parsed it.\n    key = keyByNumber(context, i);\n  }\n  // Unset!\n  if (context.has(key)) {\n    context.unset(key);\n    // check if the value was actually unset.\n    if (!context.has(key)) {\n      await ioHelper.defaults.info('Context value %s reset. It will be refreshed on next synthesis', chalk.blue(key));\n      return;\n    }\n\n    // Value must be in readonly bag\n    await ioHelper.defaults.error('Only context values specified in %s can be reset through the CLI', chalk.blue(PROJECT_CONTEXT));\n    if (!force) {\n      throw new ToolkitError(`Cannot reset readonly context value with key: ${key}`);\n    }\n  }\n\n  // check if value is expression matching keys\n  const matches = keysByExpression(context, key);\n\n  if (matches.length > 0) {\n    matches.forEach((match) => {\n      context.unset(match);\n    });\n\n    const { unset, readonly } = getUnsetAndReadonly(context, matches);\n\n    // output the reset values\n    await printUnset(ioHelper, unset);\n\n    // warn about values not reset\n    await printReadonly(ioHelper, readonly);\n\n    // throw when none of the matches were reset\n    if (!force && unset.length === 0) {\n      throw new ToolkitError('None of the matched context values could be reset');\n    }\n    return;\n  }\n  if (!force) {\n    throw new ToolkitError(`No context value matching key: ${key}`);\n  }\n}\n\nasync function printUnset(ioHelper: IoHelper, unset: string[]) {\n  if (unset.length === 0) return;\n  await ioHelper.defaults.info('The following matched context values reset. They will be refreshed on next synthesis');\n  for (const match of unset) {\n    await ioHelper.defaults.info('  %s', match);\n  }\n}\n\nasync function printReadonly(ioHelper: IoHelper, readonly: string[]) {\n  if (readonly.length === 0) return;\n  await ioHelper.defaults.warn('The following matched context values could not be reset through the CLI');\n  for (const match of readonly) {\n    await ioHelper.defaults.info('  %s', match);\n  }\n  await ioHelper.defaults.info('');\n  await ioHelper.defaults.info('This usually means they are configured in %s or %s', chalk.blue(PROJECT_CONFIG), chalk.blue(USER_DEFAULTS));\n}\n\nfunction keysByExpression(context: Context, expression: string) {\n  return context.keys.filter(minimatch.filter(expression));\n}\n\nfunction getUnsetAndReadonly(context: Context, matches: string[]) {\n  return matches.reduce<{ unset: string[]; readonly: string[] }>((acc, match) => {\n    if (context.has(match)) {\n      acc.readonly.push(match);\n    } else {\n      acc.unset.push(match);\n    }\n    return acc;\n  }, { unset: [], readonly: [] });\n}\n\nfunction keyByNumber(context: Context, n: number) {\n  for (const [i, key] of contextKeys(context)) {\n    if (n === i) {\n      return key;\n    }\n  }\n  throw new ToolkitError(`No context key with number: ${n}`);\n}\n\n/**\n * Return enumerated keys in a definitive order\n */\nfunction contextKeys(context: Context): [number, string][] {\n  const keys = context.keys;\n  keys.sort();\n  return enumerate1(keys);\n}\n\nfunction enumerate1<T>(xs: T[]): Array<[number, T]> {\n  const ret = new Array<[number, T]>();\n  let i = 1;\n  for (const x of xs) {\n    ret.push([i, x]);\n    i += 1;\n  }\n  return ret;\n}\n"]}
;