UNPKG

@aws-cdk-testing/cli-integ

Version:

Integration tests for the AWS CDK CLI

346 lines 44.4 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.AwsClients = void 0; exports.isStackMissingError = isStackMissingError; exports.isBucketMissingError = isBucketMissingError; exports.retry = retry; exports.outputFromStack = outputFromStack; exports.sleep = sleep; exports.retryOnMatchingErrors = retryOnMatchingErrors; const client_cloudformation_1 = require("@aws-sdk/client-cloudformation"); const client_ecr_1 = require("@aws-sdk/client-ecr"); const client_ecr_public_1 = require("@aws-sdk/client-ecr-public"); const client_ecs_1 = require("@aws-sdk/client-ecs"); const client_iam_1 = require("@aws-sdk/client-iam"); const client_lambda_1 = require("@aws-sdk/client-lambda"); const client_s3_1 = require("@aws-sdk/client-s3"); const client_secrets_manager_1 = require("@aws-sdk/client-secrets-manager"); const client_sns_1 = require("@aws-sdk/client-sns"); const client_sso_1 = require("@aws-sdk/client-sso"); const client_sts_1 = require("@aws-sdk/client-sts"); const credential_providers_1 = require("@aws-sdk/credential-providers"); const util_retry_1 = require("@smithy/util-retry"); class AwsClients { randomString; region; output; identity; static async forIdentity(randomString, region, identity, output) { return new AwsClients(randomString, region, output, identity); } static async forRegion(randomString, region, output) { return new AwsClients(randomString, region, output); } cleanup = []; config; cloudFormation; s3; ecr; ecrPublic; ecs; sso; sns; iam; lambda; sts; secretsManager; constructor( /** A random string to use for temporary resources, like roles (should preferably match unique test-specific randomString) */ randomString, region, output, identity) { this.randomString = randomString; this.region = region; this.output = output; this.identity = identity; this.config = { credentials: this.identity ?? chainableCredentials(this.region), region: this.region, retryStrategy: new util_retry_1.ConfiguredRetryStrategy(9, (attempt) => attempt ** 500), }; this.cloudFormation = new client_cloudformation_1.CloudFormationClient(this.config); this.s3 = new client_s3_1.S3Client(this.config); this.ecr = new client_ecr_1.ECRClient(this.config); this.ecrPublic = new client_ecr_public_1.ECRPUBLICClient({ ...this.config, region: 'us-east-1' /* public gallery is only available in us-east-1 */ }); this.ecs = new client_ecs_1.ECSClient(this.config); this.sso = new client_sso_1.SSOClient(this.config); this.sns = new client_sns_1.SNSClient(this.config); this.iam = new client_iam_1.IAMClient(this.config); this.lambda = new client_lambda_1.LambdaClient(this.config); this.sts = new client_sts_1.STSClient(this.config); this.secretsManager = new client_secrets_manager_1.SecretsManagerClient(this.config); } addCleanup(cleanup) { this.cleanup.push(cleanup); } async dispose() { for (const cleanup of this.cleanup) { try { await cleanup(); } catch (e) { this.output.write(`⚠️ Error during cleanup: ${e.message}\n`); } } this.cleanup.splice(0, this.cleanup.length); } async account() { // Reduce # of retries, we use this as a circuit breaker for detecting no-config const stsClient = new client_sts_1.STSClient({ credentials: this.config.credentials, region: this.config.region, maxAttempts: 2, }); return (await stsClient.send(new client_sts_1.GetCallerIdentityCommand({}))).Account; } /** * If the clients already has an established identity (via atmosphere for example), * return an environment variable map activating it. * * Otherwise, returns undefined. */ identityEnv() { return this.identity ? { AWS_ACCESS_KEY_ID: this.identity.accessKeyId, AWS_SECRET_ACCESS_KEY: this.identity.secretAccessKey, AWS_SESSION_TOKEN: this.identity.sessionToken, // unset any previously used profile because the SDK will prefer // this over static env credentials. this is relevant for tests running on CodeBuild // because we use a profile as our main credentials source. AWS_PROFILE: '', } : undefined; } /** * Resolve the current identity or identity provider to credentials */ async credentials() { const x = this.config.credentials; if (isAwsCredentialIdentity(x)) { return x; } return x(); } async deleteStacks(...stackNames) { if (stackNames.length === 0) { return; } // We purposely do all stacks serially, because they've been ordered // to do the bootstrap stack last. for (const stackName of stackNames) { await this.cloudFormation.send(new client_cloudformation_1.UpdateTerminationProtectionCommand({ EnableTerminationProtection: false, StackName: stackName, })); await this.cloudFormation.send(new client_cloudformation_1.DeleteStackCommand({ StackName: stackName, })); await retry(this.output, `Deleting ${stackName}`, retry.forSeconds(600), async () => { const status = await this.stackStatus(stackName); if (status !== undefined && status.endsWith('_FAILED')) { throw retry.abort(new Error(`'${stackName}' is in state '${status}'`)); } if (status !== undefined) { throw new Error(`Delete of '${stackName}' not complete yet, status: '${status}'`); } }); } } async stackStatus(stackName) { try { return (await this.cloudFormation.send(new client_cloudformation_1.DescribeStacksCommand({ StackName: stackName, }))).Stacks?.[0].StackStatus; } catch (e) { if (isStackMissingError(e)) { return undefined; } throw e; } } async emptyBucket(bucketName, options) { const objects = await this.s3.send(new client_s3_1.ListObjectVersionsCommand({ Bucket: bucketName, })); const deletes = [...(objects.Versions || []), ...(objects.DeleteMarkers || [])].reduce((acc, obj) => { if (typeof obj.VersionId !== 'undefined' && typeof obj.Key !== 'undefined') { acc.push({ Key: obj.Key, VersionId: obj.VersionId }); } else if (typeof obj.Key !== 'undefined') { acc.push({ Key: obj.Key }); } return acc; }, []); if (deletes.length === 0) { return Promise.resolve(); } return this.s3.send(new client_s3_1.DeleteObjectsCommand({ Bucket: bucketName, Delete: { Objects: deletes, Quiet: false, }, BypassGovernanceRetention: options?.bypassGovernance ? true : undefined, })); } async deleteImageRepository(repositoryName) { await this.ecr.send(new client_ecr_1.DeleteRepositoryCommand({ repositoryName: repositoryName, force: true, })); } async deleteBucket(bucketName) { try { await this.emptyBucket(bucketName); await this.s3.send(new client_s3_1.DeleteBucketCommand({ Bucket: bucketName, })); } catch (e) { if (isBucketMissingError(e)) { return; } throw e; } } /** * Create a role that will be cleaned up when the AwsClients object is cleaned up */ async temporaryRole(namePrefix, assumeRolePolicyStatements, policyStatements) { const response = await this.iam.send(new client_iam_1.CreateRoleCommand({ RoleName: `${namePrefix}-${this.randomString}`, AssumeRolePolicyDocument: JSON.stringify({ Version: '2012-10-17', Statement: assumeRolePolicyStatements, }, undefined, 2), Tags: [ { Key: 'deleteme', Value: 'true', }, ], })); await this.iam.send(new client_iam_1.PutRolePolicyCommand({ RoleName: `${namePrefix}-${this.randomString}`, PolicyName: 'DefaultPolicy', PolicyDocument: JSON.stringify({ Version: '2012-10-17', Statement: policyStatements, }, undefined, 2), })); this.addCleanup(() => this.deleteRole(response.Role.RoleName)); return response.Role?.Arn ?? '*CreateRole did not return an ARN*'; } async waitForAssumeRole(roleArn) { await retryOnMatchingErrors(() => this.sts.send(new client_sts_1.AssumeRoleCommand({ RoleArn: roleArn, RoleSessionName: 'test-existence', })), ['AccessDenied'], retry.forSeconds(60)); } async deleteRole(name) { const policiesResponse = await this.iam.send(new client_iam_1.ListRolePoliciesCommand({ RoleName: name, })); for (const policyName of policiesResponse.PolicyNames ?? []) { await this.iam.send(new client_iam_1.DeleteRolePolicyCommand({ RoleName: name, PolicyName: policyName, })); } await this.iam.send(new client_iam_1.DeleteRoleCommand({ RoleName: name, })); } } exports.AwsClients = AwsClients; function isStackMissingError(e) { return e.message.indexOf('does not exist') > -1; } function isBucketMissingError(e) { return e.message.indexOf('does not exist') > -1; } /** * Retry an async operation until a deadline is hit. * * Use `retry.forSeconds()` to construct a deadline relative to right now. * * Exceptions will cause the operation to retry. Use `retry.abort` to annotate an exception * to stop the retry and end in a failure. */ async function retry(output, operation, deadline, block) { let i = 0; output.write(`💈 ${operation}\n`); while (true) { try { i++; const ret = await block(); output.write(`💈 ${operation}: succeeded after ${i} attempts\n`); return ret; } catch (e) { if (e.abort || Date.now() > deadline.getTime()) { throw new Error(`${operation}: did not succeed after ${i} attempts: ${e}`); } output.write(`⏳ ${operation} (${e.message})\n`); await sleep(5000); } } } /** * Make a deadline for the `retry` function relative to the current time. */ retry.forSeconds = (seconds) => { return new Date(Date.now() + seconds * 1000); }; /** * Annotate an error to stop the retrying */ retry.abort = (e) => { e.abort = true; return e; }; function outputFromStack(key, stack) { return (stack.Outputs ?? []).find((o) => o.OutputKey === key)?.OutputValue; } async function sleep(ms) { return new Promise((ok) => setTimeout(ok, ms)); } /** * Retry an async operation with error filtering until a deadline is hit. * * Use `retry.forSeconds()` to construct a deadline relative to right now. * * Only retries on errors with matching names in errorNames array. */ async function retryOnMatchingErrors(operation, errorNames, deadline, interval = 5000) { let i = 0; while (true) { try { i++; return await operation(); } catch (e) { if (Date.now() > deadline.getTime()) { throw new Error(`Operation did not succeed after ${i} attempts: ${e}`); } if (!errorNames.includes(e.name)) { throw e; } await sleep(interval); } } } function chainableCredentials(region) { if ((process.env.CODEBUILD_BUILD_ARN || process.env.GITHUB_RUN_ID) && process.env.AWS_PROFILE) { // in codebuild we must assume the role that the cdk uses // otherwise credentials will just be picked up by the normal sdk // heuristics and expire after an hour. return (0, credential_providers_1.fromIni)({ clientConfig: { region }, }); } // Otherwise just get what's default return (0, credential_providers_1.fromNodeProviderChain)({ clientConfig: { region } }); } function isAwsCredentialIdentity(x) { return Boolean(x && typeof x === 'object' && x.accessKeyId); } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"aws.js","sourceRoot":"","sources":["aws.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  CloudFormationClient,\n  DeleteStackCommand,\n  DescribeStacksCommand,\n  UpdateTerminationProtectionCommand,\n  type Stack,\n} from '@aws-sdk/client-cloudformation';\nimport { DeleteRepositoryCommand, ECRClient } from '@aws-sdk/client-ecr';\nimport { ECRPUBLICClient } from '@aws-sdk/client-ecr-public';\nimport { ECSClient } from '@aws-sdk/client-ecs';\nimport { CreateRoleCommand, DeleteRoleCommand, DeleteRolePolicyCommand, IAMClient, ListRolePoliciesCommand, PutRolePolicyCommand } from '@aws-sdk/client-iam';\nimport { LambdaClient } from '@aws-sdk/client-lambda';\nimport {\n  S3Client,\n  DeleteObjectsCommand,\n  ListObjectVersionsCommand,\n  type ObjectIdentifier,\n  DeleteBucketCommand,\n} from '@aws-sdk/client-s3';\nimport { SecretsManagerClient } from '@aws-sdk/client-secrets-manager';\nimport { SNSClient } from '@aws-sdk/client-sns';\nimport { SSOClient } from '@aws-sdk/client-sso';\nimport { AssumeRoleCommand, STSClient, GetCallerIdentityCommand } from '@aws-sdk/client-sts';\nimport { fromIni, fromNodeProviderChain } from '@aws-sdk/credential-providers';\nimport type { AwsCredentialIdentity, AwsCredentialIdentityProvider, NodeHttpHandlerOptions } from '@smithy/types';\nimport { ConfiguredRetryStrategy } from '@smithy/util-retry';\n\ninterface ClientConfig {\n  readonly credentials: AwsCredentialIdentityProvider | AwsCredentialIdentity;\n  readonly region: string;\n  readonly retryStrategy: ConfiguredRetryStrategy;\n  readonly requestHandler?: NodeHttpHandlerOptions;\n}\n\nexport class AwsClients {\n  public static async forIdentity(randomString: string, region: string, identity: AwsCredentialIdentity, output: NodeJS.WritableStream) {\n    return new AwsClients(randomString, region, output, identity);\n  }\n\n  public static async forRegion(randomString: string, region: string, output: NodeJS.WritableStream) {\n    return new AwsClients(randomString, region, output);\n  }\n\n  private readonly cleanup: (() => Promise<void>)[] = [];\n  private readonly config: ClientConfig;\n\n  public readonly cloudFormation: CloudFormationClient;\n  public readonly s3: S3Client;\n  public readonly ecr: ECRClient;\n  public readonly ecrPublic: ECRPUBLICClient;\n  public readonly ecs: ECSClient;\n  public readonly sso: SSOClient;\n  public readonly sns: SNSClient;\n  public readonly iam: IAMClient;\n  public readonly lambda: LambdaClient;\n  public readonly sts: STSClient;\n  public readonly secretsManager: SecretsManagerClient;\n\n  private constructor(\n    /** A random string to use for temporary resources, like roles (should preferably match unique test-specific randomString) */\n    private readonly randomString: string,\n    public readonly region: string,\n    private readonly output: NodeJS.WritableStream,\n    public readonly identity?: AwsCredentialIdentity) {\n    this.config = {\n      credentials: this.identity ?? chainableCredentials(this.region),\n      region: this.region,\n      retryStrategy: new ConfiguredRetryStrategy(9, (attempt: number) => attempt ** 500),\n    };\n\n    this.cloudFormation = new CloudFormationClient(this.config);\n    this.s3 = new S3Client(this.config);\n    this.ecr = new ECRClient(this.config);\n    this.ecrPublic = new ECRPUBLICClient({ ...this.config, region: 'us-east-1' /* public gallery is only available in us-east-1 */ });\n    this.ecs = new ECSClient(this.config);\n    this.sso = new SSOClient(this.config);\n    this.sns = new SNSClient(this.config);\n    this.iam = new IAMClient(this.config);\n    this.lambda = new LambdaClient(this.config);\n    this.sts = new STSClient(this.config);\n    this.secretsManager = new SecretsManagerClient(this.config);\n  }\n\n  public addCleanup(cleanup: () => Promise<any>) {\n    this.cleanup.push(cleanup);\n  }\n\n  public async dispose() {\n    for (const cleanup of this.cleanup) {\n      try {\n        await cleanup();\n      } catch (e: any) {\n        this.output.write(`⚠️ Error during cleanup: ${e.message}\\n`);\n      }\n    }\n    this.cleanup.splice(0, this.cleanup.length);\n  }\n\n  public async account(): Promise<string> {\n    // Reduce # of retries, we use this as a circuit breaker for detecting no-config\n    const stsClient = new STSClient({\n      credentials: this.config.credentials,\n      region: this.config.region,\n      maxAttempts: 2,\n    });\n\n    return (await stsClient.send(new GetCallerIdentityCommand({}))).Account!;\n  }\n\n  /**\n   * If the clients already has an established identity (via atmosphere for example),\n   * return an environment variable map activating it.\n   *\n   * Otherwise, returns undefined.\n   */\n  public identityEnv(): Record<string, string> | undefined {\n    return this.identity ? {\n      AWS_ACCESS_KEY_ID: this.identity.accessKeyId,\n      AWS_SECRET_ACCESS_KEY: this.identity.secretAccessKey,\n      AWS_SESSION_TOKEN: this.identity.sessionToken!,\n\n      // unset any previously used profile because the SDK will prefer\n      // this over static env credentials. this is relevant for tests running on CodeBuild\n      // because we use a profile as our main credentials source.\n      AWS_PROFILE: '',\n    } : undefined;\n  }\n\n  /**\n   * Resolve the current identity or identity provider to credentials\n   */\n  public async credentials() {\n    const x = this.config.credentials;\n    if (isAwsCredentialIdentity(x)) {\n      return x;\n    }\n    return x();\n  }\n\n  public async deleteStacks(...stackNames: string[]) {\n    if (stackNames.length === 0) {\n      return;\n    }\n\n    // We purposely do all stacks serially, because they've been ordered\n    // to do the bootstrap stack last.\n    for (const stackName of stackNames) {\n      await this.cloudFormation.send(\n        new UpdateTerminationProtectionCommand({\n          EnableTerminationProtection: false,\n          StackName: stackName,\n        }),\n      );\n      await this.cloudFormation.send(\n        new DeleteStackCommand({\n          StackName: stackName,\n        }),\n      );\n\n      await retry(this.output, `Deleting ${stackName}`, retry.forSeconds(600), async () => {\n        const status = await this.stackStatus(stackName);\n        if (status !== undefined && status.endsWith('_FAILED')) {\n          throw retry.abort(new Error(`'${stackName}' is in state '${status}'`));\n        }\n        if (status !== undefined) {\n          throw new Error(`Delete of '${stackName}' not complete yet, status: '${status}'`);\n        }\n      });\n    }\n  }\n\n  public async stackStatus(stackName: string): Promise<string | undefined> {\n    try {\n      return (\n        await this.cloudFormation.send(\n          new DescribeStacksCommand({\n            StackName: stackName,\n          }),\n        )\n      ).Stacks?.[0].StackStatus;\n    } catch (e: any) {\n      if (isStackMissingError(e)) {\n        return undefined;\n      }\n      throw e;\n    }\n  }\n\n  public async emptyBucket(bucketName: string, options?: { bypassGovernance?: boolean }) {\n    const objects = await this.s3.send(\n      new ListObjectVersionsCommand({\n        Bucket: bucketName,\n      }),\n    );\n\n    const deletes = [...(objects.Versions || []), ...(objects.DeleteMarkers || [])].reduce((acc, obj) => {\n      if (typeof obj.VersionId !== 'undefined' && typeof obj.Key !== 'undefined') {\n        acc.push({ Key: obj.Key, VersionId: obj.VersionId });\n      } else if (typeof obj.Key !== 'undefined') {\n        acc.push({ Key: obj.Key });\n      }\n      return acc;\n    }, [] as ObjectIdentifier[]);\n\n    if (deletes.length === 0) {\n      return Promise.resolve();\n    }\n\n    return this.s3.send(\n      new DeleteObjectsCommand({\n        Bucket: bucketName,\n        Delete: {\n          Objects: deletes,\n          Quiet: false,\n        },\n        BypassGovernanceRetention: options?.bypassGovernance ? true : undefined,\n      }),\n    );\n  }\n\n  public async deleteImageRepository(repositoryName: string) {\n    await this.ecr.send(\n      new DeleteRepositoryCommand({\n        repositoryName: repositoryName,\n        force: true,\n      }),\n    );\n  }\n\n  public async deleteBucket(bucketName: string) {\n    try {\n      await this.emptyBucket(bucketName);\n\n      await this.s3.send(\n        new DeleteBucketCommand({\n          Bucket: bucketName,\n        }),\n      );\n    } catch (e: any) {\n      if (isBucketMissingError(e)) {\n        return;\n      }\n      throw e;\n    }\n  }\n\n  /**\n   * Create a role that will be cleaned up when the AwsClients object is cleaned up\n   */\n  public async temporaryRole(namePrefix: string, assumeRolePolicyStatements: any[], policyStatements: any[]) {\n    const response = await this.iam.send(new CreateRoleCommand({\n      RoleName: `${namePrefix}-${this.randomString}`,\n      AssumeRolePolicyDocument: JSON.stringify({\n        Version: '2012-10-17',\n        Statement: assumeRolePolicyStatements,\n      }, undefined, 2),\n      Tags: [\n        {\n          Key: 'deleteme',\n          Value: 'true',\n        },\n      ],\n    }));\n    await this.iam.send(new PutRolePolicyCommand({\n      RoleName: `${namePrefix}-${this.randomString}`,\n      PolicyName: 'DefaultPolicy',\n      PolicyDocument: JSON.stringify({\n        Version: '2012-10-17',\n        Statement: policyStatements,\n      }, undefined, 2),\n    }));\n\n    this.addCleanup(() => this.deleteRole(response.Role!.RoleName!));\n\n    return response.Role?.Arn ?? '*CreateRole did not return an ARN*';\n  }\n\n  public async waitForAssumeRole(roleArn: string) {\n    await retryOnMatchingErrors(\n      () => this.sts.send(new AssumeRoleCommand({\n        RoleArn: roleArn,\n        RoleSessionName: 'test-existence',\n      })),\n      ['AccessDenied'],\n      retry.forSeconds(60),\n    );\n  }\n\n  public async deleteRole(name: string) {\n    const policiesResponse = await this.iam.send(new ListRolePoliciesCommand({\n      RoleName: name,\n    }));\n\n    for (const policyName of policiesResponse.PolicyNames ?? []) {\n      await this.iam.send(new DeleteRolePolicyCommand({\n        RoleName: name,\n        PolicyName: policyName,\n      }));\n    }\n\n    await this.iam.send(new DeleteRoleCommand({\n      RoleName: name,\n    }));\n  }\n}\n\nexport function isStackMissingError(e: Error) {\n  return e.message.indexOf('does not exist') > -1;\n}\n\nexport function isBucketMissingError(e: Error) {\n  return e.message.indexOf('does not exist') > -1;\n}\n\n/**\n * Retry an async operation until a deadline is hit.\n *\n * Use `retry.forSeconds()` to construct a deadline relative to right now.\n *\n * Exceptions will cause the operation to retry. Use `retry.abort` to annotate an exception\n * to stop the retry and end in a failure.\n */\nexport async function retry<A>(\n  output: NodeJS.WritableStream,\n  operation: string,\n  deadline: Date,\n  block: () => Promise<A>,\n): Promise<A> {\n  let i = 0;\n  output.write(`💈 ${operation}\\n`);\n  while (true) {\n    try {\n      i++;\n      const ret = await block();\n      output.write(`💈 ${operation}: succeeded after ${i} attempts\\n`);\n      return ret;\n    } catch (e: any) {\n      if (e.abort || Date.now() > deadline.getTime()) {\n        throw new Error(`${operation}: did not succeed after ${i} attempts: ${e}`);\n      }\n      output.write(`⏳ ${operation} (${e.message})\\n`);\n      await sleep(5000);\n    }\n  }\n}\n\n/**\n * Make a deadline for the `retry` function relative to the current time.\n */\nretry.forSeconds = (seconds: number): Date => {\n  return new Date(Date.now() + seconds * 1000);\n};\n\n/**\n * Annotate an error to stop the retrying\n */\nretry.abort = (e: Error): Error => {\n  (e as any).abort = true;\n  return e;\n};\n\nexport function outputFromStack(key: string, stack: Stack): string | undefined {\n  return (stack.Outputs ?? []).find((o) => o.OutputKey === key)?.OutputValue;\n}\n\nexport async function sleep(ms: number) {\n  return new Promise((ok) => setTimeout(ok, ms));\n}\n\n/**\n * Retry an async operation with error filtering until a deadline is hit.\n *\n * Use `retry.forSeconds()` to construct a deadline relative to right now.\n *\n * Only retries on errors with matching names in errorNames array.\n */\nexport async function retryOnMatchingErrors<T>(\n  operation: () => Promise<T>,\n  errorNames: string[],\n  deadline: Date,\n  interval: number = 5000,\n): Promise<T> {\n  let i = 0;\n  while (true) {\n    try {\n      i++;\n      return await operation();\n    } catch (e: any) {\n      if (Date.now() > deadline.getTime()) {\n        throw new Error(`Operation did not succeed after ${i} attempts: ${e}`);\n      }\n      if (!errorNames.includes(e.name)) {\n        throw e;\n      }\n      await sleep(interval);\n    }\n  }\n}\n\nfunction chainableCredentials(region: string): AwsCredentialIdentityProvider {\n  if ((process.env.CODEBUILD_BUILD_ARN || process.env.GITHUB_RUN_ID) && process.env.AWS_PROFILE) {\n    // in codebuild we must assume the role that the cdk uses\n    // otherwise credentials will just be picked up by the normal sdk\n    // heuristics and expire after an hour.\n    return fromIni({\n      clientConfig: { region },\n    });\n  }\n\n  // Otherwise just get what's default\n  return fromNodeProviderChain({ clientConfig: { region } });\n}\n\nfunction isAwsCredentialIdentity(x: any): x is AwsCredentialIdentity {\n  return Boolean(x && typeof x === 'object' && x.accessKeyId);\n}\n"]}