CDK Organizer for CDK Terraform TypeScript Library
This library contains the core features to handle CDK projects including:
- Configuration Resolver
- Stack Groups Loader
- Naming Utils
Full documentation: https://cdk-organizer.github.io/
Installation
Project Structure
To apply the pattern purposed in this library for CDK projects, the following structure is required:
CDK Start Script
The content of the bin/main.ts
file should be as follows:
#!/usr/bin/env node
import { App } from 'cdktf';
import { StackGroupLoader } from '@awslv/cdk-organizer-core';
const app = new App();
const loader = new StackGroupLoader(app);
loader.synth().then(() => {
app.synth();
});
Context Variables
The following context variables are required to the CDK Organizer to work properly:
env
region
The variables can be set in the cdk.json
file:
In the
cdktf
CLI the context variables cannot be passed as arguments, so they need to be set in thecdk.json
file. https://github.com/hashicorp/terraform-cdk/issues/2019 Theenv
variable can also be set as an environment variableCDK_ENV
.
Stack Structure
the stack class needs to inherit from class import { Stack } from '@awslv/cdkft-organizer'
for Terraform CDK.
Terraform CDK
import { Stack, StackGroup } from '@awslv/cdktf-organizer';
import { AwsProvider } from '@cdktf/provider-aws/provider';
import { S3Bucket } from '@cdktf/provider-aws/lib/s3-bucket';
import { Construct } from 'constructs';
export type S3StackProps = {
bucketName: string;
};
export class S3Stack extends Stack {
public bucket: S3Bucket;
constructor(
stackGroup: StackGroup,
scope: Construct,
id: string,
props: S3StackProps
) {
super(stackGroup, scope, id);
new AwsProvider(this, 'Aws', {
region: this.config['region'] as string,
});
this.bucket = new S3Bucket(this, 'Bucket', {
bucket: this.getBucketName(props.bucketName),
});
}
}
Using S3 Backend
To use S3 terraform backend, the following resources are required:
- S3 Bucket
- DynamoDB Table
Add the following object to the environment configuration file:
Stack Group Structure
Create a stacks
folder in the root of the project and structure it as follows:
The stack groups files follow this pattern:
Terraform
import { StackGroup } from '@awslv/cdktf-organizer';
import { BaseConfig } from '@awslv/cdk-organizer-core';
import { S3Bucket } from '@cdktf/provider-aws/lib/s3-bucket';
import { S3Stack } from '../../templates/s3-stack';
type StorageStackGroupProps = {
bucketName: string;
} & BaseConfig;
export class StorageStackGroup extends StackGroup<StorageStackGroupProps> {
public bucket!: S3Bucket;
override async loadStacks(): Promise<void> {
this.bucket = new S3Stack(this, this.app, this.getStackName('bucket'), {
bucketName: this.data.bucketName,
}).bucket;
}
}
If you are using
typescript
version <4.3 you need to remove theoverride
keyword.
Using Stack Attributes from Other Stack Groups
In some cases, you may want to use the attributes of another stack group. For example, refer the DNS Hosted Zone created by a shared stack group.
To resolve the group use the this.resolveGroup
function in the stack group class, like the example below:
import { StackGroup } from '@awslv/aws-cdk-organizer';
import { RoleStack } from '../../templates/role-stack';
import { StorageStackGroup } from '../storage/stacks';
export class IamStackGroup extends StackGroup {
override async loadStacks(): Promise<void> {
const storageGroup = await this.resolveGroup(StorageStackGroup);
new RoleStack(this, this.app, this.getStackName('role'), {
bucket: storageGroup.bucket,
});
}
}
If you are using
typescript
version <4.3 you need to remove theoverride
keyword.
The function getStackName
generates the stack name based on following pattern.
Pattern: {module_path}-{name}-{region}-{env}
Consider the following example:
module_path: myproject.myapp.www
name: spa
region: us-east-1
env: dev
Stack name will be:
myproject-myapp-www-spa-us-east-1-dev
.
Config Structure
Create a config
folder in the root of the project and structure it as follows:
.
+-- cdk.json
+-- bin/main.ts
+-- config/
| +-- <env>/
| +-- config.yaml
| +-- <region>/
| +-- config.yaml
| +-- <groupName>/
| +-- config.yaml
The first two levels are reserved to the environment name and the region name, the next levels needs to match the stack group structure.
Example:
.
+-- cdk.json
+-- bin/main.ts
+-- config/
| +-- config.yaml
| +-- dev/
| +-- config.yaml
| +-- us-east-1/
| +-- config.yaml
| +-- app1/
| +-- config.yaml
+-- stacks/
| +-- app1/
| +-- stacks.ts
+-- templates/
| +-- stacks/