@dontirun/state-machine-semaphore 0.1.0 → 0.1.3

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/.gitattributes CHANGED
@@ -16,6 +16,7 @@
16
16
  /.projen/deps.json linguist-generated
17
17
  /.projen/files.json linguist-generated
18
18
  /.projen/tasks.json linguist-generated
19
+ /API.md linguist-generated
19
20
  /LICENSE linguist-generated
20
21
  /package.json linguist-generated
21
22
  /tsconfig.dev.json linguist-generated
package/.jsii CHANGED
@@ -2989,7 +2989,7 @@
2989
2989
  },
2990
2990
  "name": "@dontirun/state-machine-semaphore",
2991
2991
  "readme": {
2992
- "markdown": "# @dontirun/state-machine-semaphore\n\n[![npm version](https://img.shields.io/npm/v/@dontirun/state-machine-semaphore.svg)](https://img.shields.io/npm/v/@dontirun/state-machine-semaphore)\n[![PyPI version](https://img.shields.io/pypi/v/state-machine-semaphore.svg)](https://pypi.org/project/state-machine-semaphore)\n[![NuGet version](https://img.shields.io/nuget/v/Dontirun.StateMachineSemaphore)](https://www.nuget.org/packages/Dontirun.StateMachineSemaphore)\n\n[![View on Construct Hub](https://constructs.dev/badge?package=%40dontirun%2Fstate-machine-semaphore)](https://constructs.dev/packages/@dontirun/state-machine-semaphore)\n\nAn [aws-cdk](https://github.com/aws/aws-cdk) construct that enables you to use AWS Step Functions to control concurrency in your distributed system. You can use this construct to distributed state machine semaphores to control concurrent invocations of contentious work.\n\nThis construct is based off of [Justin Callison's](https://github.com/JustinCallison) example [code](https://github.com/aws-samples/aws-stepfunctions-examples/blob/main/sam/app-control-concurrency-with-dynamodb/statemachines/dynamodb-semaphore.asl.json). Make sure to check out Justin's [blogpost](https://aws.amazon.com/blogs/compute/controlling-concurrency-in-distributed-systems-using-aws-step-functions/) to learn about how the system works.\n\n## Examples\n\n### Example 1) A state machine with a controlled job\n\n<details>\n\n<summary>Click to see code</summary>\n\n```typescript\nimport { Function } from 'aws-cdk-lib/aws-lambda';\nimport { Duration, Stack, StackProps } from 'aws-cdk-lib';\nimport { StateMachine, Succeed, Wait, WaitTime } from 'aws-cdk-lib/aws-stepfunctions';\nimport { LambdaInvoke } from 'aws-cdk-lib/aws-stepfunctions-tasks';\nimport { Construct } from 'constructs';\nimport { Semaphore } from '@dontirun/state-machine-semaphore';\n\n\nexport class CdkTestStack extends Stack {\n constructor(scope: Construct, id: string, props?: StackProps) {\n super(scope, id, props);\n\n const contestedJob = new LambdaInvoke(this, 'ContestedJobPart1', {\n lambdaFunction: Function.fromFunctionName(this, 'JobFunctionPart1', 'cool-function'),\n }).next(new Wait(this, 'Wait', { time: WaitTime.duration(Duration.seconds(7)) }))\n .next(new Wait(this, 'AnotherWait', { time: WaitTime.duration(Duration.seconds(7)) }))\n .next(new Wait(this, 'YetAnotherWait', { time: WaitTime.duration(Duration.seconds(7)) }));\n\n const afterContestedJob = new Succeed(this, 'Succeed');\n\n const stateMachineFragment = new Semaphore(stack, 'Semaphore', { lockName: 'life', limit: 42, job: contestedJob, nextState: afterContestedJob });\n\n new StateMachine(this, 'StateMachine', {\n definition: stateMachineFragment,\n });\n }\n}\n```\n\n</details>\n\n\n<details>\n\n<summary>Click to see the state machine definition</summary>\n\n![Example 1 Definition](./images/Example1_Graph_Edit.png)\n</details>\n\n\n### Example 2) A state machine with multiple semaphores\n\n<details>\n\n<summary>Click to see code</summary>\n\n```typescript\nimport { Function } from 'aws-cdk-lib/aws-lambda';\nimport { Duration, Stack, StackProps } from 'aws-cdk-lib';\nimport { StateMachine, Succeed, Wait, WaitTime } from 'aws-cdk-lib/aws-stepfunctions';\nimport { LambdaInvoke } from 'aws-cdk-lib/aws-stepfunctions-tasks';\nimport { Construct } from 'constructs';\nimport { Semaphore } from '@dontirun/state-machine-semaphore';\n\n\nexport class CdkTestStack extends Stack {\n constructor(scope: Construct, id: string, props?: StackProps) {\n super(scope, id, props);\n\n const contestedJob = new LambdaInvoke(this, 'ContestedJobPart1', {\n lambdaFunction: Function.fromFunctionName(this, 'JobFunctionPart1', 'cool-function'),\n })\n const notContestedJob = new LambdaInvoke(this, 'NotContestedJob', {\n lambdaFunction: Function.fromFunctionName(this, 'NotContestedJobFunction', 'cooler-function'),\n })\n const contestedJob2 = new LambdaInvoke(this, 'ContestedJobPart2', {\n lambdaFunction: Function.fromFunctionName(this, 'JobFunctionPart2', 'coolest-function'),\n })\n const afterContestedJob2 = new Succeed(this, 'Succeed');\n\n const definition = new Semaphore(stack, 'Semaphore', { lockName: 'life', limit: 42, job: contestedJob, nextState: notContestedJob })\n .next(new Semaphore(stack, 'Semaphore2', { lockName: 'liberty', limit: 7, job: contestedJob2, nextState: afterContestedJob2 }));\n\n new StateMachine(this, 'StateMachine', {\n definition: definition,\n });\n }\n}\n```\n\n</details>\n\n<details>\n\n<summary>Click to see the state machine definition</summary>\n\n![Example 2 Definition](./images/Example2_Graph_Edit.png)\n</details>\n\n## API Reference\n\nSee [API.md](./API.md).\n\n## License\n\nThis project is licensed under the Apache-2.0 License.\n"
2992
+ "markdown": "# @dontirun/state-machine-semaphore\n\n[![npm version](https://img.shields.io/npm/v/@dontirun/state-machine-semaphore.svg)](https://img.shields.io/npm/v/@dontirun/state-machine-semaphore)\n[![PyPI version](https://img.shields.io/pypi/v/state-machine-semaphore.svg)](https://pypi.org/project/state-machine-semaphore)\n[![NuGet version](https://img.shields.io/nuget/v/Dontirun.StateMachineSemaphore)](https://www.nuget.org/packages/Dontirun.StateMachineSemaphore)\n[![Maven version](https://img.shields.io/maven-central/v/io.github.dontirun/statemachinesemaphore)](https://search.maven.org/artifact/io.github.dontirun/statemachinesemaphore)\n[![NuGet version](https://img.shields.io/nuget/v/Dontirun.StateMachineSemaphore)](https://www.nuget.org/packages/Dontirun.StateMachineSemaphore)\n\n[![View on Construct Hub](https://constructs.dev/badge?package=%40dontirun%2Fstate-machine-semaphore)](https://constructs.dev/packages/@dontirun/state-machine-semaphore)\n\nAn [aws-cdk](https://github.com/aws/aws-cdk) construct that enables you to use AWS Step Functions to control concurrency in your distributed system. You can use this construct to distributed state machine semaphores to control concurrent invocations of contentious work.\n\nThis construct is based off of [Justin Callison's](https://github.com/JustinCallison) example [code](https://github.com/aws-samples/aws-stepfunctions-examples/blob/main/sam/app-control-concurrency-with-dynamodb/statemachines/dynamodb-semaphore.asl.json). Make sure to check out Justin's [blogpost](https://aws.amazon.com/blogs/compute/controlling-concurrency-in-distributed-systems-using-aws-step-functions/) to learn about how the system works.\n\n## Examples\n\n### Example 1) A state machine with a controlled job\n\n<details>\n\n<summary>Click to see code</summary>\n\n```typescript\nimport { Function } from 'aws-cdk-lib/aws-lambda';\nimport { Duration, Stack, StackProps } from 'aws-cdk-lib';\nimport { StateMachine, Succeed, Wait, WaitTime } from 'aws-cdk-lib/aws-stepfunctions';\nimport { LambdaInvoke } from 'aws-cdk-lib/aws-stepfunctions-tasks';\nimport { Construct } from 'constructs';\nimport { Semaphore } from '@dontirun/state-machine-semaphore';\n\n\nexport class CdkTestStack extends Stack {\n constructor(scope: Construct, id: string, props?: StackProps) {\n super(scope, id, props);\n\n const contestedJob = new LambdaInvoke(this, 'ContestedJobPart1', {\n lambdaFunction: Function.fromFunctionName(this, 'JobFunctionPart1', 'cool-function'),\n }).next(new Wait(this, 'Wait', { time: WaitTime.duration(Duration.seconds(7)) }))\n .next(new Wait(this, 'AnotherWait', { time: WaitTime.duration(Duration.seconds(7)) }))\n .next(new Wait(this, 'YetAnotherWait', { time: WaitTime.duration(Duration.seconds(7)) }));\n\n const afterContestedJob = new Succeed(this, 'Succeed');\n\n const stateMachineFragment = new Semaphore(stack, 'Semaphore', { lockName: 'life', limit: 42, job: contestedJob, nextState: afterContestedJob });\n\n new StateMachine(this, 'StateMachine', {\n definition: stateMachineFragment,\n });\n }\n}\n```\n\n</details>\n\n\n<details>\n\n<summary>Click to see the state machine definition</summary>\n\n![Example 1 Definition](./images/Example1_Graph_Edit.png)\n</details>\n\n\n### Example 2) A state machine with multiple semaphores\n\n<details>\n\n<summary>Click to see code</summary>\n\n```typescript\nimport { Function } from 'aws-cdk-lib/aws-lambda';\nimport { Duration, Stack, StackProps } from 'aws-cdk-lib';\nimport { StateMachine, Succeed, Wait, WaitTime } from 'aws-cdk-lib/aws-stepfunctions';\nimport { LambdaInvoke } from 'aws-cdk-lib/aws-stepfunctions-tasks';\nimport { Construct } from 'constructs';\nimport { Semaphore } from '@dontirun/state-machine-semaphore';\n\n\nexport class CdkTestStack extends Stack {\n constructor(scope: Construct, id: string, props?: StackProps) {\n super(scope, id, props);\n\n const contestedJob = new LambdaInvoke(this, 'ContestedJobPart1', {\n lambdaFunction: Function.fromFunctionName(this, 'JobFunctionPart1', 'cool-function'),\n })\n const notContestedJob = new LambdaInvoke(this, 'NotContestedJob', {\n lambdaFunction: Function.fromFunctionName(this, 'NotContestedJobFunction', 'cooler-function'),\n })\n const contestedJob2 = new LambdaInvoke(this, 'ContestedJobPart2', {\n lambdaFunction: Function.fromFunctionName(this, 'JobFunctionPart2', 'coolest-function'),\n })\n const afterContestedJob2 = new Succeed(this, 'Succeed');\n\n const definition = new Semaphore(stack, 'Semaphore', { lockName: 'life', limit: 42, job: contestedJob, nextState: notContestedJob })\n .next(new Semaphore(stack, 'Semaphore2', { lockName: 'liberty', limit: 7, job: contestedJob2, nextState: afterContestedJob2 }));\n\n new StateMachine(this, 'StateMachine', {\n definition: definition,\n });\n }\n}\n```\n\n</details>\n\n<details>\n\n<summary>Click to see the state machine definition</summary>\n\n![Example 2 Definition](./images/Example2_Graph_Edit.png)\n</details>\n\n## API Reference\n\nSee [API.md](./API.md).\n\n## License\n\nThis project is licensed under the Apache-2.0 License.\n"
2993
2993
  },
