UNPKG

@aws-cdk/integ-tests-alpha

Version:

CDK Integration Testing Constructs

154 lines 18.2 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.WaiterStateMachine = void 0; const jsiiDeprecationWarnings = require("../../.warnings.jsii.js"); const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const core_1 = require("aws-cdk-lib/core"); const constructs_1 = require("constructs"); const providers_1 = require("./providers"); /** * A very simple StateMachine construct highly customized to the provider framework. * This is so that this package does not need to depend on aws-stepfunctions module. * * The state machine continuously calls the isCompleteHandler, until it succeeds or times out. * The handler is called `maxAttempts` times with an `interval` duration and a `backoffRate` rate. * * For example with: * - maxAttempts = 360 (30 minutes) * - interval = 5 * - backoffRate = 1 (no backoff) * * it will make the API Call every 5 seconds and fail after 360 failures. * * If the backoff rate is changed to 2 (for example), it will * - make the first call * - wait 5 seconds * - make the second call * - wait 15 seconds * - etc. */ class WaiterStateMachine extends constructs_1.Construct { static [JSII_RTTI_SYMBOL_1] = { fqn: "@aws-cdk/integ-tests-alpha.WaiterStateMachine", version: "2.231.0-alpha.0" }; /** * The ARN of the statemachine */ stateMachineArn; /** * The IAM Role ARN of the role used by the state machine */ roleArn; /** * The AssertionsProvide that handles async requests */ isCompleteProvider; constructor(scope, id, props = {}) { super(scope, id); try { jsiiDeprecationWarnings._aws_cdk_integ_tests_alpha_WaiterStateMachineProps(props); } catch (error) { if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") { Error.captureStackTrace(error, WaiterStateMachine); } throw error; } const interval = props.interval || core_1.Duration.seconds(5); const totalTimeout = props.totalTimeout || core_1.Duration.minutes(30); const maxAttempts = calculateMaxRetries(totalTimeout.toSeconds(), interval.toSeconds(), props.backoffRate ?? 1); if (Math.round(maxAttempts) !== maxAttempts) { throw new Error(`Cannot determine retry count since totalTimeout=${totalTimeout.toSeconds()}s is not integrally dividable by queryInterval=${interval.toSeconds()}s`); } this.isCompleteProvider = new providers_1.AssertionsProvider(this, 'IsCompleteProvider', { handler: 'index.isComplete', uuid: '76b3e830-a873-425f-8453-eddd85c86925', }); const timeoutProvider = new providers_1.AssertionsProvider(this, 'TimeoutProvider', { handler: 'index.onTimeout', uuid: '5c1898e0-96fb-4e3e-95d5-f6c67f3ce41a', }); const role = new core_1.CfnResource(this, 'Role', { type: 'AWS::IAM::Role', properties: { AssumeRolePolicyDocument: { Version: '2012-10-17', Statement: [{ Action: 'sts:AssumeRole', Effect: 'Allow', Principal: { Service: 'states.amazonaws.com' } }], }, Policies: [ { PolicyName: 'InlineInvokeFunctions', PolicyDocument: { Version: '2012-10-17', Statement: [{ Action: 'lambda:InvokeFunction', Effect: 'Allow', Resource: [ this.isCompleteProvider.serviceToken, timeoutProvider.serviceToken, ], }], }, }, ], }, }); const definition = core_1.Stack.of(this).toJsonString({ StartAt: 'framework-isComplete-task', States: { 'framework-isComplete-task': { End: true, Retry: [{ ErrorEquals: ['States.ALL'], IntervalSeconds: interval.toSeconds(), MaxAttempts: maxAttempts, BackoffRate: props.backoffRate ?? 1, }], Catch: [{ ErrorEquals: ['States.ALL'], Next: 'framework-onTimeout-task', }], Type: 'Task', Resource: this.isCompleteProvider.serviceToken, }, 'framework-onTimeout-task': { End: true, Type: 'Task', Resource: timeoutProvider.serviceToken, }, }, }); const resource = new core_1.CfnResource(this, 'Resource', { type: 'AWS::StepFunctions::StateMachine', properties: { DefinitionString: definition, RoleArn: role.getAtt('Arn'), }, }); resource.node.addDependency(role); this.stateMachineArn = resource.ref; this.roleArn = role.getAtt('Arn').toString(); this.isCompleteProvider.grantInvoke(this.roleArn); timeoutProvider.grantInvoke(this.roleArn); } } exports.WaiterStateMachine = WaiterStateMachine; /** * Calculate the max number of retries */ function calculateMaxRetries(maxSeconds, intervalSeconds, backoff) { // if backoff === 1 then we aren't really using backoff if (backoff === 1) { return Math.floor(maxSeconds / intervalSeconds); } let retries = 1; let nextInterval = intervalSeconds; let i = 0; while (i < maxSeconds) { nextInterval = nextInterval + nextInterval * backoff; i += nextInterval; if (i >= maxSeconds) break; retries++; } return retries; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"waiter-state-machine.js","sourceRoot":"","sources":["waiter-state-machine.ts"],"names":[],"mappings":"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","sourcesContent":["import { CfnResource, Duration, Stack } from 'aws-cdk-lib/core';\nimport { Construct } from 'constructs';\nimport { AssertionsProvider } from './providers';\n\n/**\n * Options for creating a WaiterStateMachine\n */\nexport interface WaiterStateMachineOptions {\n  /**\n   * The total time that the state machine will wait\n   * for a successful response\n   *\n   * @default Duration.minutes(30)\n   */\n  readonly totalTimeout?: Duration;\n\n  /**\n   * The interval (number of seconds) to wait between attempts.\n   *\n   * @default Duration.seconds(5)\n   */\n  readonly interval?: Duration;\n\n  /**\n   * Backoff between attempts.\n   *\n   * This is the multiplier by which the retry interval increases\n   * after each retry attempt.\n   *\n   * By default there is no backoff. Each retry will wait the amount of time\n   * specified by `interval`.\n   *\n   * @default 1 (no backoff)\n   */\n  readonly backoffRate?: number;\n}\n\n/**\n * Props for creating a WaiterStateMachine\n */\nexport interface WaiterStateMachineProps extends WaiterStateMachineOptions {}\n\n/**\n * A very simple StateMachine construct highly customized to the provider framework.\n * This is so that this package does not need to depend on aws-stepfunctions module.\n *\n * The state machine continuously calls the isCompleteHandler, until it succeeds or times out.\n * The handler is called `maxAttempts` times with an `interval` duration and a `backoffRate` rate.\n *\n * For example with:\n * - maxAttempts = 360 (30 minutes)\n * - interval = 5\n * - backoffRate = 1 (no backoff)\n *\n * it will make the API Call every 5 seconds and fail after 360 failures.\n *\n * If the backoff rate is changed to 2 (for example), it will\n * - make the first call\n * - wait 5 seconds\n * - make the second call\n * - wait 15 seconds\n * - etc.\n */\nexport class WaiterStateMachine extends Construct {\n  /**\n   * The ARN of the statemachine\n   */\n  public readonly stateMachineArn: string;\n\n  /**\n   * The IAM Role ARN of the role used by the state machine\n   */\n  public readonly roleArn: string;\n\n  /**\n   * The AssertionsProvide that handles async requests\n   */\n  public readonly isCompleteProvider: AssertionsProvider;\n\n  constructor(scope: Construct, id: string, props: WaiterStateMachineProps = {}) {\n    super(scope, id);\n    const interval = props.interval || Duration.seconds(5);\n    const totalTimeout = props.totalTimeout || Duration.minutes(30);\n    const maxAttempts = calculateMaxRetries(totalTimeout.toSeconds(), interval.toSeconds(), props.backoffRate ?? 1);\n\n    if (Math.round(maxAttempts) !== maxAttempts) {\n      throw new Error(`Cannot determine retry count since totalTimeout=${totalTimeout.toSeconds()}s is not integrally dividable by queryInterval=${interval.toSeconds()}s`);\n    }\n\n    this.isCompleteProvider = new AssertionsProvider(this, 'IsCompleteProvider', {\n      handler: 'index.isComplete',\n      uuid: '76b3e830-a873-425f-8453-eddd85c86925',\n    });\n\n    const timeoutProvider = new AssertionsProvider(this, 'TimeoutProvider', {\n      handler: 'index.onTimeout',\n      uuid: '5c1898e0-96fb-4e3e-95d5-f6c67f3ce41a',\n    });\n\n    const role = new CfnResource(this, 'Role', {\n      type: 'AWS::IAM::Role',\n      properties: {\n        AssumeRolePolicyDocument: {\n          Version: '2012-10-17',\n          Statement: [{ Action: 'sts:AssumeRole', Effect: 'Allow', Principal: { Service: 'states.amazonaws.com' } }],\n        },\n        Policies: [\n          {\n            PolicyName: 'InlineInvokeFunctions',\n            PolicyDocument: {\n              Version: '2012-10-17',\n              Statement: [{\n                Action: 'lambda:InvokeFunction',\n                Effect: 'Allow',\n                Resource: [\n                  this.isCompleteProvider.serviceToken,\n                  timeoutProvider.serviceToken,\n                ],\n              }],\n            },\n          },\n        ],\n      },\n    });\n\n    const definition = Stack.of(this).toJsonString({\n      StartAt: 'framework-isComplete-task',\n      States: {\n        'framework-isComplete-task': {\n          End: true,\n          Retry: [{\n            ErrorEquals: ['States.ALL'],\n            IntervalSeconds: interval.toSeconds(),\n            MaxAttempts: maxAttempts,\n            BackoffRate: props.backoffRate ?? 1,\n          }],\n          Catch: [{\n            ErrorEquals: ['States.ALL'],\n            Next: 'framework-onTimeout-task',\n          }],\n          Type: 'Task',\n          Resource: this.isCompleteProvider.serviceToken,\n        },\n        'framework-onTimeout-task': {\n          End: true,\n          Type: 'Task',\n          Resource: timeoutProvider.serviceToken,\n        },\n      },\n    });\n\n    const resource = new CfnResource(this, 'Resource', {\n      type: 'AWS::StepFunctions::StateMachine',\n      properties: {\n        DefinitionString: definition,\n        RoleArn: role.getAtt('Arn'),\n      },\n    });\n    resource.node.addDependency(role);\n\n    this.stateMachineArn = resource.ref;\n    this.roleArn = role.getAtt('Arn').toString();\n    this.isCompleteProvider.grantInvoke(this.roleArn);\n    timeoutProvider.grantInvoke(this.roleArn);\n  }\n}\n\n/**\n * Calculate the max number of retries\n */\nfunction calculateMaxRetries(maxSeconds: number, intervalSeconds: number, backoff: number): number {\n  // if backoff === 1 then we aren't really using backoff\n  if (backoff === 1) {\n    return Math.floor(maxSeconds / intervalSeconds);\n  }\n  let retries = 1;\n  let nextInterval = intervalSeconds;\n  let i = 0;\n  while (i < maxSeconds) {\n    nextInterval = nextInterval+nextInterval*backoff;\n    i+=nextInterval;\n    if (i >= maxSeconds) break;\n    retries++;\n  }\n  return retries;\n}\n"]}