Socket
Socket
Sign inDemoInstall

@almamedia-open-source/cdk-project-names

Package Overview
Dependencies
57
Maintainers
2
Versions
1
Alerts
File Explorer

Advanced tools

Install Socket

Detect and block malicious and high-risk dependencies

Install

    @almamedia-open-source/cdk-project-names

Opinionated AWS CDK utility for explicitly naming resources.


Version published
Maintainers
2
Install size
415 kB
Created

Readme

Source

Alma CDK Project Names

CDK Version Stability release

Opinionated AWS CDK utility for explicitly naming resources.

AWS CDK resource naming best practises state that you should not explicitly name resources, but instead let CDK generate resource names to avoid naming collisions and enable replacement operations during deployments.

But, having explicit resource naming with sensible conventions can:

  1. also prevent naming collisions
  2. prevent accidental/unwanted replacement operations
  3. make debugging and finding correct resources via CloudWatch or X-Ray easier

There are many valid arguments why you should aim for totally immutable infrastructure and use generated resource names. But for example, how often you want to change DynamoDB partition key on the fly for production database with existing data? You might also have a lot of resources that others (even in different AWS accounts) rely on and hence the name (or ARN) of those resources must not change suddenly! In the end, how you name (or don't name) your resources is up to you to decide; If you decide to explicitly name (all or some) resources, this utility might be for you!


Important

🚧 This tool is work-in-progress and experimental!

All @almamedia-open-source/cdk- prefixed constructs/utilities are based on existing CDK constructs/utilities we've developed & used (in production) internally at Alma Media since 2019.

Breaking changes may occur at any given time without prior warning before first v1 major is released, as we rewrite them for CDK v2 and use this opportunity to also redesign & refactor.

Feedback is most welcome, but do note that we intend to implement these new constructs/utilities and their APIs in such manner that our existing CDK v1 production workloads can easily migrate into these new @almamedia-open-source/cdk- constructs/utilities.


Installation

  1. Ensure you meet following requirements:

  2. Install peer dependency @almamedia-open-source/cdk-project-context:

    npm i -D @almamedia-open-source/cdk-project-context
    
  3. Install this tool:

    npm i -D @almamedia-open-source/cdk-project-names
    

Usage

  1. Initialize your CDK App with Project construct as documented in @almamedia-open-source/cdk-project-context:

    import { Project } from '@almamedia-open-source/cdk-project-context';
    
    // new Project instead of new App
    const project = new Project({
      name: 'my-cool-project',
      author: {
        organization: 'Acme Corp',
        name: 'Mad Scientists',
        email: 'mad.scientists@acme.example.com',
      },
      defaultRegion: 'eu-west-1', // defaults to one of: $CDK_DEFAULT_REGION, $AWS_REGION or us-east-1
      accounts: {
        dev: {
          id: '111111111111',
          config: {
            baseDomain: 'example.net',
          },
        },
        prod: {
          id: '222222222222',
          config: {
            baseDomain: 'example.com',
          },
        },
      },
    })
    
  2. Define your resource names:

    import { Name, UrlName, PathName } from '@almamedia-open-source/cdk-project-names';
    
    // somewhere inside your CDK stack:
    
    new dynamodb.Table(this, 'Table', {
      queueName: Name.it(this, 'MyTable'),
    });
    
    new events.EventBus(this, 'EventBus', {
      topicName: Name.withProject(this, 'MyEventBus'),
    });
    
    new s3.Bucket(this, 'Bucket', {
      bucketName: UrlName.globally(this, 'MyBucket'),
    });
    
    new ssm.StringParameter(this, 'Parameter', {
      parameterName: PathName.withProject(this, 'MyNamespace/MyParameter'),
      stringValue: 'Foo',
      tier: ssm.ParameterTier.ADVANCED,
    });
    
  3. Run CDK commands with (optional) environment-type (or shorthand: environment or env) CLI context flag, for example:

    npx cdk deploy --context environment=feature/foo-bar
    
  4. The resources will be named as following:

    ResourceResource Name
    TableFeatureFooBarMyTable
    EventBusMyCoolProjectFeatureFooBarMyEventBus
    Bucketacme-corp-my-cool-project-feature-foo-bar-my-bucket
    Parameter/MyCoolProject/FeatureFooBar/MyNamespace/MyParameter

High-level Naming Conventions

Case Styles

Class nameStylePurposeExample output
NamePascalCaseDefault style for naming resourcesMyResource
UrlNameparam-caseURL/DNS compatible values
(e.g. S3 bucketName)
my-resource
PathNamePascalCase
separated by
/ (slash)
Slash separated values
(e.g. SSM parameterName with hierarchies)
/MyNamespace/MyParameter

Specificity Levels

There are three different “specificity levels” of naming you may choose via available methods:

Method namePurposeExample with application environment info
itOnly base name with environment type if that is availableStagingMyResource
withProjectSame as above, but prefix with project name (recommended default)MyCoolProjectStagingMyResource
globallySame as above, but prefix with your organization name as wellAcmeCorpMyCoolProjectStagingMyResource
Shorthand syntax

Since withProject is often the most sensible default, this tool exposes the following shorthand functions for brevity:

  • name – same as Name.withProject
  • urlName – same as UrlName.withProject
  • pathName – same as PathName.withProject

Note the lowercase first letter.

import { name, urlName, pathName } from '@almamedia-open-source/cdk-project-names';

name(scope, 'MyResource');
urlName(scope, 'MyResource');
pathName(scope, 'MyResource');

Prefixes

Depending on the configuration, CDK context and method that is being used, this utility will prefix the names with some or all following values:

OrderValueControlled by
1OrganizationUsed method: globally only
2Project NameUsed method: globally or withProject
3Application EnvironmentIf provided via --context environment
4Base NameRequired string value given by user
Name Structure
StyleDefaultApplication environment info present
PascalCase default[Organization][ProjectName]Basename[Organization][ProjectName]EnvironmentBasename
param-case URL/DNS compatible[organization-][project-name-]basename[organization-][project-name-]environment-basename
PascalCase
separated by /
/[Organization/][ProjectName/]Basename/[Organization/][ProjectName/]Environment/Basename

Values in square brackets [] are optional and they are printed depending on which specificity level is used.


Resource Name Limitations

Allowed Characters

This tool does not validate for allowed characters, as they vary from service to service. Mostly you should stick to basic alphanumeric characters (a-z and 0-9), with the exception of PathName class and it's methods where you may use slash / character to describe SSM Parameter name hierarchies.

Length

Most AWS resources have resource name length limiation of around 63 characters but as always, there are exceptions such as:

The various limitations can be found via service specific API docs or somewhat nicely aggregated for most popular services in this StackOverflow answer.

If you have lenghty values in organization name, project name, environment type and/or base name you may run into problems. Due to that reason by default this utility will create an error if your resource name exceeds 63 characters.

PathName exception

The exception is PathName as it is mostly used with AWS Systems Manager Parameter Store Paremeters, which have the ARN length limit of 1011: Therefore with PathName we have decided to set the default character limit to 900. You can always configure that with maxLength: number property.

Strategies
  1. If the resource accepts longer resource names (like AWS Lambda accepts 140 characters for functionName), you may specify a custom max length as a prop:

    Name.globally(this, 'MyFunction', { maxLength: 140 });
    
  2. Select the naming method carefully:

    • If your AWS account only has single project in it, you should default to using it which results into shortest possible resource name, for example StagingMyResource. But also consider future-proofing: Will there be other projects in that AWS account in the future?

    • If your AWS account has multiple projects (e.g. microservices) in it, you should default to using withProject, which results into values such as MyCoolProjectStagingMyResource.

    • Only use globally method which prints the longest form (for example AcmeCorpMyCoolProjectStagingMyResource) for things such as S3 bucket names.

      You shouldn't really have the need to separate different organizations internally within a single AWS account. Having multiple organizations (business units or development teams etc) deploying workloads into the same AWS account suggests your AWS account organization setup is not necessarily following best practises.

  3. If the name is not important to you: Don't specify the name at all and let CDK handle it!

  4. Consider shorter base name.

  5. You should consider of course if you can somehow logically abbreviate/shorten your organization name for example. Be careful with this, as it will affect resources (i.e. perform replacement) that are already deployed!

  6. Consider trimming.

  7. Roll your own naming for that specific resource. You may want to utilize some of the methods provided by @almamedia-open-source/cdk-project-context.

Trimming

Set the maximum length and enable trimming:

Name.withProject(this, 'MyApplicationTargetGroup', {
  maxLength: 32,
  trim: true,
});

If the output value of Name.withProject is within the maxLength (32) character limit, then it returns immediately. If not (as in the above example), the following happens:

  1. It creates a hash value from the basename: E43A285509B095FCE0E474A4E9DF0A1D1D41F09D91F70D6F4873688BC07E6C2B
  2. Picks first 3 characters from the hash: E43
  3. Cuts the output value of Name.withProject to first 29 characters: MyCoolProjectStagingMyApplica
  4. Adds the first 3 characters from the hash as suffix MyCoolProjectStagingMyApplicaE43

Note that trimming happens for the whole output value of Name.withProject, which means your organization, project name and environment type prefixes might be affected as well (depending on their length).

Keywords

FAQs

Last updated on 17 Jan 2022

Did you know?

Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.

Install

Related posts

SocketSocket SOC 2 Logo

Product

  • Package Alerts
  • Integrations
  • Docs
  • Pricing
  • FAQ
  • Roadmap

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc