UNPKG

aws-cdk

Version:

AWS CDK CLI, the command line tool for CDK apps

152 lines 20.7 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.CredentialPlugins = void 0; const util_1 = require("util"); const provider_caching_1 = require("./provider-caching"); const api_1 = require("../../../../@aws-cdk/tmp-toolkit-helpers/src/api"); const logging_1 = require("../../logging"); const util_2 = require("../../util"); const plugin_1 = require("../plugin/plugin"); /** * Cache for credential providers. * * Given an account and an operating mode (read or write) will return an * appropriate credential provider for credentials for the given account. The * credential provider will be cached so that multiple AWS clients for the same * environment will not make multiple network calls to obtain credentials. * * Will use default credentials if they are for the right account; otherwise, * all loaded credential provider plugins will be tried to obtain credentials * for the given account. */ class CredentialPlugins { constructor(host) { this.cache = {}; this.host = host ?? plugin_1.PluginHost.instance; } async fetchCredentialsFor(awsAccountId, mode) { const key = `${awsAccountId}-${mode}`; if (!(key in this.cache)) { this.cache[key] = await this.lookupCredentials(awsAccountId, mode); } return this.cache[key]; } get availablePluginNames() { return this.host.credentialProviderSources.map((s) => s.name); } async lookupCredentials(awsAccountId, mode) { const triedSources = []; // Otherwise, inspect the various credential sources we have for (const source of this.host.credentialProviderSources) { let available; try { available = await source.isAvailable(); } catch (e) { // This shouldn't happen, but let's guard against it anyway (0, logging_1.warning)(`Uncaught exception in ${source.name}: ${(0, util_2.formatErrorMessage)(e)}`); available = false; } if (!available) { (0, logging_1.debug)('Credentials source %s is not available, ignoring it.', source.name); continue; } triedSources.push(source); let canProvide; try { canProvide = await source.canProvideCredentials(awsAccountId); } catch (e) { // This shouldn't happen, but let's guard against it anyway (0, logging_1.warning)(`Uncaught exception in ${source.name}: ${(0, util_2.formatErrorMessage)(e)}`); canProvide = false; } if (!canProvide) { continue; } (0, logging_1.debug)(`Using ${source.name} credentials for account ${awsAccountId}`); return { credentials: await v3ProviderFromPlugin(() => source.getProvider(awsAccountId, mode, { supportsV3Providers: true, })), pluginName: source.name, }; } return undefined; } } exports.CredentialPlugins = CredentialPlugins; /** * Take a function that calls the plugin, and turn it into an SDKv3-compatible credential provider. * * What we will do is the following: * * - Query the plugin and see what kind of result it gives us. * - If the result is self-refreshing or doesn't need refreshing, we turn it into an SDKv3 provider * and return it directly. * * If the underlying return value is a provider, we will make it a caching provider * (because we can't know if it will cache by itself or not). * * If the underlying return value is a static credential, caching isn't relevant. * * If the underlying return value is V2 credentials, those have caching built-in. * - If the result is a static credential that expires, we will wrap it in an SDKv3 provider * that will query the plugin again when the credential expires. */ async function v3ProviderFromPlugin(producer) { const initial = await producer(); if (isV3Provider(initial)) { // Already a provider, make caching return (0, provider_caching_1.makeCachingProvider)(initial); } else if (isV3Credentials(initial) && initial.expiration === undefined) { // Static credentials that don't need refreshing nor caching return () => Promise.resolve(initial); } else if (isV3Credentials(initial) && initial.expiration !== undefined) { // Static credentials that do need refreshing and caching return refreshFromPluginProvider(initial, producer); } else if (isV2Credentials(initial)) { // V2 credentials that refresh and cache themselves return v3ProviderFromV2Credentials(initial); } else { throw new api_1.AuthenticationError(`Plugin returned a value that doesn't resemble AWS credentials: ${(0, util_1.inspect)(initial)}`); } } /** * Converts a V2 credential into a V3-compatible provider */ function v3ProviderFromV2Credentials(x) { return async () => { // Get will fetch or refresh as necessary await x.getPromise(); return { accessKeyId: x.accessKeyId, secretAccessKey: x.secretAccessKey, sessionToken: x.sessionToken, expiration: x.expireTime ?? undefined, }; }; } function refreshFromPluginProvider(current, producer) { return async () => { if ((0, provider_caching_1.credentialsAboutToExpire)(current)) { const newCreds = await producer(); if (!isV3Credentials(newCreds)) { throw new api_1.AuthenticationError(`Plugin initially returned static V3 credentials but now returned something else: ${(0, util_1.inspect)(newCreds)}`); } current = newCreds; } return current; }; } function isV3Provider(x) { return typeof x === 'function'; } function isV2Credentials(x) { return !!(x && typeof x === 'object' && x.getPromise); } function isV3Credentials(x) { return !!(x && typeof x === 'object' && x.accessKeyId && !isV2Credentials(x)); } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"credential-plugins.js","sourceRoot":"","sources":["credential-plugins.ts"],"names":[],"mappings":"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","sourcesContent":["import { inspect } from 'util';\nimport type { CredentialProviderSource, ForReading, ForWriting, PluginProviderResult, SDKv2CompatibleCredentials, SDKv3CompatibleCredentialProvider, SDKv3CompatibleCredentials } from '@aws-cdk/cli-plugin-contract';\nimport type { AwsCredentialIdentity, AwsCredentialIdentityProvider } from '@smithy/types';\nimport { credentialsAboutToExpire, makeCachingProvider } from './provider-caching';\nimport { AuthenticationError } from '../../../../@aws-cdk/tmp-toolkit-helpers/src/api';\nimport { debug, warning } from '../../logging';\nimport { formatErrorMessage } from '../../util';\nimport type { Mode } from '../plugin/mode';\nimport { PluginHost } from '../plugin/plugin';\n\n/**\n * Cache for credential providers.\n *\n * Given an account and an operating mode (read or write) will return an\n * appropriate credential provider for credentials for the given account. The\n * credential provider will be cached so that multiple AWS clients for the same\n * environment will not make multiple network calls to obtain credentials.\n *\n * Will use default credentials if they are for the right account; otherwise,\n * all loaded credential provider plugins will be tried to obtain credentials\n * for the given account.\n */\nexport class CredentialPlugins {\n  private readonly cache: { [key: string]: PluginCredentialsFetchResult | undefined } = {};\n  private readonly host: PluginHost;\n\n  constructor(host?: PluginHost) {\n    this.host = host ?? PluginHost.instance;\n  }\n\n  public async fetchCredentialsFor(awsAccountId: string, mode: Mode): Promise<PluginCredentialsFetchResult | undefined> {\n    const key = `${awsAccountId}-${mode}`;\n    if (!(key in this.cache)) {\n      this.cache[key] = await this.lookupCredentials(awsAccountId, mode);\n    }\n    return this.cache[key];\n  }\n\n  public get availablePluginNames(): string[] {\n    return this.host.credentialProviderSources.map((s) => s.name);\n  }\n\n  private async lookupCredentials(awsAccountId: string, mode: Mode): Promise<PluginCredentialsFetchResult | undefined> {\n    const triedSources: CredentialProviderSource[] = [];\n    // Otherwise, inspect the various credential sources we have\n    for (const source of this.host.credentialProviderSources) {\n      let available: boolean;\n      try {\n        available = await source.isAvailable();\n      } catch (e: any) {\n        // This shouldn't happen, but let's guard against it anyway\n        warning(`Uncaught exception in ${source.name}: ${formatErrorMessage(e)}`);\n        available = false;\n      }\n\n      if (!available) {\n        debug('Credentials source %s is not available, ignoring it.', source.name);\n        continue;\n      }\n      triedSources.push(source);\n      let canProvide: boolean;\n      try {\n        canProvide = await source.canProvideCredentials(awsAccountId);\n      } catch (e: any) {\n        // This shouldn't happen, but let's guard against it anyway\n        warning(`Uncaught exception in ${source.name}: ${formatErrorMessage(e)}`);\n        canProvide = false;\n      }\n      if (!canProvide) {\n        continue;\n      }\n      debug(`Using ${source.name} credentials for account ${awsAccountId}`);\n\n      return {\n        credentials: await v3ProviderFromPlugin(() => source.getProvider(awsAccountId, mode as ForReading | ForWriting, {\n          supportsV3Providers: true,\n        })),\n        pluginName: source.name,\n      };\n    }\n    return undefined;\n  }\n}\n\n/**\n * Result from trying to fetch credentials from the Plugin host\n */\nexport interface PluginCredentialsFetchResult {\n  /**\n   * SDK-v3 compatible credential provider\n   */\n  readonly credentials: AwsCredentialIdentityProvider;\n\n  /**\n   * Name of plugin that successfully provided credentials\n   */\n  readonly pluginName: string;\n}\n\n/**\n * Take a function that calls the plugin, and turn it into an SDKv3-compatible credential provider.\n *\n * What we will do is the following:\n *\n * - Query the plugin and see what kind of result it gives us.\n * - If the result is self-refreshing or doesn't need refreshing, we turn it into an SDKv3 provider\n *   and return it directly.\n *   * If the underlying return value is a provider, we will make it a caching provider\n *     (because we can't know if it will cache by itself or not).\n *   * If the underlying return value is a static credential, caching isn't relevant.\n *   * If the underlying return value is V2 credentials, those have caching built-in.\n * - If the result is a static credential that expires, we will wrap it in an SDKv3 provider\n *   that will query the plugin again when the credential expires.\n */\nasync function v3ProviderFromPlugin(producer: () => Promise<PluginProviderResult>): Promise<AwsCredentialIdentityProvider> {\n  const initial = await producer();\n\n  if (isV3Provider(initial)) {\n    // Already a provider, make caching\n    return makeCachingProvider(initial);\n  } else if (isV3Credentials(initial) && initial.expiration === undefined) {\n    // Static credentials that don't need refreshing nor caching\n    return () => Promise.resolve(initial);\n  } else if (isV3Credentials(initial) && initial.expiration !== undefined) {\n    // Static credentials that do need refreshing and caching\n    return refreshFromPluginProvider(initial, producer);\n  } else if (isV2Credentials(initial)) {\n    // V2 credentials that refresh and cache themselves\n    return v3ProviderFromV2Credentials(initial);\n  } else {\n    throw new AuthenticationError(`Plugin returned a value that doesn't resemble AWS credentials: ${inspect(initial)}`);\n  }\n}\n\n/**\n * Converts a V2 credential into a V3-compatible provider\n */\nfunction v3ProviderFromV2Credentials(x: SDKv2CompatibleCredentials): AwsCredentialIdentityProvider {\n  return async () => {\n    // Get will fetch or refresh as necessary\n    await x.getPromise();\n\n    return {\n      accessKeyId: x.accessKeyId,\n      secretAccessKey: x.secretAccessKey,\n      sessionToken: x.sessionToken,\n      expiration: x.expireTime ?? undefined,\n    };\n  };\n}\n\nfunction refreshFromPluginProvider(current: AwsCredentialIdentity, producer: () => Promise<PluginProviderResult>): AwsCredentialIdentityProvider {\n  return async () => {\n    if (credentialsAboutToExpire(current)) {\n      const newCreds = await producer();\n      if (!isV3Credentials(newCreds)) {\n        throw new AuthenticationError(`Plugin initially returned static V3 credentials but now returned something else: ${inspect(newCreds)}`);\n      }\n      current = newCreds;\n    }\n    return current;\n  };\n}\n\nfunction isV3Provider(x: PluginProviderResult): x is SDKv3CompatibleCredentialProvider {\n  return typeof x === 'function';\n}\n\nfunction isV2Credentials(x: PluginProviderResult): x is SDKv2CompatibleCredentials {\n  return !!(x && typeof x === 'object' && (x as SDKv2CompatibleCredentials).getPromise);\n}\n\nfunction isV3Credentials(x: PluginProviderResult): x is SDKv3CompatibleCredentials {\n  return !!(x && typeof x === 'object' && x.accessKeyId && !isV2Credentials(x));\n}\n"]}