2994
2994
  "repository": {
2995
2995
  "type": "git",
@@ -3001,6 +3001,13 @@
3001
3001
  "namespace": "Dontirun.StateMachineSemaphore",
3002
3002
  "packageId": "Dontirun.StateMachineSemaphore"
3003
3003
  },
3004
+ "java": {
3005
+ "maven": {
3006
+ "artifactId": "statemachinesemaphore",
3007
+ "groupId": "io.github.dontirun"
3008
+ },
3009
+ "package": "io.github.dontirun.statemachinesemaphore"
3010
+ },
3004
3011
  "js": {
3005
3012
  "npm": "@dontirun/state-machine-semaphore"
3006
3013
  },
@@ -3298,6 +3305,6 @@
3298
3305
  "symbolId": "src/index:TableReadWriteCapacity"
3299
3306
  }
3300
3307
  },
3301
- "version": "0.1.0",
3302
- "fingerprint": "y6oYFeIJXupfNQnaMCZlL8ffdTZGJuqJ0qGm58Mz1gE="
3308
+ "version": "0.1.3",
3309
+ "fingerprint": "1rn+UNba0giHEMzM34GntGKWZuje7LpAeLOTmek1fyU="
3303
3310
  }
package/README.md CHANGED
@@ -3,6 +3,8 @@
3
3
  [![npm version](https://img.shields.io/npm/v/@dontirun/state-machine-semaphore.svg)](https://img.shields.io/npm/v/@dontirun/state-machine-semaphore)
4
4
  [![PyPI version](https://img.shields.io/pypi/v/state-machine-semaphore.svg)](https://pypi.org/project/state-machine-semaphore)
5
5
  [![NuGet version](https://img.shields.io/nuget/v/Dontirun.StateMachineSemaphore)](https://www.nuget.org/packages/Dontirun.StateMachineSemaphore)
6
+ [![Maven version](https://img.shields.io/maven-central/v/io.github.dontirun/statemachinesemaphore)](https://search.maven.org/artifact/io.github.dontirun/statemachinesemaphore)
7
+ [![NuGet version](https://img.shields.io/nuget/v/Dontirun.StateMachineSemaphore)](https://www.nuget.org/packages/Dontirun.StateMachineSemaphore)
6
8
 
7
9
  [![View on Construct Hub](https://constructs.dev/badge?package=%40dontirun%2Fstate-machine-semaphore)](https://constructs.dev/packages/@dontirun/state-machine-semaphore)
8
10
 
package/lib/index.js CHANGED
@@ -155,6 +155,9 @@ class Semaphore extends aws_stepfunctions_1.StateMachineFragment {
155
155
  ensureTable(props) {
156
156
  const existing = aws_cdk_lib_1.Stack.of(this).node.tryFindChild(this.tableName);
157
157
  if (existing) {
158
+ if (props.tableReadWriteCapacity) {
159
+ throw new Error('`tableReadWriteCapacity` can only be specified on the first instance of the `Semaphore` construct in each stack.');
160
+ }
158
161
  // Just assume this is true
159
162
  return existing;
160
163
  }
@@ -173,9 +176,9 @@ class Semaphore extends aws_stepfunctions_1.StateMachineFragment {
173
176
  }
174
177
  exports.Semaphore = Semaphore;
175
178
  _a = JSII_RTTI_SYMBOL_1;
176
- Semaphore[_a] = { fqn: "@dontirun/state-machine-semaphore.Semaphore", version: "0.1.0" };
179
+ Semaphore[_a] = { fqn: "@dontirun/state-machine-semaphore.Semaphore", version: "0.1.3" };
177
180
  /**
178
181
  * The names and associated concurrency limits and number of uses of the sempahores.
179
182
  */
180
183
  Semaphore.semaphoreTracker = new Map();
181
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["import { Duration, Names, Stack } 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/**\n * Interface for creating a Semaphore\n */\nexport interface SemaphoreProps {\n  /**\n   * The name of the semaphore.\n   */\n  readonly lockName: string;\n  /**\n   * The maximum number of concurrent executions for the given lock.\n   */\n  readonly limit: number;\n  /**\n   * The job (or chained jobs) to be semaphored.\n   */\n  readonly job: IChainNextable;\n  /**\n   * The State to go to after the semaphored job completes.\n   */\n  readonly nextState: State;\n  /**\n   *  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   */\n  readonly reuseLock?: boolean;\n  /**\n   * Add detailed comments to lock related states. Significantly increases CloudFormation template size. Default: false.\n   */\n  readonly comments?: boolean;\n  /**\n   * Optionally set the DynamoDB table to have a specific read/write capacity with PROVISIONED billing.\n   * Note: This property can only be set on the first instantiation of a `Semaphore` per stack\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/**\n * Generates a semaphore for a StepFunction job (or chained set of jobs) to limit parallelism across executions.\n */\nexport class Semaphore extends StateMachineFragment {\n\n  /**\n   * The names and associated concurrency limits and number of uses of the sempahores.\n   */\n  private static semaphoreTracker = new Map<string, Map<string, UsageTracker>>();\n\n  /**\n   * The DynamoDB table used to store semaphores.\n   */\n  private semaphoreTable: Table;\n  /**\n   * The DynamoDB table used to store semaphores.\n   */\n  private tableName = 'StateMachineSempahoreTable920751a65a584e8ab7583460f6db686a';\n\n  public readonly startState: State;\n  public readonly endStates: INextable[];\n\n\n  constructor(scope: Construct, id: string, props: SemaphoreProps) {\n    super(scope, id);\n    const stackTracker = this.setUpMap();\n    this.semaphoreTable = this.ensureTable(props);\n    let lockInfo = stackTracker.get(props.lockName);\n    if (lockInfo) {\n      if (props.reuseLock) {\n        if (lockInfo.limit != props.limit) {\n          throw new Error(`The reused \\`lockName\\` \"${props.lockName}\" was given a different \\`limit\\` than previously defined. Given: ${props.limit}, Previous: ${lockInfo.limit}.`);\n        } else {\n          lockInfo = { limit: lockInfo.limit, timesUsed: lockInfo.timesUsed + 1 };\n          stackTracker.set(props.lockName, lockInfo);\n        }\n      } else {\n        throw new Error(`The \\`lockName\\` \"${props.lockName}\" was reused without explicitly allowing reuse. Set \\`reuseLock\\` to \\`true\\` if you want to reuse the lock.`);\n      }\n    } else {\n      lockInfo = { limit: props.limit, timesUsed: 1 };\n      stackTracker.set(props.lockName, lockInfo);\n    }\n\n    const getLock = new Parallel(this, `Get ${props.lockName} Lock: ${lockInfo.timesUsed}`, { resultPath: JsonPath.DISCARD });\n    const acquireLock = new DynamoUpdateItem(this, `Acquire ${props.lockName} Lock: ${lockInfo.timesUsed}`,\n      {\n        comment: props.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(props.lockName) },\n        expressionAttributeNames: {\n          '#currentlockcount': 'currentlockcount',\n          '#lockownerid.$': '$$.Execution.Id',\n        },\n        expressionAttributeValues: {\n          ':increase': DynamoAttributeValue.fromNumber(1),\n          ':limit': DynamoAttributeValue.fromNumber(props.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 ${props.lockName} Lock Item: ${lockInfo.timesUsed}`, {\n      comment: props.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(props.lockName),\n        currentlockcount: DynamoAttributeValue.fromNumber(0),\n      },\n      conditionExpression: 'LockName <> :lockname',\n      expressionAttributeValues: {\n        ':lockname': DynamoAttributeValue.fromString(props.lockName),\n      },\n      resultPath: JsonPath.DISCARD,\n    });\n\n    const getCurrentLockRecord = new DynamoGetItem(this, `Get Current ${props.lockName} Lock Record: ${lockInfo.timesUsed}`, {\n      comment: props.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(props.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 ${props.lockName} Lock Already Acquired: ${lockInfo.timesUsed}`, {\n      comment: props.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 ${props.lockName} Lock Was Already Acquired: ${lockInfo.timesUsed}`, {\n      comment: props.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 ${props.lockName} Lock: ${lockInfo.timesUsed}`, {\n      comment: props.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 ${props.lockName} Lock: ${lockInfo.timesUsed}`, {\n      table: this.semaphoreTable,\n      key: { LockName: DynamoAttributeValue.fromString(props.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(props.nextState, { errors: ['DynamoDB.ConditionalCheckFailedException'], resultPath: '$.lockinfo.acquisitionerror' })\n      .next(props.nextState);\n    getLock.branch(acquireLock);\n    getLock.endStates.forEach(j => j.next(props.job));\n    props.job.next(releaseLock);\n\n    this.startState = getLock;\n    this.endStates = props.nextState.endStates;\n  }\n\n  private setUpMap(): Map<string, UsageTracker> {\n    const stackId = Names.uniqueId(Stack.of(this));\n    const existing = Stack.of(this).node.tryFindChild(this.tableName);\n    if (existing) {\n      return <Map<string, UsageTracker>>(Semaphore.semaphoreTracker.get(stackId));\n    } else {\n      const m = new Map<string, UsageTracker>();\n      Semaphore.semaphoreTracker.set(stackId, m);\n      return m;\n    }\n  }\n\n  private ensureTable(props: SemaphoreProps): Table {\n    const existing = Stack.of(this).node.tryFindChild(this.tableName);\n    if (existing) {\n      // Just assume this is true\n      return existing as Table;\n    } else {\n      return new Table(Stack.of(this), this.tableName, {\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"]}
184
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["import { Duration, Names, Stack } 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/**\n * Interface for creating a Semaphore\n */\nexport interface SemaphoreProps {\n  /**\n   * The name of the semaphore.\n   */\n  readonly lockName: string;\n  /**\n   * The maximum number of concurrent executions for the given lock.\n   */\n  readonly limit: number;\n  /**\n   * The job (or chained jobs) to be semaphored.\n   */\n  readonly job: IChainNextable;\n  /**\n   * The State to go to after the semaphored job completes.\n   */\n  readonly nextState: State;\n  /**\n   *  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   */\n  readonly reuseLock?: boolean;\n  /**\n   * Add detailed comments to lock related states. Significantly increases CloudFormation template size. Default: false.\n   */\n  readonly comments?: boolean;\n  /**\n   * Optionally set the DynamoDB table to have a specific read/write capacity with PROVISIONED billing.\n   * Note: This property can only be set on the first instantiation of a `Semaphore` per stack\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/**\n * Generates a semaphore for a StepFunction job (or chained set of jobs) to limit parallelism across executions.\n */\nexport class Semaphore extends StateMachineFragment {\n\n  /**\n   * The names and associated concurrency limits and number of uses of the sempahores.\n   */\n  private static semaphoreTracker = new Map<string, Map<string, UsageTracker>>();\n\n  /**\n   * The DynamoDB table used to store semaphores.\n   */\n  private semaphoreTable: Table;\n  /**\n   * The DynamoDB table used to store semaphores.\n   */\n  private tableName = 'StateMachineSempahoreTable920751a65a584e8ab7583460f6db686a';\n\n  public readonly startState: State;\n  public readonly endStates: INextable[];\n\n\n  constructor(scope: Construct, id: string, props: SemaphoreProps) {\n    super(scope, id);\n    const stackTracker = this.setUpMap();\n    this.semaphoreTable = this.ensureTable(props);\n    let lockInfo = stackTracker.get(props.lockName);\n    if (lockInfo) {\n      if (props.reuseLock) {\n        if (lockInfo.limit != props.limit) {\n          throw new Error(`The reused \\`lockName\\` \"${props.lockName}\" was given a different \\`limit\\` than previously defined. Given: ${props.limit}, Previous: ${lockInfo.limit}.`);\n        } else {\n          lockInfo = { limit: lockInfo.limit, timesUsed: lockInfo.timesUsed + 1 };\n          stackTracker.set(props.lockName, lockInfo);\n        }\n      } else {\n        throw new Error(`The \\`lockName\\` \"${props.lockName}\" was reused without explicitly allowing reuse. Set \\`reuseLock\\` to \\`true\\` if you want to reuse the lock.`);\n      }\n    } else {\n      lockInfo = { limit: props.limit, timesUsed: 1 };\n      stackTracker.set(props.lockName, lockInfo);\n    }\n\n    const getLock = new Parallel(this, `Get ${props.lockName} Lock: ${lockInfo.timesUsed}`, { resultPath: JsonPath.DISCARD });\n    const acquireLock = new DynamoUpdateItem(this, `Acquire ${props.lockName} Lock: ${lockInfo.timesUsed}`,\n      {\n        comment: props.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(props.lockName) },\n        expressionAttributeNames: {\n          '#currentlockcount': 'currentlockcount',\n          '#lockownerid.$': '$$.Execution.Id',\n        },\n        expressionAttributeValues: {\n          ':increase': DynamoAttributeValue.fromNumber(1),\n          ':limit': DynamoAttributeValue.fromNumber(props.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 ${props.lockName} Lock Item: ${lockInfo.timesUsed}`, {\n      comment: props.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(props.lockName),\n        currentlockcount: DynamoAttributeValue.fromNumber(0),\n      },\n      conditionExpression: 'LockName <> :lockname',\n      expressionAttributeValues: {\n        ':lockname': DynamoAttributeValue.fromString(props.lockName),\n      },\n      resultPath: JsonPath.DISCARD,\n    });\n\n    const getCurrentLockRecord = new DynamoGetItem(this, `Get Current ${props.lockName} Lock Record: ${lockInfo.timesUsed}`, {\n      comment: props.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(props.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 ${props.lockName} Lock Already Acquired: ${lockInfo.timesUsed}`, {\n      comment: props.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 ${props.lockName} Lock Was Already Acquired: ${lockInfo.timesUsed}`, {\n      comment: props.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 ${props.lockName} Lock: ${lockInfo.timesUsed}`, {\n      comment: props.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 ${props.lockName} Lock: ${lockInfo.timesUsed}`, {\n      table: this.semaphoreTable,\n      key: { LockName: DynamoAttributeValue.fromString(props.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(props.nextState, { errors: ['DynamoDB.ConditionalCheckFailedException'], resultPath: '$.lockinfo.acquisitionerror' })\n      .next(props.nextState);\n    getLock.branch(acquireLock);\n    getLock.endStates.forEach(j => j.next(props.job));\n    props.job.next(releaseLock);\n\n    this.startState = getLock;\n    this.endStates = props.nextState.endStates;\n  }\n\n  private setUpMap(): Map<string, UsageTracker> {\n    const stackId = Names.uniqueId(Stack.of(this));\n    const existing = Stack.of(this).node.tryFindChild(this.tableName);\n    if (existing) {\n      return <Map<string, UsageTracker>>(Semaphore.semaphoreTracker.get(stackId));\n    } else {\n      const m = new Map<string, UsageTracker>();\n      Semaphore.semaphoreTracker.set(stackId, m);\n      return m;\n    }\n  }\n\n  private ensureTable(props: SemaphoreProps): Table {\n    const existing = Stack.of(this).node.tryFindChild(this.tableName);\n    if (existing) {\n      if (props.tableReadWriteCapacity) {\n        throw new Error('`tableReadWriteCapacity` can only be specified on the first instance of the `Semaphore` construct in each stack.');\n      }\n      // Just assume this is true\n      return existing as Table;\n    } else {\n      return new Table(Stack.of(this), this.tableName, {\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"]}
package/package.json CHANGED
@@ -18,6 +18,7 @@
18
18
  "package": "npx projen package",
19
19
  "package-all": "npx projen package-all",
20
20
  "package:dotnet": "npx projen package:dotnet",
21
+ "package:java": "npx projen package:java",
21
22
  "package:js": "npx projen package:js",
22
23
  "package:python": "npx projen package:python",
23
24
  "post-compile": "npx projen post-compile",
@@ -53,11 +54,11 @@
53
54
  "jest-junit": "^13",
54
55
  "jsii": "^1.63.2",
55
56
  "jsii-diff": "^1.63.2",
56
- "jsii-docgen": "^7.0.61",
57
+ "jsii-docgen": "^7.0.65",
57
58
  "jsii-pacmak": "^1.63.2",
58
59
  "json-schema": "^0.4.0",
59
60
  "npm-check-updates": "^15",
60
- "projen": "^0.60.10",
61
+ "projen": "^0.61.2",
61
62
  "standard-version": "^9",
62
63
  "ts-jest": "^27",
63
64
  "typescript": "^4.7.4"
@@ -74,7 +75,7 @@
74
75
  ],
75
76
  "main": "lib/index.js",
76
77
  "license": "Apache-2.0",
77
- "version": "0.1.0",
78
+ "version": "0.1.3",
78
79
  "jest": {
79
80
  "testMatch": [
80
81
  "<rootDir>/src/**/__tests__/**/*.ts?(x)",
@@ -120,6 +121,13 @@
120
121
  "jsii": {
121
122
  "outdir": "dist",
122
123
  "targets": {
124
+ "java": {
125
+ "package": "io.github.dontirun.statemachinesemaphore",
126
+ "maven": {
127
+ "groupId": "io.github.dontirun",
128
+ "artifactId": "statemachinesemaphore"
129
+ }
130
+ },
123
131
  "python": {
124
132
  "distName": "state-machine-semaphore",
125
133
  "module": "state_machine_semaphore"