DynamoDB With LocalStack And The AWS CDK (Part 1)
Published: Aug 9, 2021
Last updated: Aug 9, 2021
This post is #4 of a multi-part series on the AWS CDK with TypeScript written during Melbourne lockdown #6.
This post will use the AWS TypeScript CDK to deploy table to DynamoDB on LocalStack with single-table schema design.
The code will build off the work done in the first two articles of the "Working with the TypeScript AWS CDK" series.
The final code can be found here.
Prerequisites
- Basic familiarity with AWS CDK for TypeScript.
- Read the first two posts in the "Working with the TypeScript AWS CDK" series.
- Familiary with DynamoDB. This post will not be an overview of DynamoDB development.
Getting started
We will clone the working repo from the previous blog post as a starting point:
$ git clone https://github.com/okeeffed/using-the-aws-cdk-with-localstack-and-aws-cdk-local dynamodb-with-localstack-and-the-aws-cdk $ cd building-a-cdn-with-s3-cloudfront-and-the-aws-cdk $ npm i # if install fails, remove the lockfile and try again
At this stage, we will have the app in a basic working state.
In order to create the CDN stack, we will require the following AWS CDK libraries:
@aws-cdk/core @aws-cdk/aws-dynamodb
We can install these prior to doing any work:
npm i @aws-cdk/core @aws-cdk/aws-dynamodb
The repository that we cloned already has a upgrade
script supplied. We can use this to ensure our CDK packages are at parity:
npm run upgrade
We are now at a stage to write our construct.
Writing a single-table construct
In our particular case, I am aiming to write a re-useable construct that has single-table schema design at the forefront as well as being inclusive of two secondary indexes GSI1
and GSI2
.
Let's create a folder lib/construct-single-table-dynamo
and add a barrel file and a file for the construct:
$ mkdir lib/construct-single-table-dynamo $ touch lib/construct-single-table-dynamo/construct-single-table-dynamo.ts lib/construct-single-table-dynamo/index.ts
We want our construct to do three things:
- Instantiating the table.
- Add a global secondary index
GSI1
. - Add a global secondary index
GSI2
.
As much as possible, we want to invert control to the consumer of the construct while also locking in some best practices around single-table design.
As a starter, let's prepare our lib/construct-single-table-dynamo/construct-single-table-dynamo.ts
with the following code:
import * as cdk from "@aws-cdk/core"; import * as dynamodb from "@aws-cdk/aws-dynamodb"; type SingleTableGlobalSecondaryIndex = Omit< dynamodb.GlobalSecondaryIndexProps, "indexName" | "partitionKey" | "sortKey" >; export interface ConstructSingleTableDynamoProps extends cdk.StackProps { tableProps: Omit<dynamodb.TableProps, "partitionKey" | "sortKey">; GSI1Props?: SingleTableGlobalSecondaryIndex; GSI2Props?: SingleTableGlobalSecondaryIndex; } export class ConstructSingleTableDynamo extends cdk.Construct { constructor( scope: cdk.Construct, id: string, props?: ConstructSingleTableDynamoProps ) { super(scope, id); // ... ready to add our code } }
This code will enable our construct to take three props:
tableProps
to extend our table without allow the user to alter thepartitionKey
orsortKey
.- Optional
GSI1Props
to extend the first global secondary index without being able to alter theindexName
,partitionKey
orsortKey
. - Optional
GSI2Props
to extend the second global secondary index without being able to alter theindexName
,partitionKey
orsortKey
.
We will provide sensible defaults here. You are free to alter those props if you want full control after setting sensible defaults!
Let's now go through and add our table and global secondary indexes.
Instantiating the table
We want to invoke dynamodb.Table
to create an instance of a table and pass sensible props for that table.
We will also assign that instance to a publicly available accessor dynamoTable
so that the consumer has access to the table when creating our construct within a stack.
import * as cdk from "@aws-cdk/core"; import * as dynamodb from "@aws-cdk/aws-dynamodb"; type SingleTableGlobalSecondaryIndex = Omit< dynamodb.GlobalSecondaryIndexProps, "indexName" | "partitionKey" | "sortKey" >; export interface ConstructSingleTableDynamoProps extends cdk.StackProps { tableProps: Omit<dynamodb.TableProps, "partitionKey" | "sortKey">; GSI1Props?: SingleTableGlobalSecondaryIndex; GSI2Props?: SingleTableGlobalSecondaryIndex; } export class ConstructSingleTableDynamo extends cdk.Construct { // code added here public readonly dynamoTable: dynamodb.Table; constructor( scope: cdk.Construct, id: string, props?: ConstructSingleTableDynamoProps ) { super(scope, id); // added code this.dynamoTable = new dynamodb.Table(this, `SingleTable`, { partitionKey: { name: "pk", type: dynamodb.AttributeType.STRING, }, sortKey: { name: "sk", type: dynamodb.AttributeType.STRING, }, readCapacity: 1, writeCapacity: 1, ...props?.tableProps, }); } }
Adding the first global secondary index
We can use the addGlobalSecondaryIndex
method on an instance of dynamodb.Table
to create a global secondary index.
import * as cdk from "@aws-cdk/core"; import * as dynamodb from "@aws-cdk/aws-dynamodb"; type SingleTableGlobalSecondaryIndex = Omit< dynamodb.GlobalSecondaryIndexProps, "indexName" | "partitionKey" | "sortKey" >; export interface ConstructSingleTableDynamoProps extends cdk.StackProps { tableProps: Omit<dynamodb.TableProps, "partitionKey" | "sortKey">; GSI1Props?: SingleTableGlobalSecondaryIndex; GSI2Props?: SingleTableGlobalSecondaryIndex; } export class ConstructSingleTableDynamo extends cdk.Construct { public readonly dynamoTable: dynamodb.Table; constructor( scope: cdk.Construct, id: string, props?: ConstructSingleTableDynamoProps ) { super(scope, id); this.dynamoTable = new dynamodb.Table(this, `SingleTable`, { partitionKey: { name: "pk", type: dynamodb.AttributeType.STRING, }, sortKey: { name: "sk", type: dynamodb.AttributeType.STRING, }, readCapacity: 1, writeCapacity: 1, ...props?.tableProps, }); // added code this.dynamoTable.addGlobalSecondaryIndex({ indexName: "GSI1", partitionKey: { name: "GSI1PK", type: dynamodb.AttributeType.STRING, }, sortKey: { name: "GSI1SK", type: dynamodb.AttributeType.STRING, }, readCapacity: 1, writeCapacity: 1, ...props?.GSI1Props, }); } }
Adding the second global secondary index
We can repeat similar to the above for the second index.
import * as cdk from "@aws-cdk/core"; import * as dynamodb from "@aws-cdk/aws-dynamodb"; type SingleTableGlobalSecondaryIndex = Omit< dynamodb.GlobalSecondaryIndexProps, "indexName" | "partitionKey" | "sortKey" >; export interface ConstructSingleTableDynamoProps extends cdk.StackProps { tableProps: Omit<dynamodb.TableProps, "partitionKey" | "sortKey">; GSI1Props?: SingleTableGlobalSecondaryIndex; GSI2Props?: SingleTableGlobalSecondaryIndex; } export class ConstructSingleTableDynamo extends cdk.Construct { public readonly dynamoTable: dynamodb.Table; constructor( scope: cdk.Construct, id: string, props?: ConstructSingleTableDynamoProps ) { super(scope, id); this.dynamoTable = new dynamodb.Table(this, `SingleTable`, { partitionKey: { name: "pk", type: dynamodb.AttributeType.STRING, }, sortKey: { name: "sk", type: dynamodb.AttributeType.STRING, }, readCapacity: 1, writeCapacity: 1, ...props?.tableProps, }); this.dynamoTable.addGlobalSecondaryIndex({ indexName: "GSI1", partitionKey: { name: "GSI1PK", type: dynamodb.AttributeType.STRING, }, sortKey: { name: "GSI1SK", type: dynamodb.AttributeType.STRING, }, readCapacity: 1, writeCapacity: 1, ...props?.GSI1Props, }); // added code this.dynamoTable.addGlobalSecondaryIndex({ indexName: "GSI2", partitionKey: { name: "GSI2PK", type: dynamodb.AttributeType.STRING, }, sortKey: { name: "GSI2SK", type: dynamodb.AttributeType.STRING, }, readCapacity: 1, writeCapacity: 1, ...props?.GSI2Props, }); } }
At this stage, we are now able to begin creating DynamoDB constructs in our AWS CDK stacks.
Adding the construct to our app
The repository that we cloned already creates an app with stack AwsCdkWithTypescriptFoundationsStack
.
We can see this happen in the file bin/aws-cdk-with-typescript-foundations.ts
.
What we can do then is to edit that stack code to create an instance of our constructed DynamoDB table.
Update the file lib/aws-cdk-with-typescript-foundations-stack.ts
to have the following code:
import * as cdk from "@aws-cdk/core"; import { ConstructSingleTableDynamo } from "./construct-single-table-dynamo"; export class AwsCdkWithTypescriptFoundationsStack extends cdk.Stack { constructor(scope: cdk.Construct, id: string, props?: cdk.StackProps) { super(scope, id, props); // The code that defines your stack goes here new ConstructSingleTableDynamo(this, "MyFirstTable", { tableProps: { tableName: "my-first-table", }, }); } }
This stack will construct our DynamoDB table named my-first-table
.
At this stage, we are ready to synthesize and deploy our stack to LocalStack.
Deploying to LocalStack
The docker-compose.yml
file already has what is required for us to startup LocalStack.
Run docker compose up
for LocalStack to start on Docker.
Once that is started, we can go through our usual lifecycle of the AWS CDK by synthesizing and deploying our app.
$ npm run local synth > aws-cdk-with-typescript-foundations@0.1.0 cdk > cdk "synth" Resources: MyFirstTableSingleTable60CA70C8: Type: AWS::DynamoDB::Table Properties: KeySchema: - AttributeName: pk KeyType: HASH - AttributeName: sk KeyType: RANGE AttributeDefinitions: - AttributeName: pk AttributeType: S - AttributeName: sk AttributeType: S - AttributeName: GSI1PK AttributeType: S - AttributeName: GSI1SK AttributeType: S - AttributeName: GSI2PK AttributeType: S - AttributeName: GSI2SK AttributeType: S GlobalSecondaryIndexes: - IndexName: GSI1 KeySchema: - AttributeName: GSI1PK KeyType: HASH - AttributeName: GSI1SK KeyType: RANGE Projection: ProjectionType: ALL ProvisionedThroughput: ReadCapacityUnits: 1 WriteCapacityUnits: 1 - IndexName: GSI2 KeySchema: - AttributeName: GSI2PK KeyType: HASH - AttributeName: GSI2SK KeyType: RANGE Projection: ProjectionType: ALL ProvisionedThroughput: ReadCapacityUnits: 1 WriteCapacityUnits: 1 ProvisionedThroughput: ReadCapacityUnits: 1 WriteCapacityUnits: 1 TableName: my-first-table UpdateReplacePolicy: Retain DeletionPolicy: Retain Metadata: aws:cdk:path: AwsCdkWithTypescriptFoundationsStack/MyFirstTable/SingleTable/Resource CDKMetadata: Type: AWS::CDK::Metadata Properties: Analytics: v2:deflate64:H4sIAAAAAAAAE02NTQ6CMBBGz+J+GOyC6NKEG6AXKNMxKchM0h+MaXp3EDeuvrd4eZ9BYy54Pt3sOzbk5raQBsZyT5Zm6FViCpkSDBw1B2Lon/LPu+F88ioVvgX3EbuoG7E87Pg6jANqBVHHOMV2NR1e98cpet+ELMkvjMNvNw7k/m2OAAAA Metadata: aws:cdk:path: AwsCdkWithTypescriptFoundationsStack/CDKMetadata/Default Condition: CDKMetadataAvailable Conditions: CDKMetadataAvailable: Fn::Or: - Fn::Or: - Fn::Equals: - Ref: AWS::Region - af-south-1 - Fn::Equals: - Ref: AWS::Region - ap-east-1 - Fn::Equals: - Ref: AWS::Region - ap-northeast-1 - Fn::Equals: - Ref: AWS::Region - ap-northeast-2 - Fn::Equals: - Ref: AWS::Region - ap-south-1 - Fn::Equals: - Ref: AWS::Region - ap-southeast-1 - Fn::Equals: - Ref: AWS::Region - ap-southeast-2 - Fn::Equals: - Ref: AWS::Region - ca-central-1 - Fn::Equals: - Ref: AWS::Region - cn-north-1 - Fn::Equals: - Ref: AWS::Region - cn-northwest-1 - Fn::Or: - Fn::Equals: - Ref: AWS::Region - eu-central-1 - Fn::Equals: - Ref: AWS::Region - eu-north-1 - Fn::Equals: - Ref: AWS::Region - eu-south-1 - Fn::Equals: - Ref: AWS::Region - eu-west-1 - Fn::Equals: - Ref: AWS::Region - eu-west-2 - Fn::Equals: - Ref: AWS::Region - eu-west-3 - Fn::Equals: - Ref: AWS::Region - me-south-1 - Fn::Equals: - Ref: AWS::Region - sa-east-1 - Fn::Equals: - Ref: AWS::Region - us-east-1 - Fn::Equals: - Ref: AWS::Region - us-east-2 - Fn::Or: - Fn::Equals: - Ref: AWS::Region - us-west-1 - Fn::Equals: - Ref: AWS::Region - us-west-2 $ npm run local deploy > aws-cdk-with-typescript-foundations@0.1.0 local > cdklocal "deploy" AwsCdkWithTypescriptFoundationsStack: deploying... AwsCdkWithTypescriptFoundationsStack: creating CloudFormation changeset... ✅ AwsCdkWithTypescriptFoundationsStack Stack ARN: arn:aws:cloudformation:us-east-1:000000000000:stack/AwsCdkWithTypescriptFoundationsStack/1653ab7b
At this stage we can check to see our table exists with the awslocal package
(or aws
with the LocalStack endpoint url http://localhost:4566
):
$ awslocal dynamodb describe-table --table-name my-first-table { "Table": { "AttributeDefinitions": [ { "AttributeName": "pk", "AttributeType": "S" }, { "AttributeName": "sk", "AttributeType": "S" }, { "AttributeName": "GSI1PK", "AttributeType": "S" }, { "AttributeName": "GSI1SK", "AttributeType": "S" }, { "AttributeName": "GSI2PK", "AttributeType": "S" }, { "AttributeName": "GSI2SK", "AttributeType": "S" } ], "TableName": "my-first-table", "KeySchema": [ { "AttributeName": "pk", "KeyType": "HASH" }, { "AttributeName": "sk", "KeyType": "RANGE" } ], "TableStatus": "ACTIVE", "CreationDateTime": 1628495490.523, "ProvisionedThroughput": { "LastIncreaseDateTime": 0.0, "LastDecreaseDateTime": 0.0, "NumberOfDecreasesToday": 0, "ReadCapacityUnits": 1, "WriteCapacityUnits": 1 }, "TableSizeBytes": 0, "ItemCount": 0, "TableArn": "arn:aws:dynamodb:us-east-1:000000000000:table/my-first-table", "GlobalSecondaryIndexes": [ { "IndexName": "GSI2", "KeySchema": [ { "AttributeName": "GSI2PK", "KeyType": "HASH" }, { "AttributeName": "GSI2SK", "KeyType": "RANGE" } ], "Projection": { "ProjectionType": "ALL" }, "IndexStatus": "ACTIVE", "ProvisionedThroughput": { "ReadCapacityUnits": 1, "WriteCapacityUnits": 1 }, "IndexSizeBytes": 0, "ItemCount": 0, "IndexArn": "arn:aws:dynamodb:ddblocal:000000000000:table/my-first-table/index/GSI2" }, { "IndexName": "GSI1", "KeySchema": [ { "AttributeName": "GSI1PK", "KeyType": "HASH" }, { "AttributeName": "GSI1SK", "KeyType": "RANGE" } ], "Projection": { "ProjectionType": "ALL" }, "IndexStatus": "ACTIVE", "ProvisionedThroughput": { "ReadCapacityUnits": 1, "WriteCapacityUnits": 1 }, "IndexSizeBytes": 0, "ItemCount": 0, "IndexArn": "arn:aws:dynamodb:ddblocal:000000000000:table/my-first-table/index/GSI1" } ] } }
Perfect! We have successfully constructed our DynamoDB table onto LocalStack.
Teardown
As always, we will also teardown the stack:
npm run local destroy > aws-cdk-with-typescript-foundations@0.1.0 local > cdklocal "destroy" Are you sure you want to delete: AwsCdkWithTypescriptFoundationsStack (y/n)? y AwsCdkWithTypescriptFoundationsStack: destroying... ✅ AwsCdkWithTypescriptFoundationsStack: destroyed
Summary
Today's post demonstrated how to create a single-table design DynamoDB table using the AWS TypeScript CDK. It did this by first creating a reusable construct, then followed it up by demonstrating usage of it in an app by deploying to LocalStack.
Tomorrow's post (as part two) will demonstrate this table in action with DynamoDB Toolbox.
Resources and further reading
Photo credit: benwksi
DynamoDB With LocalStack And The AWS CDK (Part 1)
Introduction