@dontirun/state-machine-semaphore 0.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/lib/index.d.ts ADDED
@@ -0,0 +1,47 @@
1
+ import { StateMachineFragment, State, IChainable, INextable } from 'aws-cdk-lib/aws-stepfunctions';
2
+ import { Construct } from 'constructs';
3
+ /**
4
+ * Interface for creating a SemaphoreGenerator
5
+ */
6
+ export interface SemaphoreGeneratorProps {
7
+ /**
8
+ * Optionally set the DynamoDB table to have a specific read/write capacity with PROVISIONED billing.
9
+ * @default PAY_PER_REQUEST
10
+ */
11
+ readonly tableReadWriteCapacity?: TableReadWriteCapacity;
12
+ }
13
+ /**
14
+ * Read and write capacity for a PROVISIONED billing DynamoDB table.
15
+ */
16
+ export interface TableReadWriteCapacity {
17
+ readonly readCapacity: number;
18
+ readonly writeCapacity: number;
19
+ }
20
+ export interface IChainNextable extends IChainable, INextable {
21
+ }
22
+ /**
23
+ * Sets up up the DynamoDB table that stores the State Machine semaphores.
24
+ * Call `generateSemaphoredJob` to generate semaphored jobs.
25
+ */
26
+ export declare class SemaphoreGenerator extends Construct {
27
+ /**
28
+ * The DynamoDB table used to store semaphores.
29
+ */
30
+ private semaphoreTable;
31
+ /**
32
+ * The names and associated concurrency limits and number of uses of the sempahores.
33
+ */
34
+ private semaphoreTracker;
35
+ constructor(scope: Construct, id: string, props?: SemaphoreGeneratorProps);
36
+ /**
37
+ * Generates a semaphore for a StepFunction job (or chained set of jobs) to limit parallelism across executions.
38
+ * @param lockName The name of the semaphore.
39
+ * @param limit The maximum number of concurrent executions for the given lock.
40
+ * @param job The job (or chained jobs) to be semaphored.
41
+ * @param nextState The State to go to after the semaphored job completes.
42
+ * @param reuseLock Explicility allow the reuse of a named lock from a previously generated job. Throws an error if a different `limit` is specified. Default: false.
43
+ * @param comments Adds detailed comments to lock related states. Significantly increases CloudFormation template size. Default: false.
44
+ * @returns A StateMachineFragment that can chained to other states in the State Machine.
45
+ */
46
+ generateSemaphoredJob(lockName: string, limit: number, job: IChainNextable, nextState: State, reuseLock?: boolean, comments?: boolean): StateMachineFragment;
47
+ }
package/lib/index.js ADDED
@@ -0,0 +1,171 @@
1
+ "use strict";
2
+ var _a;
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.SemaphoreGenerator = void 0;
5
+ const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
+ const aws_cdk_lib_1 = require("aws-cdk-lib");
7
+ const aws_dynamodb_1 = require("aws-cdk-lib/aws-dynamodb");
8
+ const aws_stepfunctions_1 = require("aws-cdk-lib/aws-stepfunctions");
9
+ const aws_stepfunctions_tasks_1 = require("aws-cdk-lib/aws-stepfunctions-tasks");
10
+ const constructs_1 = require("constructs");
11
+ /**
12
+ * Sets up up the DynamoDB table that stores the State Machine semaphores.
13
+ * Call `generateSemaphoredJob` to generate semaphored jobs.
14
+ */
15
+ class SemaphoreGenerator extends constructs_1.Construct {
16
+ constructor(scope, id, props) {
17
+ super(scope, id);
18
+ this.semaphoreTracker = new Map();
19
+ this.semaphoreTable = new aws_dynamodb_1.Table(this, 'StateMachineSempahoreTable', {
20
+ partitionKey: {
21
+ name: 'LockName',
22
+ type: aws_dynamodb_1.AttributeType.STRING,
23
+ },
24
+ readCapacity: props?.tableReadWriteCapacity?.readCapacity,
25
+ writeCapacity: props?.tableReadWriteCapacity?.writeCapacity,
26
+ billingMode: props?.tableReadWriteCapacity ? aws_dynamodb_1.BillingMode.PROVISIONED : aws_dynamodb_1.BillingMode.PAY_PER_REQUEST,
27
+ });
28
+ }
29
+ /**
30
+ * Generates a semaphore for a StepFunction job (or chained set of jobs) to limit parallelism across executions.
31
+ * @param lockName The name of the semaphore.
32
+ * @param limit The maximum number of concurrent executions for the given lock.
33
+ * @param job The job (or chained jobs) to be semaphored.
34
+ * @param nextState The State to go to after the semaphored job completes.
35
+ * @param reuseLock Explicility allow the reuse of a named lock from a previously generated job. Throws an error if a different `limit` is specified. Default: false.
36
+ * @param comments Adds detailed comments to lock related states. Significantly increases CloudFormation template size. Default: false.
37
+ * @returns A StateMachineFragment that can chained to other states in the State Machine.
38
+ */
39
+ generateSemaphoredJob(lockName, limit, job, nextState, reuseLock, comments) {
40
+ let lockInfo = this.semaphoreTracker.get(lockName);
41
+ if (lockInfo) {
42
+ if (reuseLock) {
43
+ if (lockInfo.limit != limit) {
44
+ throw new Error(`The reused \`lockName\` "${lockName}" was given a different \`limit\` than previously defined. Given: ${limit}, Previous: ${lockInfo.limit}.`);
45
+ }
46
+ else {
47
+ lockInfo = { limit: lockInfo.limit, timesUsed: lockInfo.timesUsed + 1 };
48
+ this.semaphoreTracker.set(lockName, lockInfo);
49
+ }
50
+ }
51
+ else {
52
+ throw new Error(`The \`lockName\` "${lockName}" was reused without explicitly allowing reuse. Set \`reuseLock\` to \`true\` if you want to reuse the lock.`);
53
+ }
54
+ }
55
+ else {
56
+ lockInfo = { limit: limit, timesUsed: 1 };
57
+ this.semaphoreTracker.set(lockName, lockInfo);
58
+ }
59
+ const getLock = new aws_stepfunctions_1.Parallel(this, `Get ${lockName} Lock: ${lockInfo.timesUsed}`, { resultPath: aws_stepfunctions_1.JsonPath.DISCARD });
60
+ const acquireLock = new aws_stepfunctions_tasks_1.DynamoUpdateItem(this, `Acquire ${lockName} Lock: ${lockInfo.timesUsed}`, {
61
+ comment: comments ? `Acquire a lock using a conditional update to DynamoDB. This update will do two things:
62
+ 1) increment a counter for the number of held locks
63
+ 2) add an attribute to the DynamoDB Item with a unique key for this execution and with a value of the time when the lock was Acquired
64
+ The Update includes a conditional expression that will fail under two circumstances:
65
+ 1) if the maximum number of locks have already been distributed
66
+ 2) if the current execution already owns a lock. The latter check is important to ensure the same execution doesn't increase the counter more than once
67
+ If either of these conditions are not met, then the task will fail with a DynamoDB.ConditionalCheckFailedException error, retry a few times, then if it is still not successful \
68
+ it will move off to another branch of the workflow. If this is the first time that a given lockname has been used, there will not be a row in DynamoDB \
69
+ so the update will fail with DynamoDB.AmazonDynamoDBException. In that case, this state sends the workflow to state that will create that row to initialize.
70
+ ` : undefined,
71
+ table: this.semaphoreTable,
72
+ key: { LockName: aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString(lockName) },
73
+ expressionAttributeNames: {
74
+ '#currentlockcount': 'currentlockcount',
75
+ '#lockownerid.$': '$$.Execution.Id',
76
+ },
77
+ expressionAttributeValues: {
78
+ ':increase': aws_stepfunctions_tasks_1.DynamoAttributeValue.fromNumber(1),
79
+ ':limit': aws_stepfunctions_tasks_1.DynamoAttributeValue.fromNumber(limit),
80
+ ':lockacquiredtime': aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString(aws_stepfunctions_1.JsonPath.stringAt('$$.State.EnteredTime')),
81
+ },
82
+ updateExpression: 'SET #currentlockcount = #currentlockcount + :increase, #lockownerid = :lockacquiredtime',
83
+ conditionExpression: 'currentlockcount <> :limit and attribute_not_exists(#lockownerid)',
84
+ returnValues: aws_stepfunctions_tasks_1.DynamoReturnValues.UPDATED_NEW,
85
+ resultPath: '$.lockinfo.acquirelock',
86
+ });
87
+ const initializeLockItem = new aws_stepfunctions_tasks_1.DynamoPutItem(this, `Initialize ${lockName} Lock Item: ${lockInfo.timesUsed}`, {
88
+ comment: comments ? `This state handles the case where an item hasn't been created for this lock yet. \
89
+ In that case, it will insert an initial item that includes the lock name as the key and currentlockcount of 0. \
90
+ The Put to DynamoDB includes a conditonal expression to fail if the an item with that key already exists, which avoids a race condition if multiple executions start at the same time. \
91
+ There are other reasons that the previous state could fail and end up here, so this is safe in those cases too.` : undefined,
92
+ table: this.semaphoreTable,
93
+ item: {
94
+ LockName: aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString(lockName),
95
+ currentlockcount: aws_stepfunctions_tasks_1.DynamoAttributeValue.fromNumber(0),
96
+ },
97
+ conditionExpression: 'LockName <> :lockname',
98
+ expressionAttributeValues: {
99
+ ':lockname': aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString(lockName),
100
+ },
101
+ resultPath: aws_stepfunctions_1.JsonPath.DISCARD,
102
+ });
103
+ const getCurrentLockRecord = new aws_stepfunctions_tasks_1.DynamoGetItem(this, `Get Current ${lockName} Lock Record: ${lockInfo.timesUsed}`, {
104
+ comment: comments ? 'This state is called when the execution is unable to acquire a lock because there limit has either been exceeded or because this execution already holds a lock. \
105
+ In that case, this task loads info from DDB for the current lock item so that the right decision can be made in subsequent states.' : undefined,
106
+ table: this.semaphoreTable,
107
+ key: { LockName: aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString(lockName) },
108
+ expressionAttributeNames: { '#lockownerid.$': '$$.Execution.Id' },
109
+ projectionExpression: [new aws_stepfunctions_tasks_1.DynamoProjectionExpression().withAttribute('#lockownerid')],
110
+ resultSelector: {
111
+ 'Item.$': '$.Item',
112
+ 'ItemString.$': 'States.JsonToString($.Item)',
113
+ },
114
+ resultPath: '$.lockinfo.currentlockitem',
115
+ });
116
+ const checkIfLockAcquired = new aws_stepfunctions_1.Choice(this, `Check if ${lockName} Lock Already Acquired: ${lockInfo.timesUsed}`, {
117
+ comment: comments ? `This state checks to see if the current execution already holds a lock. It can tell that by looking for Z, which will be indicative of the timestamp value. \
118
+ That will only be there in the stringified version of the data returned from DDB if this execution holds a lock.` : undefined,
119
+ });
120
+ const continueBecauseLockWasAlreadyAcquired = new aws_stepfunctions_1.Pass(this, `Continue Because ${lockName} Lock Was Already Acquired: ${lockInfo.timesUsed}`, {
121
+ comment: comments ? 'In this state, we have confimed that lock is already held, so we pass the original execution input into the the function that does the work.' : undefined,
122
+ });
123
+ const waitToGetLock = new aws_stepfunctions_1.Wait(this, `Wait to Get ${lockName} Lock: ${lockInfo.timesUsed}`, {
124
+ comment: comments ? 'If the lock indeed not been succesfully Acquired, then wait for a bit before trying again.' : undefined,
125
+ time: aws_stepfunctions_1.WaitTime.duration(aws_cdk_lib_1.Duration.seconds(3)),
126
+ });
127
+ acquireLock.addRetry({ errors: ['DynamoDB.AmazonDynamoDBException'], maxAttempts: 0 })
128
+ .addRetry({ maxAttempts: 6, backoffRate: 2 })
129
+ .addCatch(initializeLockItem, { errors: ['DynamoDB.AmazonDynamoDBException'], resultPath: '$.lockinfo.acquisitionerror' })
130
+ .addCatch(getCurrentLockRecord, { errors: ['DynamoDB.ConditionalCheckFailedException'], resultPath: '$.lockinfo.acquisitionerror' });
131
+ initializeLockItem.addCatch(acquireLock, { resultPath: aws_stepfunctions_1.JsonPath.DISCARD });
132
+ getCurrentLockRecord.next(checkIfLockAcquired);
133
+ checkIfLockAcquired.when(aws_stepfunctions_1.Condition.and(aws_stepfunctions_1.Condition.isPresent('$.lockinfo.currentlockitem.ItemString'), aws_stepfunctions_1.Condition.stringMatches('$.lockinfo.currentlockitem.ItemString', '*Z')), continueBecauseLockWasAlreadyAcquired);
134
+ checkIfLockAcquired.otherwise(waitToGetLock);
135
+ waitToGetLock.next(acquireLock);
136
+ const releaseLock = new aws_stepfunctions_tasks_1.DynamoUpdateItem(this, `Release ${lockName} Lock: ${lockInfo.timesUsed}`, {
137
+ table: this.semaphoreTable,
138
+ key: { LockName: aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString(lockName) },
139
+ expressionAttributeNames: {
140
+ '#currentlockcount': 'currentlockcount',
141
+ '#lockownerid.$': '$$.Execution.Id',
142
+ },
143
+ expressionAttributeValues: {
144
+ ':decrease': aws_stepfunctions_tasks_1.DynamoAttributeValue.fromNumber(1),
145
+ },
146
+ updateExpression: 'SET #currentlockcount = #currentlockcount - :decrease REMOVE #lockownerid',
147
+ conditionExpression: 'attribute_exists(#lockownerid)',
148
+ returnValues: aws_stepfunctions_tasks_1.DynamoReturnValues.UPDATED_NEW,
149
+ resultPath: aws_stepfunctions_1.JsonPath.DISCARD,
150
+ });
151
+ releaseLock.addRetry({ errors: ['DynamoDB.ConditionalCheckFailedException'], maxAttempts: 0 })
152
+ .addRetry({ maxAttempts: 5, backoffRate: 1.5 })
153
+ .addCatch(nextState, { errors: ['DynamoDB.ConditionalCheckFailedException'], resultPath: '$.lockinfo.acquisitionerror' })
154
+ .next(nextState);
155
+ getLock.branch(acquireLock);
156
+ getLock.endStates.forEach(j => j.next(job));
157
+ job.next(releaseLock);
158
+ class SemaphoredJob extends aws_stepfunctions_1.StateMachineFragment {
159
+ constructor() {
160
+ super(...arguments);
161
+ this.startState = getLock;
162
+ this.endStates = nextState.endStates;
163
+ }
164
+ }
165
+ return new SemaphoredJob(this, `${lockName}${lockInfo.timesUsed}`);
166
+ }
167
+ }
168
+ exports.SemaphoreGenerator = SemaphoreGenerator;
169
+ _a = JSII_RTTI_SYMBOL_1;
170
+ SemaphoreGenerator[_a] = { fqn: "@dontirun/state-machine-semaphore.SemaphoreGenerator", version: "0.0.0" };
171
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["import { Duration } from 'aws-cdk-lib';\nimport { Table, BillingMode, AttributeType } from 'aws-cdk-lib/aws-dynamodb';\nimport { Parallel, StateMachineFragment, JsonPath, Choice, Pass, Wait, WaitTime, Condition, State, IChainable, INextable } from 'aws-cdk-lib/aws-stepfunctions';\nimport { DynamoAttributeValue, DynamoGetItem, DynamoProjectionExpression, DynamoPutItem, DynamoReturnValues, DynamoUpdateItem } from 'aws-cdk-lib/aws-stepfunctions-tasks';\nimport { Construct } from 'constructs';\n\n/**\n * Interface for creating a SemaphoreGenerator\n */\nexport interface SemaphoreGeneratorProps {\n  /**\n   * Optionally set the DynamoDB table to have a specific read/write capacity with PROVISIONED billing.\n   * @default PAY_PER_REQUEST\n   */\n  readonly tableReadWriteCapacity?: TableReadWriteCapacity;\n}\n\n/**\n * Read and write capacity for a PROVISIONED billing DynamoDB table.\n */\nexport interface TableReadWriteCapacity {\n  readonly readCapacity: number;\n  readonly writeCapacity: number;\n}\n\ninterface UsageTracker {\n  readonly limit: number;\n  readonly timesUsed: number;\n}\n\nexport interface IChainNextable extends IChainable, INextable { }\n\n/**\n * Sets up up the DynamoDB table that stores the State Machine semaphores.\n * Call `generateSemaphoredJob` to generate semaphored jobs.\n */\nexport class SemaphoreGenerator extends Construct {\n\n  /**\n   * The DynamoDB table used to store semaphores.\n   */\n  private semaphoreTable: Table;\n\n  /**\n   * The names and associated concurrency limits and number of uses of the sempahores.\n   */\n  private semaphoreTracker: Map<string, UsageTracker>;\n\n  constructor(scope: Construct, id: string, props?: SemaphoreGeneratorProps) {\n    super(scope, id);\n    this.semaphoreTracker = new Map<string, UsageTracker>();\n    this.semaphoreTable = new Table(this, 'StateMachineSempahoreTable', {\n      partitionKey: {\n        name: 'LockName',\n        type: AttributeType.STRING,\n      },\n      readCapacity: props?.tableReadWriteCapacity?.readCapacity,\n      writeCapacity: props?.tableReadWriteCapacity?.writeCapacity,\n      billingMode: props?.tableReadWriteCapacity ? BillingMode.PROVISIONED : BillingMode.PAY_PER_REQUEST,\n    });\n\n  }\n  /**\n   * Generates a semaphore for a StepFunction job (or chained set of jobs) to limit parallelism across executions.\n   * @param lockName The name of the semaphore.\n   * @param limit The maximum number of concurrent executions for the given lock.\n   * @param job The job (or chained jobs) to be semaphored.\n   * @param nextState The State to go to after the semaphored job completes.\n   * @param reuseLock Explicility allow the reuse of a named lock from a previously generated job. Throws an error if a different `limit` is specified. Default: false.\n   * @param comments Adds detailed comments to lock related states. Significantly increases CloudFormation template size. Default: false.\n   * @returns A StateMachineFragment that can chained to other states in the State Machine.\n   */\n  public generateSemaphoredJob(\n    lockName: string, limit: number, job: IChainNextable, nextState: State, reuseLock?: boolean, comments?: boolean,\n  ): StateMachineFragment {\n    let lockInfo = this.semaphoreTracker.get(lockName);\n    if (lockInfo) {\n      if (reuseLock) {\n        if (lockInfo.limit != limit) {\n          throw new Error(`The reused \\`lockName\\` \"${lockName}\" was given a different \\`limit\\` than previously defined. Given: ${limit}, Previous: ${lockInfo.limit}.`);\n        } else {\n          lockInfo = { limit: lockInfo.limit, timesUsed: lockInfo.timesUsed + 1 };\n          this.semaphoreTracker.set(lockName, lockInfo);\n        }\n      } else {\n        throw new Error(`The \\`lockName\\` \"${lockName}\" was reused without explicitly allowing reuse. Set \\`reuseLock\\` to \\`true\\` if you want to reuse the lock.`);\n      }\n    } else {\n      lockInfo = { limit: limit, timesUsed: 1 };\n      this.semaphoreTracker.set(lockName, lockInfo);\n    }\n\n    const getLock = new Parallel(this, `Get ${lockName} Lock: ${lockInfo.timesUsed}`, { resultPath: JsonPath.DISCARD });\n    const acquireLock = new DynamoUpdateItem(this, `Acquire ${lockName} Lock: ${lockInfo.timesUsed}`,\n      {\n        comment: comments ? `Acquire a lock using a conditional update to DynamoDB. This update will do two things:\n          1) increment a counter for the number of held locks\n          2) add an attribute to the DynamoDB Item with a unique key for this execution and with a value of the time when the lock was Acquired\n          The Update includes a conditional expression that will fail under two circumstances:\n          1) if the maximum number of locks have already been distributed\n          2) if the current execution already owns a lock. The latter check is important to ensure the same execution doesn't increase the counter more than once\n          If either of these conditions are not met, then the task will fail with a DynamoDB.ConditionalCheckFailedException error, retry a few times, then if it is still not successful \\\n          it will move off to another branch of the workflow. If this is the first time that a given lockname has been used, there will not be a row in DynamoDB \\\n          so the update will fail with DynamoDB.AmazonDynamoDBException. In that case, this state sends the workflow to state that will create that row to initialize.\n          ` : undefined,\n        table: this.semaphoreTable,\n        key: { LockName: DynamoAttributeValue.fromString(lockName) },\n        expressionAttributeNames: {\n          '#currentlockcount': 'currentlockcount',\n          '#lockownerid.$': '$$.Execution.Id',\n        },\n        expressionAttributeValues: {\n          ':increase': DynamoAttributeValue.fromNumber(1),\n          ':limit': DynamoAttributeValue.fromNumber(limit),\n          ':lockacquiredtime': DynamoAttributeValue.fromString(JsonPath.stringAt('$$.State.EnteredTime')),\n        },\n        updateExpression: 'SET #currentlockcount = #currentlockcount + :increase, #lockownerid = :lockacquiredtime',\n        conditionExpression: 'currentlockcount <> :limit and attribute_not_exists(#lockownerid)',\n        returnValues: DynamoReturnValues.UPDATED_NEW,\n        resultPath: '$.lockinfo.acquirelock',\n      },\n    );\n    const initializeLockItem = new DynamoPutItem(this, `Initialize ${lockName} Lock Item: ${lockInfo.timesUsed}`, {\n      comment: comments ? `This state handles the case where an item hasn't been created for this lock yet. \\\n      In that case, it will insert an initial item that includes the lock name as the key and currentlockcount of 0. \\ \n      The Put to DynamoDB includes a conditonal expression to fail if the an item with that key already exists, which avoids a race condition if multiple executions start at the same time. \\ \n      There are other reasons that the previous state could fail and end up here, so this is safe in those cases too.` : undefined,\n      table: this.semaphoreTable,\n      item: {\n        LockName: DynamoAttributeValue.fromString(lockName),\n        currentlockcount: DynamoAttributeValue.fromNumber(0),\n      },\n      conditionExpression: 'LockName <> :lockname',\n      expressionAttributeValues: {\n        ':lockname': DynamoAttributeValue.fromString(lockName),\n      },\n      resultPath: JsonPath.DISCARD,\n    });\n\n    const getCurrentLockRecord = new DynamoGetItem(this, `Get Current ${lockName} Lock Record: ${lockInfo.timesUsed}`, {\n      comment: comments ? 'This state is called when the execution is unable to acquire a lock because there limit has either been exceeded or because this execution already holds a lock. \\\n      In that case, this task loads info from DDB for the current lock item so that the right decision can be made in subsequent states.': undefined,\n      table: this.semaphoreTable,\n      key: { LockName: DynamoAttributeValue.fromString(lockName) },\n      expressionAttributeNames: { '#lockownerid.$': '$$.Execution.Id' },\n      projectionExpression: [new DynamoProjectionExpression().withAttribute('#lockownerid')],\n      resultSelector: {\n        'Item.$': '$.Item',\n        'ItemString.$': 'States.JsonToString($.Item)',\n      },\n      resultPath: '$.lockinfo.currentlockitem',\n    });\n    const checkIfLockAcquired = new Choice(this, `Check if ${lockName} Lock Already Acquired: ${lockInfo.timesUsed}`, {\n      comment: comments ? `This state checks to see if the current execution already holds a lock. It can tell that by looking for Z, which will be indicative of the timestamp value. \\ \n      That will only be there in the stringified version of the data returned from DDB if this execution holds a lock.`: undefined,\n    });\n    const continueBecauseLockWasAlreadyAcquired = new Pass(this, `Continue Because ${lockName} Lock Was Already Acquired: ${lockInfo.timesUsed}`, {\n      comment: comments ? 'In this state, we have confimed that lock is already held, so we pass the original execution input into the the function that does the work.' : undefined,\n    });\n    const waitToGetLock = new Wait(this, `Wait to Get ${lockName} Lock: ${lockInfo.timesUsed}`, {\n      comment: comments ? 'If the lock indeed not been succesfully Acquired, then wait for a bit before trying again.' : undefined,\n      time: WaitTime.duration(Duration.seconds(3)),\n    });\n    acquireLock.addRetry({ errors: ['DynamoDB.AmazonDynamoDBException'], maxAttempts: 0 })\n      .addRetry({ maxAttempts: 6, backoffRate: 2 })\n      .addCatch(initializeLockItem, { errors: ['DynamoDB.AmazonDynamoDBException'], resultPath: '$.lockinfo.acquisitionerror' })\n      .addCatch(getCurrentLockRecord, { errors: ['DynamoDB.ConditionalCheckFailedException'], resultPath: '$.lockinfo.acquisitionerror' });\n    initializeLockItem.addCatch(acquireLock, { resultPath: JsonPath.DISCARD });\n    getCurrentLockRecord.next(checkIfLockAcquired);\n    checkIfLockAcquired.when(Condition.and(\n      Condition.isPresent('$.lockinfo.currentlockitem.ItemString'),\n      Condition.stringMatches('$.lockinfo.currentlockitem.ItemString', '*Z')), continueBecauseLockWasAlreadyAcquired,\n    );\n    checkIfLockAcquired.otherwise(waitToGetLock);\n    waitToGetLock.next(acquireLock);\n\n    const releaseLock = new DynamoUpdateItem(this, `Release ${lockName} Lock: ${lockInfo.timesUsed}`, {\n      table: this.semaphoreTable,\n      key: { LockName: DynamoAttributeValue.fromString(lockName) },\n      expressionAttributeNames: {\n        '#currentlockcount': 'currentlockcount',\n        '#lockownerid.$': '$$.Execution.Id',\n      },\n      expressionAttributeValues: {\n        ':decrease': DynamoAttributeValue.fromNumber(1),\n      },\n      updateExpression: 'SET #currentlockcount = #currentlockcount - :decrease REMOVE #lockownerid',\n      conditionExpression: 'attribute_exists(#lockownerid)',\n      returnValues: DynamoReturnValues.UPDATED_NEW,\n      resultPath: JsonPath.DISCARD,\n    });\n\n    releaseLock.addRetry({ errors: ['DynamoDB.ConditionalCheckFailedException'], maxAttempts: 0 })\n      .addRetry({ maxAttempts: 5, backoffRate: 1.5 })\n      .addCatch(nextState, { errors: ['DynamoDB.ConditionalCheckFailedException'], resultPath: '$.lockinfo.acquisitionerror' })\n      .next(nextState);\n    getLock.branch(acquireLock);\n    getLock.endStates.forEach(j => j.next(job));\n    job.next(releaseLock);\n    class SemaphoredJob extends StateMachineFragment {\n      public readonly startState = getLock;\n      public readonly endStates = nextState.endStates;\n    }\n    return new SemaphoredJob(this, `${lockName}${lockInfo.timesUsed}`);\n  }\n}\n"]}
package/package.json ADDED
@@ -0,0 +1,141 @@
1
+ {
2
+ "name": "@dontirun/state-machine-semaphore",
3
+ "description": "Create distributed semaphores using AWS Step Functions and Amazon DynamoDB to control concurrent invocations of contentious work.",
4
+ "repository": {
5
+ "type": "git",
6
+ "url": "https://github.com/dontirun/state-machine-semaphore.git"
7
+ },
8
+ "scripts": {
9
+ "build": "npx projen build",
10
+ "bump": "npx projen bump",
11
+ "clobber": "npx projen clobber",
12
+ "compat": "npx projen compat",
13
+ "compile": "npx projen compile",
14
+ "default": "npx projen default",
15
+ "docgen": "npx projen docgen",
16
+ "eject": "npx projen eject",
17
+ "eslint": "npx projen eslint",
18
+ "package": "npx projen package",
19
+ "package-all": "npx projen package-all",
20
+ "package:dotnet": "npx projen package:dotnet",
21
+ "package:js": "npx projen package:js",
22
+ "package:python": "npx projen package:python",
23
+ "post-compile": "npx projen post-compile",
24
+ "post-upgrade": "npx projen post-upgrade",
25
+ "pre-compile": "npx projen pre-compile",
26
+ "release": "npx projen release",
27
+ "test": "npx projen test",
28
+ "test:update": "npx projen test:update",
29
+ "test:watch": "npx projen test:watch",
30
+ "unbump": "npx projen unbump",
31
+ "upgrade": "npx projen upgrade",
32
+ "watch": "npx projen watch",
33
+ "projen": "npx projen"
34
+ },
35
+ "author": {
36
+ "name": "Arun Donti",
37
+ "email": "dontirun@gmail.com",
38
+ "organization": false
39
+ },
40
+ "devDependencies": {
41
+ "@types/jest": "^27",
42
+ "@types/node": "^14",
43
+ "@typescript-eslint/eslint-plugin": "^5",
44
+ "@typescript-eslint/parser": "^5",
45
+ "aws-cdk-lib": "2.22.0",
46
+ "cdk-nag": "^2.15.32",
47
+ "constructs": "10.0.5",
48
+ "eslint": "^8",
49
+ "eslint-import-resolver-node": "^0.3.6",
50
+ "eslint-import-resolver-typescript": "^3.4.0",
51
+ "eslint-plugin-import": "^2.26.0",
52
+ "jest": "^27",
53
+ "jest-junit": "^13",
54
+ "jsii": "^1.63.2",
55
+ "jsii-diff": "^1.63.2",
56
+ "jsii-docgen": "^7.0.61",
57
+ "jsii-pacmak": "^1.63.2",
58
+ "json-schema": "^0.4.0",
59
+ "npm-check-updates": "^15",
60
+ "projen": "^0.60.10",
61
+ "standard-version": "^9",
62
+ "ts-jest": "^27",
63
+ "typescript": "^4.7.4"
64
+ },
65
+ "peerDependencies": {
66
+ "aws-cdk-lib": "^2.22.0",
67
+ "constructs": "^10.0.5"
68
+ },
69
+ "keywords": [
70
+ "cdk",
71
+ "dynamodb",
72
+ "state machine",
73
+ "step functions"
74
+ ],
75
+ "main": "lib/index.js",
76
+ "license": "Apache-2.0",
77
+ "version": "0.0.0",
78
+ "jest": {
79
+ "testMatch": [
80
+ "<rootDir>/src/**/__tests__/**/*.ts?(x)",
81
+ "<rootDir>/(test|src)/**/*(*.)@(spec|test).ts?(x)"
82
+ ],
83
+ "clearMocks": true,
84
+ "collectCoverage": true,
85
+ "coverageReporters": [
86
+ "json",
87
+ "lcov",
88
+ "clover",
89
+ "cobertura",
90
+ "text"
91
+ ],
92
+ "coverageDirectory": "coverage",
93
+ "coveragePathIgnorePatterns": [
94
+ "/node_modules/"
95
+ ],
96
+ "testPathIgnorePatterns": [
97
+ "/node_modules/"
98
+ ],
99
+ "watchPathIgnorePatterns": [
100
+ "/node_modules/"
101
+ ],
102
+ "reporters": [
103
+ "default",
104
+ [
105
+ "jest-junit",
106
+ {
107
+ "outputDirectory": "test-reports"
108
+ }
109
+ ]
110
+ ],
111
+ "preset": "ts-jest",
112
+ "globals": {
113
+ "ts-jest": {
114
+ "tsconfig": "tsconfig.dev.json"
115
+ }
116
+ }
117
+ },
118
+ "types": "lib/index.d.ts",
119
+ "stability": "stable",
120
+ "jsii": {
121
+ "outdir": "dist",
122
+ "targets": {
123
+ "python": {
124
+ "distName": "state-machine-semaphore",
125
+ "module": "state_machine_semaphore"
126
+ },
127
+ "dotnet": {
128
+ "namespace": "Dontirun.StateMachineSemaphore",
129
+ "packageId": "Dontirun.StateMachineSemaphore"
130
+ }
131
+ },
132
+ "tsc": {
133
+ "outDir": "lib",
134
+ "rootDir": "src"
135
+ }
136
+ },
137
+ "resolutions": {
138
+ "@types/prettier": "2.6.0"
139
+ },
140
+ "//": "~~ Generated by projen. To modify, edit .projenrc.js and run \"npx projen\"."
141
+ }