@inizioevoke/aws-cdk-helpers 1.0.1 → 1.0.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/dist/constructs/certificatemanager.d.ts +13 -0
- package/dist/constructs/certificatemanager.js +19 -0
- package/dist/constructs/cloudfront.d.ts +44 -0
- package/dist/constructs/cloudfront.js +78 -0
- package/dist/constructs/codebuild.d.ts +26 -0
- package/dist/constructs/codebuild.js +76 -0
- package/dist/constructs/codepipeline.d.ts +30 -0
- package/dist/constructs/codepipeline.js +58 -0
- package/dist/constructs/route53.d.ts +14 -0
- package/dist/constructs/route53.js +24 -0
- package/dist/constructs/s3.d.ts +12 -0
- package/dist/constructs/s3.js +17 -0
- package/dist/constructs/waf.d.ts +21 -0
- package/dist/constructs/waf.js +86 -0
- package/dist/lib/utils.d.ts +5 -0
- package/dist/lib/utils.js +19 -0
- package/dist/stacks/web-static-serverless/codepipeline-stack.d.ts +17 -0
- package/dist/stacks/web-static-serverless/codepipeline-stack.js +28 -0
- package/dist/stacks/web-static-serverless/web-global-stack.d.ts +33 -0
- package/dist/stacks/web-static-serverless/web-global-stack.js +85 -0
- package/dist/stages/str8r-clm-stage.d.ts +17 -0
- package/dist/stages/str8r-clm-stage.js +39 -0
- package/dist/stages/web-static-serverless-stage.d.ts +17 -0
- package/dist/stages/web-static-serverless-stage.js +39 -0
- package/package.json +11 -1
- package/{constructs → src/constructs}/certificatemanager.ts +1 -1
- package/{constructs → src/constructs}/cloudfront.ts +4 -2
- package/{constructs → src/constructs}/codebuild.ts +1 -1
- package/{constructs → src/constructs}/codepipeline.ts +2 -2
- package/{constructs → src/constructs}/s3.ts +1 -1
- package/{constructs → src/constructs}/waf.ts +2 -2
- package/{stacks → src/stacks}/web-static-serverless/codepipeline-stack.ts +2 -2
- package/{stacks → src/stacks}/web-static-serverless/web-global-stack.ts +6 -6
- package/{stages → src/stages}/str8r-clm-stage.ts +3 -3
- package/{stages → src/stages}/web-static-serverless-stage.ts +3 -3
- package/src/templates/cffn-default-doc-basic-auth.js +148 -0
- package/src/templates/cffn-default-doc.js +125 -0
- package/tasks/clean.js +7 -0
- package/tasks/copy-templates.js +13 -0
- package/tsconfig.json +4 -1
- /package/{templates → dist/templates}/cffn-default-doc-basic-auth.js +0 -0
- /package/{templates → dist/templates}/cffn-default-doc.js +0 -0
- /package/{constructs → src/constructs}/route53.ts +0 -0
- /package/{lib → src/lib}/types.d.ts +0 -0
- /package/{lib → src/lib}/utils.ts +0 -0
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import { Construct } from 'constructs';
|
|
2
|
+
import { Certificate } from "aws-cdk-lib/aws-certificatemanager";
|
|
3
|
+
import { IHostedZone } from "aws-cdk-lib/aws-route53";
|
|
4
|
+
export interface SSLCertificateProps {
|
|
5
|
+
hostedZone: IHostedZone;
|
|
6
|
+
domainName: string;
|
|
7
|
+
destroy?: boolean;
|
|
8
|
+
tags?: Record<string, string>;
|
|
9
|
+
}
|
|
10
|
+
export declare class SSLCertificate extends Construct {
|
|
11
|
+
readonly certificate: Certificate;
|
|
12
|
+
constructor(scope: Construct, id: string, props: SSLCertificateProps);
|
|
13
|
+
}
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import { RemovalPolicy } from "aws-cdk-lib";
|
|
2
|
+
import { Construct } from 'constructs';
|
|
3
|
+
import { Certificate, CertificateValidation } from "aws-cdk-lib/aws-certificatemanager";
|
|
4
|
+
import { tagResource } from "../lib/utils.js";
|
|
5
|
+
export class SSLCertificate extends Construct {
|
|
6
|
+
certificate;
|
|
7
|
+
constructor(scope, id, props) {
|
|
8
|
+
super(scope, id);
|
|
9
|
+
this.certificate = new Certificate(this, `Certificate-${props.domainName}`, {
|
|
10
|
+
domainName: props.domainName,
|
|
11
|
+
validation: CertificateValidation.fromDns(props.hostedZone),
|
|
12
|
+
});
|
|
13
|
+
tagResource(this.certificate, props.tags);
|
|
14
|
+
if (typeof props.destroy == 'boolean') {
|
|
15
|
+
this.certificate.applyRemovalPolicy(props.destroy ? RemovalPolicy.DESTROY : RemovalPolicy.RETAIN);
|
|
16
|
+
}
|
|
17
|
+
}
|
|
18
|
+
}
|
|
19
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2VydGlmaWNhdGVtYW5hZ2VyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2NvbnN0cnVjdHMvY2VydGlmaWNhdGVtYW5hZ2VyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxhQUFhLENBQUM7QUFDNUMsT0FBTyxFQUFFLFNBQVMsRUFBRSxNQUFNLFlBQVksQ0FBQztBQUN2QyxPQUFPLEVBQUUsV0FBVyxFQUFFLHFCQUFxQixFQUFFLE1BQU0sb0NBQW9DLENBQUM7QUFFeEYsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBUzlDLE1BQU0sT0FBTyxjQUFlLFNBQVEsU0FBUztJQUMzQixXQUFXLENBQWM7SUFFekMsWUFBWSxLQUFnQixFQUFFLEVBQVUsRUFBRSxLQUEwQjtRQUNsRSxLQUFLLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxDQUFDO1FBRWpCLElBQUksQ0FBQyxXQUFXLEdBQUcsSUFBSSxXQUFXLENBQUMsSUFBSSxFQUFFLGVBQWUsS0FBSyxDQUFDLFVBQVUsRUFBRSxFQUFFO1lBQzFFLFVBQVUsRUFBRSxLQUFLLENBQUMsVUFBVTtZQUM1QixVQUFVLEVBQUUscUJBQXFCLENBQUMsT0FBTyxDQUFDLEtBQUssQ0FBQyxVQUFVLENBQUM7U0FDNUQsQ0FBQyxDQUFDO1FBQ0gsV0FBVyxDQUFDLElBQUksQ0FBQyxXQUFXLEVBQUUsS0FBSyxDQUFDLElBQUksQ0FBQyxDQUFDO1FBQzFDLElBQUksT0FBTyxLQUFLLENBQUMsT0FBTyxJQUFJLFNBQVMsRUFBRSxDQUFDO1lBQ3RDLElBQUksQ0FBQyxXQUFXLENBQUMsa0JBQWtCLENBQUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsYUFBYSxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsYUFBYSxDQUFDLE1BQU0sQ0FBQyxDQUFDO1FBQ3BHLENBQUM7SUFDSCxDQUFDO0NBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBSZW1vdmFsUG9saWN5IH0gZnJvbSBcImF3cy1jZGstbGliXCI7XG5pbXBvcnQgeyBDb25zdHJ1Y3QgfSBmcm9tICdjb25zdHJ1Y3RzJztcbmltcG9ydCB7IENlcnRpZmljYXRlLCBDZXJ0aWZpY2F0ZVZhbGlkYXRpb24gfSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWNlcnRpZmljYXRlbWFuYWdlclwiO1xuaW1wb3J0IHsgSUhvc3RlZFpvbmUgfSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLXJvdXRlNTNcIjtcbmltcG9ydCB7IHRhZ1Jlc291cmNlIH0gZnJvbSBcIi4uL2xpYi91dGlscy5qc1wiO1xuXG5leHBvcnQgaW50ZXJmYWNlIFNTTENlcnRpZmljYXRlUHJvcHMge1xuICBob3N0ZWRab25lOiBJSG9zdGVkWm9uZTtcbiAgZG9tYWluTmFtZTogc3RyaW5nO1xuICBkZXN0cm95PzogYm9vbGVhbjtcbiAgdGFncz86IFJlY29yZDxzdHJpbmcsIHN0cmluZz47XG59XG5cbmV4cG9ydCBjbGFzcyBTU0xDZXJ0aWZpY2F0ZSBleHRlbmRzIENvbnN0cnVjdCB7XG4gIHB1YmxpYyByZWFkb25seSBjZXJ0aWZpY2F0ZTogQ2VydGlmaWNhdGU7XG5cbiAgY29uc3RydWN0b3Ioc2NvcGU6IENvbnN0cnVjdCwgaWQ6IHN0cmluZywgcHJvcHM6IFNTTENlcnRpZmljYXRlUHJvcHMpIHtcbiAgICBzdXBlcihzY29wZSwgaWQpO1xuXG4gICAgdGhpcy5jZXJ0aWZpY2F0ZSA9IG5ldyBDZXJ0aWZpY2F0ZSh0aGlzLCBgQ2VydGlmaWNhdGUtJHtwcm9wcy5kb21haW5OYW1lfWAsIHtcbiAgICAgIGRvbWFpbk5hbWU6IHByb3BzLmRvbWFpbk5hbWUsXG4gICAgICB2YWxpZGF0aW9uOiBDZXJ0aWZpY2F0ZVZhbGlkYXRpb24uZnJvbURucyhwcm9wcy5ob3N0ZWRab25lKSxcbiAgICB9KTtcbiAgICB0YWdSZXNvdXJjZSh0aGlzLmNlcnRpZmljYXRlLCBwcm9wcy50YWdzKTtcbiAgICBpZiAodHlwZW9mIHByb3BzLmRlc3Ryb3kgPT0gJ2Jvb2xlYW4nKSB7XG4gICAgICB0aGlzLmNlcnRpZmljYXRlLmFwcGx5UmVtb3ZhbFBvbGljeShwcm9wcy5kZXN0cm95ID8gUmVtb3ZhbFBvbGljeS5ERVNUUk9ZIDogUmVtb3ZhbFBvbGljeS5SRVRBSU4pO1xuICAgIH1cbiAgfVxufSJdfQ==
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
import { Construct } from 'constructs';
|
|
2
|
+
import { ICertificate } from 'aws-cdk-lib/aws-certificatemanager';
|
|
3
|
+
import { BehaviorOptions, Distribution, Function } from 'aws-cdk-lib/aws-cloudfront';
|
|
4
|
+
import type { IResourceProps } from '../lib/types.js';
|
|
5
|
+
export interface CloudFrontDistributionProps extends IResourceProps {
|
|
6
|
+
domainName: string;
|
|
7
|
+
description?: string;
|
|
8
|
+
defaultBehavior: BehaviorOptions;
|
|
9
|
+
additionalBehaviors?: Record<string, BehaviorOptions>;
|
|
10
|
+
priceClass?: 'PRICE_CLASS_ALL' | 'PRICE_CLASS_100' | 'PRICE_CLASS_200' | 'NOT_PROD' | 'PROD';
|
|
11
|
+
certificate: ICertificate;
|
|
12
|
+
webAclId?: string;
|
|
13
|
+
}
|
|
14
|
+
export declare class CloudFrontDistribution extends Construct {
|
|
15
|
+
readonly distribution: Distribution;
|
|
16
|
+
constructor(scope: Construct, id: string, props: CloudFrontDistributionProps);
|
|
17
|
+
}
|
|
18
|
+
export type CloudFrontFunctionRedirectPaths = Record<string, string | [string, 301 | 302]>;
|
|
19
|
+
export interface CloudFrontFunctionRedirects {
|
|
20
|
+
trailingSlash?: boolean;
|
|
21
|
+
defaultDocument?: string;
|
|
22
|
+
redirectDefaultDocument?: boolean;
|
|
23
|
+
paths?: CloudFrontFunctionRedirectPaths;
|
|
24
|
+
}
|
|
25
|
+
export interface DefaultDocFunctionProps extends IResourceProps {
|
|
26
|
+
functionName?: string;
|
|
27
|
+
defaultDocument?: string;
|
|
28
|
+
redirects?: CloudFrontFunctionRedirects;
|
|
29
|
+
}
|
|
30
|
+
export interface BasicAuthDefaultDocFunctionProps extends DefaultDocFunctionProps {
|
|
31
|
+
user: string;
|
|
32
|
+
password: string;
|
|
33
|
+
}
|
|
34
|
+
export interface CloudFrontFunctionProps extends IResourceProps {
|
|
35
|
+
code: string;
|
|
36
|
+
functionName?: string;
|
|
37
|
+
runtime?: 'JS_1_0' | 'JS_2_0';
|
|
38
|
+
}
|
|
39
|
+
export declare class CloudFrontFunction extends Construct {
|
|
40
|
+
readonly fn: Function;
|
|
41
|
+
constructor(scope: Construct, id: string, props: CloudFrontFunctionProps);
|
|
42
|
+
static createDefaultDoc(scope: Construct, id: string, props: DefaultDocFunctionProps): CloudFrontFunction;
|
|
43
|
+
static createBasicAuthDefaultDoc(scope: Construct, id: string, props: BasicAuthDefaultDocFunctionProps): CloudFrontFunction;
|
|
44
|
+
}
|
|
@@ -0,0 +1,78 @@
|
|
|
1
|
+
import { resolve } from 'node:path';
|
|
2
|
+
import { readFileSync } from 'node:fs';
|
|
3
|
+
import { Buffer } from 'node:buffer';
|
|
4
|
+
import { RemovalPolicy } from "aws-cdk-lib";
|
|
5
|
+
import { Construct } from 'constructs';
|
|
6
|
+
import { Distribution, Function, FunctionCode, FunctionRuntime, PriceClass } from 'aws-cdk-lib/aws-cloudfront';
|
|
7
|
+
import { tagResource } from '../lib/utils.js';
|
|
8
|
+
const __dirname = import.meta.dirname;
|
|
9
|
+
export class CloudFrontDistribution extends Construct {
|
|
10
|
+
distribution;
|
|
11
|
+
constructor(scope, id, props) {
|
|
12
|
+
super(scope, id);
|
|
13
|
+
let priceClass = PriceClass.PRICE_CLASS_ALL;
|
|
14
|
+
switch (props.priceClass) {
|
|
15
|
+
case 'NOT_PROD':
|
|
16
|
+
case 'PRICE_CLASS_100':
|
|
17
|
+
priceClass = PriceClass.PRICE_CLASS_100;
|
|
18
|
+
break;
|
|
19
|
+
case 'PRICE_CLASS_200':
|
|
20
|
+
priceClass = PriceClass.PRICE_CLASS_200;
|
|
21
|
+
break;
|
|
22
|
+
}
|
|
23
|
+
this.distribution = new Distribution(this, `${id}|CloudFrontDistribution`, {
|
|
24
|
+
comment: props.description,
|
|
25
|
+
defaultBehavior: props.defaultBehavior,
|
|
26
|
+
additionalBehaviors: props.additionalBehaviors,
|
|
27
|
+
domainNames: [props.domainName],
|
|
28
|
+
certificate: props.certificate,
|
|
29
|
+
priceClass,
|
|
30
|
+
webAclId: props.webAclId
|
|
31
|
+
});
|
|
32
|
+
tagResource(this.distribution, props.tags);
|
|
33
|
+
if (typeof props.destroy == 'boolean') {
|
|
34
|
+
this.distribution.applyRemovalPolicy(props.destroy ? RemovalPolicy.DESTROY : RemovalPolicy.RETAIN);
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
export class CloudFrontFunction extends Construct {
|
|
39
|
+
fn;
|
|
40
|
+
constructor(scope, id, props) {
|
|
41
|
+
super(scope, id);
|
|
42
|
+
this.fn = new Function(this, 'CloudFrontFunction', {
|
|
43
|
+
functionName: props.functionName,
|
|
44
|
+
runtime: FunctionRuntime[props.runtime ?? 'JS_2_0'],
|
|
45
|
+
code: FunctionCode.fromInline(props.code)
|
|
46
|
+
});
|
|
47
|
+
tagResource(this.fn, props.tags);
|
|
48
|
+
if (typeof props.destroy == 'boolean') {
|
|
49
|
+
this.fn.applyRemovalPolicy(props.destroy ? RemovalPolicy.DESTROY : RemovalPolicy.RETAIN);
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
static createDefaultDoc(scope, id, props) {
|
|
53
|
+
const code = readFileSync(resolve(__dirname, '../templates/cffn-default-doc.js'), 'utf8')
|
|
54
|
+
.replace('__DEFAULT_DOCUMENT__', props.defaultDocument ?? 'index.html')
|
|
55
|
+
.replace('__REDIRECTS__', props.redirects?.paths ? JSON.stringify(props.redirects?.paths) : 'null')
|
|
56
|
+
.replace('__TRAILING_SLASH__', props.redirects?.trailingSlash?.toString() ?? 'null')
|
|
57
|
+
.replace('__REDIR_DEF_DOC__', props.redirects?.redirectDefaultDocument?.toString() ?? 'false');
|
|
58
|
+
return new CloudFrontFunction(scope, 'DefaultDocFunction', {
|
|
59
|
+
functionName: props.functionName,
|
|
60
|
+
runtime: 'JS_2_0',
|
|
61
|
+
code
|
|
62
|
+
});
|
|
63
|
+
}
|
|
64
|
+
static createBasicAuthDefaultDoc(scope, id, props) {
|
|
65
|
+
const code = readFileSync(resolve(__dirname, '../templates/cffn-default-doc-basic-auth.js'), 'utf8')
|
|
66
|
+
.replace('__DEFAULT_DOCUMENT__', props.defaultDocument ?? 'index.html')
|
|
67
|
+
.replace('__CREDENTIALS__', Buffer.from(`${props.user}:${props.password}`).toString('base64'))
|
|
68
|
+
.replace('__REDIRECTS__', props.redirects?.paths ? JSON.stringify(props.redirects?.paths) : 'null')
|
|
69
|
+
.replace('__TRAILING_SLASH__', props.redirects?.trailingSlash?.toString() ?? 'null')
|
|
70
|
+
.replace('__REDIR_DEF_DOC__', props.redirects?.redirectDefaultDocument?.toString() ?? 'false');
|
|
71
|
+
return new CloudFrontFunction(scope, 'BasicAuthDefaultDocFunction', {
|
|
72
|
+
functionName: props.functionName,
|
|
73
|
+
runtime: 'JS_2_0',
|
|
74
|
+
code
|
|
75
|
+
});
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloudfront.js","sourceRoot":"","sources":["../../src/constructs/cloudfront.ts"],"names":[],"mappings":"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","sourcesContent":["import { resolve } from 'node:path';\nimport { readFileSync } from 'node:fs';\nimport { Buffer } from 'node:buffer';\nimport { RemovalPolicy } from \"aws-cdk-lib\";\nimport { Construct } from 'constructs';\nimport { ICertificate } from 'aws-cdk-lib/aws-certificatemanager';\nimport { BehaviorOptions, Distribution, Function, FunctionCode, FunctionEventType, FunctionRuntime, IOrigin, PriceClass } from 'aws-cdk-lib/aws-cloudfront';\nimport { tagResource } from '../lib/utils.js';\nimport type { IResourceProps } from '../lib/types.js';\n\nconst __dirname = import.meta.dirname;\n\nexport interface CloudFrontDistributionProps extends IResourceProps {\n  domainName: string;\n  description?: string;\n  defaultBehavior: BehaviorOptions;\n  additionalBehaviors?: Record<string, BehaviorOptions>;\n  priceClass?: 'PRICE_CLASS_ALL' | 'PRICE_CLASS_100' | 'PRICE_CLASS_200' | 'NOT_PROD' | 'PROD';\n  certificate: ICertificate;\n  webAclId?: string;\n}\nexport class CloudFrontDistribution extends Construct {\n  public readonly distribution: Distribution;\n\n  constructor(scope: Construct, id: string, props: CloudFrontDistributionProps) {\n    super(scope, id);\n\n    let priceClass = PriceClass.PRICE_CLASS_ALL;\n    switch (props.priceClass) {\n      case 'NOT_PROD':\n      case 'PRICE_CLASS_100':\n        priceClass = PriceClass.PRICE_CLASS_100;\n        break;\n      case 'PRICE_CLASS_200':\n        priceClass = PriceClass.PRICE_CLASS_200;\n        break;\n    }\n\n    this.distribution = new Distribution(this, `${id}|CloudFrontDistribution`, {\n      comment: props.description,\n      defaultBehavior: props.defaultBehavior,\n      additionalBehaviors: props.additionalBehaviors,\n      domainNames: [props.domainName],\n      certificate: props.certificate,\n      priceClass,\n      webAclId: props.webAclId\n    });\n    tagResource(this.distribution, props.tags);\n    if (typeof props.destroy == 'boolean') {\n      this.distribution.applyRemovalPolicy(props.destroy ? RemovalPolicy.DESTROY : RemovalPolicy.RETAIN);\n    }\n  }\n}\n\nexport type CloudFrontFunctionRedirectPaths = Record<string, string | [string, 301|302]>; \n\nexport interface CloudFrontFunctionRedirects {\n  trailingSlash?: boolean;\n  defaultDocument?: string;\n  redirectDefaultDocument?: boolean;\n  paths?: CloudFrontFunctionRedirectPaths;\n}\n\nexport interface DefaultDocFunctionProps extends IResourceProps {\n  functionName?: string;\n  defaultDocument?: string;\n  redirects?: CloudFrontFunctionRedirects;\n}\n\nexport interface BasicAuthDefaultDocFunctionProps extends DefaultDocFunctionProps {\n  user: string;\n  password: string;\n}\n\nexport interface CloudFrontFunctionProps extends IResourceProps {\n  code: string;\n  functionName?: string;\n  runtime?: 'JS_1_0' | 'JS_2_0';\n}\nexport class CloudFrontFunction extends Construct {\n  public readonly fn: Function;\n\n  constructor(scope: Construct, id: string, props: CloudFrontFunctionProps) {\n    super(scope, id);\n\n    this.fn = new Function(this, 'CloudFrontFunction', {\n      functionName: props.functionName,\n      runtime: FunctionRuntime[props.runtime ?? 'JS_2_0'],\n      code: FunctionCode.fromInline(props.code)\n    });\n    tagResource(this.fn, props.tags);\n\n    if (typeof props.destroy == 'boolean') {\n      this.fn.applyRemovalPolicy(props.destroy ? RemovalPolicy.DESTROY : RemovalPolicy.RETAIN);\n    }\n  }\n\n  static createDefaultDoc(scope: Construct, id: string, props: DefaultDocFunctionProps) {\n    const code = readFileSync(resolve(__dirname, '../templates/cffn-default-doc.js'), 'utf8')\n      .replace('__DEFAULT_DOCUMENT__', props.defaultDocument ?? 'index.html')\n      .replace('__REDIRECTS__', props.redirects?.paths ? JSON.stringify(props.redirects?.paths ) : 'null')\n      .replace('__TRAILING_SLASH__', props.redirects?.trailingSlash?.toString() ?? 'null')\n      .replace('__REDIR_DEF_DOC__', props.redirects?.redirectDefaultDocument?.toString() ?? 'false');\n\n    return new CloudFrontFunction(scope, 'DefaultDocFunction', {\n      functionName: props.functionName,\n      runtime: 'JS_2_0',\n      code\n    });\n  }\n\n  static createBasicAuthDefaultDoc(scope: Construct, id: string, props: BasicAuthDefaultDocFunctionProps) {\n    const code = readFileSync(resolve(__dirname, '../templates/cffn-default-doc-basic-auth.js'), 'utf8')\n      .replace('__DEFAULT_DOCUMENT__', props.defaultDocument ?? 'index.html')\n      .replace('__CREDENTIALS__', Buffer.from(`${props.user}:${props.password}`).toString('base64'))\n      .replace('__REDIRECTS__', props.redirects?.paths ? JSON.stringify(props.redirects?.paths ) : 'null')\n      .replace('__TRAILING_SLASH__', props.redirects?.trailingSlash?.toString() ?? 'null')\n      .replace('__REDIR_DEF_DOC__', props.redirects?.redirectDefaultDocument?.toString() ?? 'false');\n    \n    return new CloudFrontFunction(scope, 'BasicAuthDefaultDocFunction', {\n      functionName: props.functionName,\n      runtime: 'JS_2_0',\n      code\n    });\n  }\n}\n"]}
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
import { Construct } from 'constructs';
|
|
2
|
+
import { BuildEnvironmentVariableType, IBuildImage, PipelineProject } from 'aws-cdk-lib/aws-codebuild';
|
|
3
|
+
export type EnvironmentVariables = Record<string, {
|
|
4
|
+
value: any;
|
|
5
|
+
type?: BuildEnvironmentVariableType;
|
|
6
|
+
}>;
|
|
7
|
+
export interface EcrBuildImage {
|
|
8
|
+
repository: string;
|
|
9
|
+
tag: string;
|
|
10
|
+
}
|
|
11
|
+
export interface CodeBuildPipelineProjectProps {
|
|
12
|
+
projectName?: string;
|
|
13
|
+
buildSpec?: string | object;
|
|
14
|
+
buildImage?: ('STANDARD_5_0' | 'STANDARD_6_0' | 'STANDARD_7_0') | IBuildImage;
|
|
15
|
+
ecrBuildImage?: EcrBuildImage;
|
|
16
|
+
computeType?: ('SMALL' | 'MEDIUM' | 'LARGE' | 'X_LARGE' | 'X2_LARGE') | string;
|
|
17
|
+
environmentVariables?: EnvironmentVariables;
|
|
18
|
+
cloudFrontDistributionArn?: string;
|
|
19
|
+
s3BucketName?: string;
|
|
20
|
+
destroy?: boolean;
|
|
21
|
+
tags?: Record<string, string>;
|
|
22
|
+
}
|
|
23
|
+
export declare class CodeBuildPipelineProject extends Construct {
|
|
24
|
+
readonly project: PipelineProject;
|
|
25
|
+
constructor(scope: Construct, id: string, props: CodeBuildPipelineProjectProps);
|
|
26
|
+
}
|
|
@@ -0,0 +1,76 @@
|
|
|
1
|
+
import { RemovalPolicy } from "aws-cdk-lib";
|
|
2
|
+
import { Construct } from 'constructs';
|
|
3
|
+
import { BuildSpec, ComputeType, LinuxBuildImage, PipelineProject } from 'aws-cdk-lib/aws-codebuild';
|
|
4
|
+
import { Repository } from "aws-cdk-lib/aws-ecr";
|
|
5
|
+
import { PolicyStatement } from 'aws-cdk-lib/aws-iam';
|
|
6
|
+
import { Bucket } from 'aws-cdk-lib/aws-s3';
|
|
7
|
+
import { tagResource } from "../lib/utils.js";
|
|
8
|
+
export class CodeBuildPipelineProject extends Construct {
|
|
9
|
+
project;
|
|
10
|
+
constructor(scope, id, props) {
|
|
11
|
+
super(scope, id);
|
|
12
|
+
if (!props.environmentVariables) {
|
|
13
|
+
props.environmentVariables = {};
|
|
14
|
+
}
|
|
15
|
+
if (props.cloudFrontDistributionArn && !props.environmentVariables.CLOUDFRONT_DISTRIBUTION_ID) {
|
|
16
|
+
props.environmentVariables.CLOUDFRONT_DISTRIBUTION_ID = {
|
|
17
|
+
value: props.cloudFrontDistributionArn.split('/').pop()
|
|
18
|
+
};
|
|
19
|
+
}
|
|
20
|
+
if (props.s3BucketName && !props.environmentVariables.S3_BUCKET_NAME) {
|
|
21
|
+
props.environmentVariables.S3_BUCKET_NAME = {
|
|
22
|
+
value: props.s3BucketName
|
|
23
|
+
};
|
|
24
|
+
}
|
|
25
|
+
let buildImage = undefined;
|
|
26
|
+
if (props.ecrBuildImage) {
|
|
27
|
+
const repository = props.ecrBuildImage.repository.startsWith('arn:aws') ?
|
|
28
|
+
Repository.fromRepositoryArn(this, 'EcrRepository', props.ecrBuildImage.repository) :
|
|
29
|
+
Repository.fromRepositoryName(this, 'EcrRepository', props.ecrBuildImage.repository);
|
|
30
|
+
buildImage = LinuxBuildImage.fromEcrRepository(repository, props.ecrBuildImage.tag);
|
|
31
|
+
}
|
|
32
|
+
else if (props.buildImage) {
|
|
33
|
+
if (typeof props.buildImage == 'string') {
|
|
34
|
+
buildImage = LinuxBuildImage[props.buildImage ?? 'STANDARD_7_0'];
|
|
35
|
+
}
|
|
36
|
+
else {
|
|
37
|
+
buildImage = props.buildImage;
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
this.project = new PipelineProject(this, 'CodeBuildPipelineProjectProps', {
|
|
41
|
+
projectName: props.projectName,
|
|
42
|
+
buildSpec: props.buildSpec ? typeof props.buildSpec == 'object' ? BuildSpec.fromObjectToYaml(props.buildSpec) : BuildSpec.fromSourceFilename(props.buildSpec ?? 'buildspec.yml') : undefined,
|
|
43
|
+
environment: {
|
|
44
|
+
buildImage: buildImage ?? LinuxBuildImage.STANDARD_7_0,
|
|
45
|
+
computeType: ComputeType[props.computeType ?? 'SMALL'],
|
|
46
|
+
environmentVariables: props.environmentVariables
|
|
47
|
+
}
|
|
48
|
+
});
|
|
49
|
+
tagResource(this.project, props.tags);
|
|
50
|
+
if (typeof props.destroy == 'boolean') {
|
|
51
|
+
this.project.applyRemovalPolicy(props.destroy ? RemovalPolicy.DESTROY : RemovalPolicy.RETAIN);
|
|
52
|
+
}
|
|
53
|
+
// Add CloudWatch log permissions
|
|
54
|
+
this.project.addToRolePolicy(new PolicyStatement({
|
|
55
|
+
actions: [
|
|
56
|
+
'logs:CreateLogGroup',
|
|
57
|
+
'logs:CreateLogStream',
|
|
58
|
+
'logs:PutLogEvents'
|
|
59
|
+
],
|
|
60
|
+
resources: [this.project.projectArn]
|
|
61
|
+
}));
|
|
62
|
+
// Allow the project to create an invalidation
|
|
63
|
+
if (props.cloudFrontDistributionArn) {
|
|
64
|
+
this.project.addToRolePolicy(new PolicyStatement({
|
|
65
|
+
actions: ['cloudfront:CreateInvalidation'],
|
|
66
|
+
resources: [props.cloudFrontDistributionArn]
|
|
67
|
+
}));
|
|
68
|
+
}
|
|
69
|
+
// Allow the project to sync files to S3
|
|
70
|
+
if (props.s3BucketName) {
|
|
71
|
+
const s3Bucket = Bucket.fromBucketName(this, 'S3BucketByName', props.s3BucketName);
|
|
72
|
+
s3Bucket.grantReadWrite(this.project);
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"codebuild.js","sourceRoot":"","sources":["../../src/constructs/codebuild.ts"],"names":[],"mappings":"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","sourcesContent":["import { RemovalPolicy } from \"aws-cdk-lib\";\nimport { Construct } from 'constructs';\nimport { BuildEnvironmentVariableType, BuildSpec, ComputeType, IBuildImage, LinuxBuildImage, PipelineProject } from 'aws-cdk-lib/aws-codebuild';\nimport { Repository } from \"aws-cdk-lib/aws-ecr\";\nimport { PolicyStatement } from 'aws-cdk-lib/aws-iam';\nimport { Bucket, IBucket } from 'aws-cdk-lib/aws-s3';\n\nimport { tagResource } from \"../lib/utils.js\";\n\nexport type EnvironmentVariables = Record<string, { value: any, type?: BuildEnvironmentVariableType }>;\n\nexport interface EcrBuildImage {\n  repository: string;\n  tag: string;\n}\nexport interface CodeBuildPipelineProjectProps {\n  projectName?: string;\n  buildSpec?: string | object;\n  buildImage?: ('STANDARD_5_0' | 'STANDARD_6_0' | 'STANDARD_7_0') | IBuildImage;\n  ecrBuildImage?: EcrBuildImage,\n  computeType?: ('SMALL' | 'MEDIUM' | 'LARGE' | 'X_LARGE' | 'X2_LARGE') | string;\n  environmentVariables?: EnvironmentVariables;\n  cloudFrontDistributionArn?: string;\n  s3BucketName?: string;\n  destroy?: boolean;\n  tags?: Record<string, string>;\n}\n\nexport class CodeBuildPipelineProject extends Construct {\n  public readonly project: PipelineProject;\n\n  constructor(scope: Construct, id: string, props: CodeBuildPipelineProjectProps) {\n    super(scope, id);\n\n    if (!props.environmentVariables) {\n      props.environmentVariables = {};\n    }\n    if (props.cloudFrontDistributionArn && !props.environmentVariables.CLOUDFRONT_DISTRIBUTION_ID) {\n      props.environmentVariables.CLOUDFRONT_DISTRIBUTION_ID = {\n        value: props.cloudFrontDistributionArn.split('/').pop()\n      };\n    }\n\n    if (props.s3BucketName && !props.environmentVariables.S3_BUCKET_NAME) {\n      props.environmentVariables.S3_BUCKET_NAME = {\n        value: props.s3BucketName\n      };\n    }\n\n    let buildImage: IBuildImage | undefined = undefined;\n    if (props.ecrBuildImage) {\n      const repository = props.ecrBuildImage.repository.startsWith('arn:aws') ? \n        Repository.fromRepositoryArn(this, 'EcrRepository', props.ecrBuildImage.repository) :\n        Repository.fromRepositoryName(this, 'EcrRepository', props.ecrBuildImage.repository);\n      buildImage = LinuxBuildImage.fromEcrRepository(repository, props.ecrBuildImage.tag);\n    } else if (props.buildImage) {\n      if (typeof props.buildImage == 'string') {\n        buildImage = LinuxBuildImage[props.buildImage ?? 'STANDARD_7_0']\n      } else {\n        buildImage = props.buildImage as IBuildImage;\n      }\n    }\n\n    this.project = new PipelineProject(this, 'CodeBuildPipelineProjectProps', {\n      projectName: props.projectName,\n      buildSpec: props.buildSpec ? typeof props.buildSpec == 'object' ? BuildSpec.fromObjectToYaml(props.buildSpec) : BuildSpec.fromSourceFilename(props.buildSpec ?? 'buildspec.yml') : undefined,\n      environment: {\n        buildImage: buildImage ?? LinuxBuildImage.STANDARD_7_0,\n        computeType: ComputeType[props.computeType as (keyof typeof ComputeType | undefined) ?? 'SMALL'],\n        environmentVariables: props.environmentVariables\n      }\n    });\n    tagResource(this.project, props.tags);\n    if (typeof props.destroy == 'boolean') {\n      this.project.applyRemovalPolicy(props.destroy ? RemovalPolicy.DESTROY : RemovalPolicy.RETAIN);\n    }\n\n    // Add CloudWatch log permissions\n    this.project.addToRolePolicy(new PolicyStatement({\n      actions: [\n        'logs:CreateLogGroup',\n        'logs:CreateLogStream',\n        'logs:PutLogEvents'\n      ],\n      resources: [this.project.projectArn]\n    }));\n\n    // Allow the project to create an invalidation\n    if (props.cloudFrontDistributionArn) {\n      this.project.addToRolePolicy(new PolicyStatement({\n        actions: ['cloudfront:CreateInvalidation'],\n        resources: [props.cloudFrontDistributionArn]\n      }));\n    }\n\n    // Allow the project to sync files to S3\n    if (props.s3BucketName) {\n      const s3Bucket = Bucket.fromBucketName(this, 'S3BucketByName', props.s3BucketName);\n      s3Bucket.grantReadWrite(this.project);\n    }\n  }\n}"]}
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
import { Construct } from 'constructs';
|
|
2
|
+
import { IProject } from "aws-cdk-lib/aws-codebuild";
|
|
3
|
+
import { Artifact, Pipeline } from "aws-cdk-lib/aws-codepipeline";
|
|
4
|
+
import { Bucket } from 'aws-cdk-lib/aws-s3';
|
|
5
|
+
import type { EnvironmentVariables } from './codebuild.js';
|
|
6
|
+
export interface AddSourceStageParams {
|
|
7
|
+
stageName?: string;
|
|
8
|
+
codestarConnectionArn: string;
|
|
9
|
+
repoOwner: string;
|
|
10
|
+
repo: string;
|
|
11
|
+
branch: string;
|
|
12
|
+
}
|
|
13
|
+
export interface AddBuildStageParams {
|
|
14
|
+
stageName?: string;
|
|
15
|
+
codeBuildProject: IProject;
|
|
16
|
+
sourceArtifact: Artifact;
|
|
17
|
+
environmentVariables?: EnvironmentVariables;
|
|
18
|
+
}
|
|
19
|
+
export interface CodePipelineProjectProps {
|
|
20
|
+
pipelineName?: string;
|
|
21
|
+
destroy?: boolean;
|
|
22
|
+
tags?: Record<string, string>;
|
|
23
|
+
}
|
|
24
|
+
export declare class CodePipelineProject extends Construct {
|
|
25
|
+
readonly s3Bucket: Bucket;
|
|
26
|
+
readonly pipeline: Pipeline;
|
|
27
|
+
constructor(scope: Construct, id: string, props: CodePipelineProjectProps);
|
|
28
|
+
addSourceStage(params: AddSourceStageParams): Artifact;
|
|
29
|
+
addBuildStage(params: AddBuildStageParams): void;
|
|
30
|
+
}
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
import { RemovalPolicy, PhysicalName } from "aws-cdk-lib";
|
|
2
|
+
import { Construct } from 'constructs';
|
|
3
|
+
import { Artifact, Pipeline } from "aws-cdk-lib/aws-codepipeline";
|
|
4
|
+
import { CodeBuildAction, CodeStarConnectionsSourceAction } from "aws-cdk-lib/aws-codepipeline-actions";
|
|
5
|
+
import { Bucket } from 'aws-cdk-lib/aws-s3';
|
|
6
|
+
import { tagResource } from "../lib/utils.js";
|
|
7
|
+
export class CodePipelineProject extends Construct {
|
|
8
|
+
s3Bucket;
|
|
9
|
+
pipeline;
|
|
10
|
+
constructor(scope, id, props) {
|
|
11
|
+
super(scope, id);
|
|
12
|
+
this.s3Bucket = new Bucket(this, 'CodePipelineArtifactBucket', {
|
|
13
|
+
bucketName: PhysicalName.GENERATE_IF_NEEDED,
|
|
14
|
+
removalPolicy: props.destroy !== undefined ? props.destroy ? RemovalPolicy.DESTROY : RemovalPolicy.RETAIN : undefined,
|
|
15
|
+
autoDeleteObjects: props.destroy === true ? true : undefined
|
|
16
|
+
});
|
|
17
|
+
tagResource(this.s3Bucket, props.tags);
|
|
18
|
+
this.pipeline = new Pipeline(this, 'CodePipelineProject', {
|
|
19
|
+
pipelineName: props.pipelineName,
|
|
20
|
+
artifactBucket: this.s3Bucket
|
|
21
|
+
});
|
|
22
|
+
tagResource(this.pipeline, props.tags);
|
|
23
|
+
if (typeof props.destroy == 'boolean') {
|
|
24
|
+
this.pipeline.applyRemovalPolicy(props.destroy ? RemovalPolicy.DESTROY : RemovalPolicy.RETAIN);
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
addSourceStage(params) {
|
|
28
|
+
const artifact = new Artifact();
|
|
29
|
+
this.pipeline.addStage({
|
|
30
|
+
stageName: params.stageName ?? 'Source',
|
|
31
|
+
actions: [
|
|
32
|
+
new CodeStarConnectionsSourceAction({
|
|
33
|
+
actionName: params.stageName ?? 'Source',
|
|
34
|
+
output: artifact,
|
|
35
|
+
connectionArn: params.codestarConnectionArn,
|
|
36
|
+
owner: params.repoOwner,
|
|
37
|
+
repo: params.repo,
|
|
38
|
+
branch: params.branch
|
|
39
|
+
})
|
|
40
|
+
]
|
|
41
|
+
});
|
|
42
|
+
return artifact;
|
|
43
|
+
}
|
|
44
|
+
addBuildStage(params) {
|
|
45
|
+
this.pipeline.addStage({
|
|
46
|
+
stageName: params.stageName ?? 'Build',
|
|
47
|
+
actions: [
|
|
48
|
+
new CodeBuildAction({
|
|
49
|
+
actionName: params.stageName ?? 'Build',
|
|
50
|
+
project: params.codeBuildProject,
|
|
51
|
+
input: params.sourceArtifact,
|
|
52
|
+
environmentVariables: params.environmentVariables
|
|
53
|
+
})
|
|
54
|
+
]
|
|
55
|
+
});
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29kZXBpcGVsaW5lLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2NvbnN0cnVjdHMvY29kZXBpcGVsaW5lLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxhQUFhLEVBQUUsWUFBWSxFQUFFLE1BQU0sYUFBYSxDQUFDO0FBQzFELE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxZQUFZLENBQUM7QUFFdkMsT0FBTyxFQUFFLFFBQVEsRUFBRSxRQUFRLEVBQUUsTUFBTSw4QkFBOEIsQ0FBQztBQUNsRSxPQUFPLEVBQUUsZUFBZSxFQUFFLCtCQUErQixFQUFFLE1BQU0sc0NBQXNDLENBQUM7QUFDeEcsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLG9CQUFvQixDQUFDO0FBRzVDLE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQXFCOUMsTUFBTSxPQUFPLG1CQUFvQixTQUFRLFNBQVM7SUFDaEMsUUFBUSxDQUFTO0lBQ2pCLFFBQVEsQ0FBVztJQUVuQyxZQUFZLEtBQWdCLEVBQUUsRUFBVSxFQUFFLEtBQStCO1FBQ3ZFLEtBQUssQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDLENBQUM7UUFFakIsSUFBSSxDQUFDLFFBQVEsR0FBRyxJQUFJLE1BQU0sQ0FBQyxJQUFJLEVBQUUsNEJBQTRCLEVBQUU7WUFDN0QsVUFBVSxFQUFFLFlBQVksQ0FBQyxrQkFBa0I7WUFDM0MsYUFBYSxFQUFFLEtBQUssQ0FBQyxPQUFPLEtBQUssU0FBUyxDQUFDLENBQUMsQ0FBQyxLQUFLLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxhQUFhLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxhQUFhLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQyxTQUFTO1lBQ3JILGlCQUFpQixFQUFFLEtBQUssQ0FBQyxPQUFPLEtBQUssSUFBSSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLFNBQVM7U0FDN0QsQ0FBQyxDQUFDO1FBQ0gsV0FBVyxDQUFDLElBQUksQ0FBQyxRQUFRLEVBQUUsS0FBSyxDQUFDLElBQUksQ0FBQyxDQUFDO1FBRXZDLElBQUksQ0FBQyxRQUFRLEdBQUcsSUFBSSxRQUFRLENBQUMsSUFBSSxFQUFFLHFCQUFxQixFQUFFO1lBQ3hELFlBQVksRUFBRSxLQUFLLENBQUMsWUFBWTtZQUNoQyxjQUFjLEVBQUUsSUFBSSxDQUFDLFFBQVE7U0FDOUIsQ0FBQyxDQUFDO1FBRUgsV0FBVyxDQUFDLElBQUksQ0FBQyxRQUFRLEVBQUUsS0FBSyxDQUFDLElBQUksQ0FBQyxDQUFDO1FBQ3ZDLElBQUksT0FBTyxLQUFLLENBQUMsT0FBTyxJQUFJLFNBQVMsRUFBRSxDQUFDO1lBQ3RDLElBQUksQ0FBQyxRQUFRLENBQUMsa0JBQWtCLENBQUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsYUFBYSxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsYUFBYSxDQUFDLE1BQU0sQ0FBQyxDQUFDO1FBQ2pHLENBQUM7SUFDSCxDQUFDO0lBRUQsY0FBYyxDQUFDLE1BQTRCO1FBQ3pDLE1BQU0sUUFBUSxHQUFHLElBQUksUUFBUSxFQUFFLENBQUM7UUFDaEMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxRQUFRLENBQUM7WUFDckIsU0FBUyxFQUFFLE1BQU0sQ0FBQyxTQUFTLElBQUksUUFBUTtZQUN2QyxPQUFPLEVBQUU7Z0JBQ1AsSUFBSSwrQkFBK0IsQ0FBQztvQkFDbEMsVUFBVSxFQUFFLE1BQU0sQ0FBQyxTQUFTLElBQUksUUFBUTtvQkFDeEMsTUFBTSxFQUFFLFFBQVE7b0JBQ2hCLGFBQWEsRUFBRSxNQUFNLENBQUMscUJBQXFCO29CQUMzQyxLQUFLLEVBQUUsTUFBTSxDQUFDLFNBQVM7b0JBQ3ZCLElBQUksRUFBRSxNQUFNLENBQUMsSUFBSTtvQkFDakIsTUFBTSxFQUFFLE1BQU0sQ0FBQyxNQUFNO2lCQUN0QixDQUFDO2FBQ0g7U0FDRixDQUFDLENBQUM7UUFDSCxPQUFPLFFBQVEsQ0FBQztJQUNsQixDQUFDO0lBRUQsYUFBYSxDQUFDLE1BQTJCO1FBQ3ZDLElBQUksQ0FBQyxRQUFRLENBQUMsUUFBUSxDQUFDO1lBQ3JCLFNBQVMsRUFBRSxNQUFNLENBQUMsU0FBUyxJQUFJLE9BQU87WUFDdEMsT0FBTyxFQUFFO2dCQUNQLElBQUksZUFBZSxDQUFDO29CQUNsQixVQUFVLEVBQUUsTUFBTSxDQUFDLFNBQVMsSUFBSSxPQUFPO29CQUN2QyxPQUFPLEVBQUUsTUFBTSxDQUFDLGdCQUFnQjtvQkFDaEMsS0FBSyxFQUFFLE1BQU0sQ0FBQyxjQUFjO29CQUM1QixvQkFBb0IsRUFBRSxNQUFNLENBQUMsb0JBQW9CO2lCQUNsRCxDQUFDO2FBQ0g7U0FDRixDQUFDLENBQUM7SUFDTCxDQUFDO0NBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBSZW1vdmFsUG9saWN5LCBQaHlzaWNhbE5hbWUgfSBmcm9tIFwiYXdzLWNkay1saWJcIjtcbmltcG9ydCB7IENvbnN0cnVjdCB9IGZyb20gJ2NvbnN0cnVjdHMnO1xuaW1wb3J0IHsgSVByb2plY3QgfSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWNvZGVidWlsZFwiO1xuaW1wb3J0IHsgQXJ0aWZhY3QsIFBpcGVsaW5lIH0gZnJvbSBcImF3cy1jZGstbGliL2F3cy1jb2RlcGlwZWxpbmVcIjtcbmltcG9ydCB7IENvZGVCdWlsZEFjdGlvbiwgQ29kZVN0YXJDb25uZWN0aW9uc1NvdXJjZUFjdGlvbiB9IGZyb20gXCJhd3MtY2RrLWxpYi9hd3MtY29kZXBpcGVsaW5lLWFjdGlvbnNcIjtcbmltcG9ydCB7IEJ1Y2tldCB9IGZyb20gJ2F3cy1jZGstbGliL2F3cy1zMyc7XG5cbmltcG9ydCB0eXBlIHsgRW52aXJvbm1lbnRWYXJpYWJsZXMgfSBmcm9tICcuL2NvZGVidWlsZC5qcyc7XG5pbXBvcnQgeyB0YWdSZXNvdXJjZSB9IGZyb20gXCIuLi9saWIvdXRpbHMuanNcIjtcblxuZXhwb3J0IGludGVyZmFjZSBBZGRTb3VyY2VTdGFnZVBhcmFtcyB7XG4gIHN0YWdlTmFtZT86IHN0cmluZztcbiAgY29kZXN0YXJDb25uZWN0aW9uQXJuOiBzdHJpbmc7XG4gIHJlcG9Pd25lcjogc3RyaW5nO1xuICByZXBvOiBzdHJpbmc7XG4gIGJyYW5jaDogc3RyaW5nO1xufVxuZXhwb3J0IGludGVyZmFjZSBBZGRCdWlsZFN0YWdlUGFyYW1zIHtcbiAgc3RhZ2VOYW1lPzogc3RyaW5nO1xuICBjb2RlQnVpbGRQcm9qZWN0OiBJUHJvamVjdDtcbiAgc291cmNlQXJ0aWZhY3Q6IEFydGlmYWN0O1xuICBlbnZpcm9ubWVudFZhcmlhYmxlcz86IEVudmlyb25tZW50VmFyaWFibGVzO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIENvZGVQaXBlbGluZVByb2plY3RQcm9wcyB7XG4gIHBpcGVsaW5lTmFtZT86IHN0cmluZztcbiAgZGVzdHJveT86IGJvb2xlYW47XG4gIHRhZ3M/OiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+O1xufVxuZXhwb3J0IGNsYXNzIENvZGVQaXBlbGluZVByb2plY3QgZXh0ZW5kcyBDb25zdHJ1Y3Qge1xuICBwdWJsaWMgcmVhZG9ubHkgczNCdWNrZXQ6IEJ1Y2tldDtcbiAgcHVibGljIHJlYWRvbmx5IHBpcGVsaW5lOiBQaXBlbGluZTtcblxuICBjb25zdHJ1Y3RvcihzY29wZTogQ29uc3RydWN0LCBpZDogc3RyaW5nLCBwcm9wczogQ29kZVBpcGVsaW5lUHJvamVjdFByb3BzKSB7XG4gICAgc3VwZXIoc2NvcGUsIGlkKTtcblxuICAgIHRoaXMuczNCdWNrZXQgPSBuZXcgQnVja2V0KHRoaXMsICdDb2RlUGlwZWxpbmVBcnRpZmFjdEJ1Y2tldCcsIHtcbiAgICAgIGJ1Y2tldE5hbWU6IFBoeXNpY2FsTmFtZS5HRU5FUkFURV9JRl9ORUVERUQsXG4gICAgICByZW1vdmFsUG9saWN5OiBwcm9wcy5kZXN0cm95ICE9PSB1bmRlZmluZWQgPyBwcm9wcy5kZXN0cm95ID8gUmVtb3ZhbFBvbGljeS5ERVNUUk9ZIDogUmVtb3ZhbFBvbGljeS5SRVRBSU4gOiB1bmRlZmluZWQsXG4gICAgICBhdXRvRGVsZXRlT2JqZWN0czogcHJvcHMuZGVzdHJveSA9PT0gdHJ1ZSA/IHRydWUgOiB1bmRlZmluZWRcbiAgICB9KTtcbiAgICB0YWdSZXNvdXJjZSh0aGlzLnMzQnVja2V0LCBwcm9wcy50YWdzKTtcblxuICAgIHRoaXMucGlwZWxpbmUgPSBuZXcgUGlwZWxpbmUodGhpcywgJ0NvZGVQaXBlbGluZVByb2plY3QnLCB7XG4gICAgICBwaXBlbGluZU5hbWU6IHByb3BzLnBpcGVsaW5lTmFtZSxcbiAgICAgIGFydGlmYWN0QnVja2V0OiB0aGlzLnMzQnVja2V0XG4gICAgfSk7XG5cbiAgICB0YWdSZXNvdXJjZSh0aGlzLnBpcGVsaW5lLCBwcm9wcy50YWdzKTtcbiAgICBpZiAodHlwZW9mIHByb3BzLmRlc3Ryb3kgPT0gJ2Jvb2xlYW4nKSB7XG4gICAgICB0aGlzLnBpcGVsaW5lLmFwcGx5UmVtb3ZhbFBvbGljeShwcm9wcy5kZXN0cm95ID8gUmVtb3ZhbFBvbGljeS5ERVNUUk9ZIDogUmVtb3ZhbFBvbGljeS5SRVRBSU4pO1xuICAgIH1cbiAgfVxuXG4gIGFkZFNvdXJjZVN0YWdlKHBhcmFtczogQWRkU291cmNlU3RhZ2VQYXJhbXMpOiBBcnRpZmFjdCB7XG4gICAgY29uc3QgYXJ0aWZhY3QgPSBuZXcgQXJ0aWZhY3QoKTtcbiAgICB0aGlzLnBpcGVsaW5lLmFkZFN0YWdlKHtcbiAgICAgIHN0YWdlTmFtZTogcGFyYW1zLnN0YWdlTmFtZSA/PyAnU291cmNlJyxcbiAgICAgIGFjdGlvbnM6IFtcbiAgICAgICAgbmV3IENvZGVTdGFyQ29ubmVjdGlvbnNTb3VyY2VBY3Rpb24oe1xuICAgICAgICAgIGFjdGlvbk5hbWU6IHBhcmFtcy5zdGFnZU5hbWUgPz8gJ1NvdXJjZScsXG4gICAgICAgICAgb3V0cHV0OiBhcnRpZmFjdCxcbiAgICAgICAgICBjb25uZWN0aW9uQXJuOiBwYXJhbXMuY29kZXN0YXJDb25uZWN0aW9uQXJuLFxuICAgICAgICAgIG93bmVyOiBwYXJhbXMucmVwb093bmVyLFxuICAgICAgICAgIHJlcG86IHBhcmFtcy5yZXBvLFxuICAgICAgICAgIGJyYW5jaDogcGFyYW1zLmJyYW5jaFxuICAgICAgICB9KVxuICAgICAgXVxuICAgIH0pO1xuICAgIHJldHVybiBhcnRpZmFjdDtcbiAgfVxuXG4gIGFkZEJ1aWxkU3RhZ2UocGFyYW1zOiBBZGRCdWlsZFN0YWdlUGFyYW1zKSB7XG4gICAgdGhpcy5waXBlbGluZS5hZGRTdGFnZSh7XG4gICAgICBzdGFnZU5hbWU6IHBhcmFtcy5zdGFnZU5hbWUgPz8gJ0J1aWxkJyxcbiAgICAgIGFjdGlvbnM6IFtcbiAgICAgICAgbmV3IENvZGVCdWlsZEFjdGlvbih7XG4gICAgICAgICAgYWN0aW9uTmFtZTogcGFyYW1zLnN0YWdlTmFtZSA/PyAnQnVpbGQnLFxuICAgICAgICAgIHByb2plY3Q6IHBhcmFtcy5jb2RlQnVpbGRQcm9qZWN0LFxuICAgICAgICAgIGlucHV0OiBwYXJhbXMuc291cmNlQXJ0aWZhY3QsXG4gICAgICAgICAgZW52aXJvbm1lbnRWYXJpYWJsZXM6IHBhcmFtcy5lbnZpcm9ubWVudFZhcmlhYmxlc1xuICAgICAgICB9KVxuICAgICAgXVxuICAgIH0pO1xuICB9XG59Il19
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
import { IDistribution } from "aws-cdk-lib/aws-cloudfront";
|
|
2
|
+
import { ARecord, AaaaRecord, IHostedZone } from "aws-cdk-lib/aws-route53";
|
|
3
|
+
import { Construct } from "constructs";
|
|
4
|
+
export interface CloudFrontAliasProps {
|
|
5
|
+
hostedZone: IHostedZone;
|
|
6
|
+
domainName: string;
|
|
7
|
+
distribution: IDistribution;
|
|
8
|
+
}
|
|
9
|
+
export declare class CloudFrontAlias extends Construct {
|
|
10
|
+
readonly aRecord: ARecord;
|
|
11
|
+
readonly aaaaRecord: AaaaRecord;
|
|
12
|
+
constructor(scope: Construct, id: string, props: CloudFrontAliasProps);
|
|
13
|
+
}
|
|
14
|
+
export declare function getHostedZone(scope: Construct, id: string, domainName: string): IHostedZone;
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import { ARecord, AaaaRecord, HostedZone, RecordTarget } from "aws-cdk-lib/aws-route53";
|
|
2
|
+
import { CloudFrontTarget } from "aws-cdk-lib/aws-route53-targets";
|
|
3
|
+
import { Construct } from "constructs";
|
|
4
|
+
export class CloudFrontAlias extends Construct {
|
|
5
|
+
aRecord;
|
|
6
|
+
aaaaRecord;
|
|
7
|
+
constructor(scope, id, props) {
|
|
8
|
+
super(scope, id);
|
|
9
|
+
this.aRecord = new ARecord(this, `ARecord-${props.domainName}`, {
|
|
10
|
+
zone: props.hostedZone,
|
|
11
|
+
target: RecordTarget.fromAlias(new CloudFrontTarget(props.distribution)),
|
|
12
|
+
recordName: props.domainName
|
|
13
|
+
});
|
|
14
|
+
this.aaaaRecord = new AaaaRecord(this, `AaaaRecord-${props.domainName}`, {
|
|
15
|
+
zone: props.hostedZone,
|
|
16
|
+
target: RecordTarget.fromAlias(new CloudFrontTarget(props.distribution)),
|
|
17
|
+
recordName: props.domainName
|
|
18
|
+
});
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
export function getHostedZone(scope, id, domainName) {
|
|
22
|
+
return HostedZone.fromLookup(scope, `getHostedZone-${domainName}`, { domainName });
|
|
23
|
+
}
|
|
24
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicm91dGU1My5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9jb25zdHJ1Y3RzL3JvdXRlNTMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQ0EsT0FBTyxFQUFFLE9BQU8sRUFBRSxVQUFVLEVBQUUsVUFBVSxFQUFlLFlBQVksRUFBRSxNQUFNLHlCQUF5QixDQUFDO0FBQ3JHLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLGlDQUFpQyxDQUFDO0FBQ25FLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxZQUFZLENBQUM7QUFPdkMsTUFBTSxPQUFPLGVBQWdCLFNBQVEsU0FBUztJQUM1QixPQUFPLENBQVU7SUFDakIsVUFBVSxDQUFhO0lBRXZDLFlBQVksS0FBZ0IsRUFBRSxFQUFVLEVBQUUsS0FBMkI7UUFDbkUsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsQ0FBQztRQUVqQixJQUFJLENBQUMsT0FBTyxHQUFHLElBQUksT0FBTyxDQUFDLElBQUksRUFBRSxXQUFXLEtBQUssQ0FBQyxVQUFVLEVBQUUsRUFBRTtZQUM5RCxJQUFJLEVBQUUsS0FBSyxDQUFDLFVBQVU7WUFDdEIsTUFBTSxFQUFFLFlBQVksQ0FBQyxTQUFTLENBQUMsSUFBSSxnQkFBZ0IsQ0FBQyxLQUFLLENBQUMsWUFBWSxDQUFDLENBQUM7WUFDeEUsVUFBVSxFQUFFLEtBQUssQ0FBQyxVQUFVO1NBQzdCLENBQUMsQ0FBQztRQUVILElBQUksQ0FBQyxVQUFVLEdBQUcsSUFBSSxVQUFVLENBQUMsSUFBSSxFQUFFLGNBQWMsS0FBSyxDQUFDLFVBQVUsRUFBRSxFQUFFO1lBQ3ZFLElBQUksRUFBRSxLQUFLLENBQUMsVUFBVTtZQUN0QixNQUFNLEVBQUUsWUFBWSxDQUFDLFNBQVMsQ0FBQyxJQUFJLGdCQUFnQixDQUFDLEtBQUssQ0FBQyxZQUFZLENBQUMsQ0FBQztZQUN4RSxVQUFVLEVBQUUsS0FBSyxDQUFDLFVBQVU7U0FDN0IsQ0FBQyxDQUFDO0lBQ0wsQ0FBQztDQUNGO0FBRUQsTUFBTSxVQUFVLGFBQWEsQ0FBQyxLQUFnQixFQUFFLEVBQVUsRUFBRSxVQUFrQjtJQUM1RSxPQUFPLFVBQVUsQ0FBQyxVQUFVLENBQUMsS0FBSyxFQUFFLGlCQUFpQixVQUFVLEVBQUUsRUFBRSxFQUFFLFVBQVUsRUFBRSxDQUFDLENBQUM7QUFDckYsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IElEaXN0cmlidXRpb24gfSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWNsb3VkZnJvbnRcIjtcbmltcG9ydCB7IEFSZWNvcmQsIEFhYWFSZWNvcmQsIEhvc3RlZFpvbmUsIElIb3N0ZWRab25lLCBSZWNvcmRUYXJnZXQgfSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLXJvdXRlNTNcIjtcbmltcG9ydCB7IENsb3VkRnJvbnRUYXJnZXQgfSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLXJvdXRlNTMtdGFyZ2V0c1wiO1xuaW1wb3J0IHsgQ29uc3RydWN0IH0gZnJvbSBcImNvbnN0cnVjdHNcIjtcblxuZXhwb3J0IGludGVyZmFjZSBDbG91ZEZyb250QWxpYXNQcm9wcyB7XG4gIGhvc3RlZFpvbmU6IElIb3N0ZWRab25lO1xuICBkb21haW5OYW1lOiBzdHJpbmc7XG4gIGRpc3RyaWJ1dGlvbjogSURpc3RyaWJ1dGlvbjsgXG59XG5leHBvcnQgY2xhc3MgQ2xvdWRGcm9udEFsaWFzIGV4dGVuZHMgQ29uc3RydWN0IHtcbiAgcHVibGljIHJlYWRvbmx5IGFSZWNvcmQ6IEFSZWNvcmQ7XG4gIHB1YmxpYyByZWFkb25seSBhYWFhUmVjb3JkOiBBYWFhUmVjb3JkO1xuXG4gIGNvbnN0cnVjdG9yKHNjb3BlOiBDb25zdHJ1Y3QsIGlkOiBzdHJpbmcsIHByb3BzOiBDbG91ZEZyb250QWxpYXNQcm9wcykge1xuICAgIHN1cGVyKHNjb3BlLCBpZCk7XG5cbiAgICB0aGlzLmFSZWNvcmQgPSBuZXcgQVJlY29yZCh0aGlzLCBgQVJlY29yZC0ke3Byb3BzLmRvbWFpbk5hbWV9YCwge1xuICAgICAgem9uZTogcHJvcHMuaG9zdGVkWm9uZSxcbiAgICAgIHRhcmdldDogUmVjb3JkVGFyZ2V0LmZyb21BbGlhcyhuZXcgQ2xvdWRGcm9udFRhcmdldChwcm9wcy5kaXN0cmlidXRpb24pKSxcbiAgICAgIHJlY29yZE5hbWU6IHByb3BzLmRvbWFpbk5hbWVcbiAgICB9KTtcblxuICAgIHRoaXMuYWFhYVJlY29yZCA9IG5ldyBBYWFhUmVjb3JkKHRoaXMsIGBBYWFhUmVjb3JkLSR7cHJvcHMuZG9tYWluTmFtZX1gLCB7XG4gICAgICB6b25lOiBwcm9wcy5ob3N0ZWRab25lLFxuICAgICAgdGFyZ2V0OiBSZWNvcmRUYXJnZXQuZnJvbUFsaWFzKG5ldyBDbG91ZEZyb250VGFyZ2V0KHByb3BzLmRpc3RyaWJ1dGlvbikpLFxuICAgICAgcmVjb3JkTmFtZTogcHJvcHMuZG9tYWluTmFtZVxuICAgIH0pO1xuICB9XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBnZXRIb3N0ZWRab25lKHNjb3BlOiBDb25zdHJ1Y3QsIGlkOiBzdHJpbmcsIGRvbWFpbk5hbWU6IHN0cmluZykge1xuICByZXR1cm4gSG9zdGVkWm9uZS5mcm9tTG9va3VwKHNjb3BlLCBgZ2V0SG9zdGVkWm9uZS0ke2RvbWFpbk5hbWV9YCwgeyBkb21haW5OYW1lIH0pO1xufSJdfQ==
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import { Construct } from 'constructs';
|
|
2
|
+
import { Bucket } from 'aws-cdk-lib/aws-s3';
|
|
3
|
+
export interface S3BucketProps {
|
|
4
|
+
/** The name of the S3 bucket */
|
|
5
|
+
bucketName?: string;
|
|
6
|
+
destroy?: boolean;
|
|
7
|
+
tags?: Record<string, string>;
|
|
8
|
+
}
|
|
9
|
+
export declare class S3Bucket extends Construct {
|
|
10
|
+
readonly bucket: Bucket;
|
|
11
|
+
constructor(scope: Construct, id: string, props: S3BucketProps);
|
|
12
|
+
}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
import { RemovalPolicy } from "aws-cdk-lib";
|
|
2
|
+
import { Construct } from 'constructs';
|
|
3
|
+
import { Bucket } from 'aws-cdk-lib/aws-s3';
|
|
4
|
+
import { tagResource } from '../lib/utils.js';
|
|
5
|
+
export class S3Bucket extends Construct {
|
|
6
|
+
bucket;
|
|
7
|
+
constructor(scope, id, props) {
|
|
8
|
+
super(scope, id);
|
|
9
|
+
this.bucket = new Bucket(this, 'S3Bucket', {
|
|
10
|
+
bucketName: props.bucketName,
|
|
11
|
+
removalPolicy: props.destroy !== undefined ? props.destroy ? RemovalPolicy.DESTROY : RemovalPolicy.RETAIN : undefined,
|
|
12
|
+
autoDeleteObjects: props.destroy === true ? true : undefined
|
|
13
|
+
});
|
|
14
|
+
tagResource(this.bucket, props.tags);
|
|
15
|
+
}
|
|
16
|
+
}
|
|
17
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiczMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvY29uc3RydWN0cy9zMy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sYUFBYSxDQUFDO0FBQzVDLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxZQUFZLENBQUM7QUFDdkMsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLG9CQUFvQixDQUFDO0FBQzVDLE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQVM5QyxNQUFNLE9BQU8sUUFBUyxTQUFRLFNBQVM7SUFDckIsTUFBTSxDQUFTO0lBRS9CLFlBQVksS0FBZ0IsRUFBRSxFQUFVLEVBQUUsS0FBb0I7UUFDNUQsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsQ0FBQztRQUVqQixJQUFJLENBQUMsTUFBTSxHQUFHLElBQUksTUFBTSxDQUFDLElBQUksRUFBRSxVQUFVLEVBQUU7WUFDekMsVUFBVSxFQUFFLEtBQUssQ0FBQyxVQUFVO1lBQzVCLGFBQWEsRUFBRSxLQUFLLENBQUMsT0FBTyxLQUFLLFNBQVMsQ0FBQyxDQUFDLENBQUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsYUFBYSxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsYUFBYSxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsU0FBUztZQUNySCxpQkFBaUIsRUFBRSxLQUFLLENBQUMsT0FBTyxLQUFLLElBQUksQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxTQUFTO1NBQzdELENBQUMsQ0FBQztRQUNILFdBQVcsQ0FBQyxJQUFJLENBQUMsTUFBTSxFQUFFLEtBQUssQ0FBQyxJQUFJLENBQUMsQ0FBQztJQUN2QyxDQUFDO0NBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBSZW1vdmFsUG9saWN5IH0gZnJvbSBcImF3cy1jZGstbGliXCI7XG5pbXBvcnQgeyBDb25zdHJ1Y3QgfSBmcm9tICdjb25zdHJ1Y3RzJztcbmltcG9ydCB7IEJ1Y2tldCB9IGZyb20gJ2F3cy1jZGstbGliL2F3cy1zMyc7XG5pbXBvcnQgeyB0YWdSZXNvdXJjZSB9IGZyb20gJy4uL2xpYi91dGlscy5qcyc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgUzNCdWNrZXRQcm9wcyB7XG4gIC8qKiBUaGUgbmFtZSBvZiB0aGUgUzMgYnVja2V0ICovXG4gIGJ1Y2tldE5hbWU/OiBzdHJpbmc7XG4gIGRlc3Ryb3k/OiBib29sZWFuO1xuICB0YWdzPzogUmVjb3JkPHN0cmluZywgc3RyaW5nPjtcbn1cblxuZXhwb3J0IGNsYXNzIFMzQnVja2V0IGV4dGVuZHMgQ29uc3RydWN0IHtcbiAgcHVibGljIHJlYWRvbmx5IGJ1Y2tldDogQnVja2V0O1xuXG4gIGNvbnN0cnVjdG9yKHNjb3BlOiBDb25zdHJ1Y3QsIGlkOiBzdHJpbmcsIHByb3BzOiBTM0J1Y2tldFByb3BzKSB7XG4gICAgc3VwZXIoc2NvcGUsIGlkKTtcblxuICAgIHRoaXMuYnVja2V0ID0gbmV3IEJ1Y2tldCh0aGlzLCAnUzNCdWNrZXQnLCB7XG4gICAgICBidWNrZXROYW1lOiBwcm9wcy5idWNrZXROYW1lLFxuICAgICAgcmVtb3ZhbFBvbGljeTogcHJvcHMuZGVzdHJveSAhPT0gdW5kZWZpbmVkID8gcHJvcHMuZGVzdHJveSA/IFJlbW92YWxQb2xpY3kuREVTVFJPWSA6IFJlbW92YWxQb2xpY3kuUkVUQUlOIDogdW5kZWZpbmVkLFxuICAgICAgYXV0b0RlbGV0ZU9iamVjdHM6IHByb3BzLmRlc3Ryb3kgPT09IHRydWUgPyB0cnVlIDogdW5kZWZpbmVkXG4gICAgfSk7XG4gICAgdGFnUmVzb3VyY2UodGhpcy5idWNrZXQsIHByb3BzLnRhZ3MpO1xuICB9XG59Il19
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
import { Construct } from 'constructs';
|
|
2
|
+
import { CfnWebACL } from 'aws-cdk-lib/aws-wafv2';
|
|
3
|
+
import type { IResourceProps } from '../lib/types.js';
|
|
4
|
+
export type WebAclManagedRule = 'AWSManagedRulesAmazonIpReputationList' | 'AWSManagedRulesAnonymousIpList' | 'AWSManagedRulesCommonRuleSet' | 'AWSManagedRulesKnownBadInputsRuleSet' | 'AWSManagedRulesSQLiRuleSet';
|
|
5
|
+
export interface CloudFrontWebAclProps extends IResourceProps {
|
|
6
|
+
name: string;
|
|
7
|
+
managedRules?: (WebAclManagedRule & string)[];
|
|
8
|
+
}
|
|
9
|
+
export declare class CloudFrontWebAcl extends Construct {
|
|
10
|
+
readonly acl: CfnWebACL;
|
|
11
|
+
constructor(scope: Construct, id: string, props: CloudFrontWebAclProps);
|
|
12
|
+
}
|
|
13
|
+
export interface ApiGatewayWebAclProps extends IResourceProps {
|
|
14
|
+
name: string;
|
|
15
|
+
aclScope?: 'CLOUDFRONT' | 'REGIONAL';
|
|
16
|
+
managedRules?: (WebAclManagedRule & string)[];
|
|
17
|
+
}
|
|
18
|
+
export declare class ApiGatewayWebAcl extends Construct {
|
|
19
|
+
readonly acl: CfnWebACL;
|
|
20
|
+
constructor(scope: Construct, id: string, props: ApiGatewayWebAclProps);
|
|
21
|
+
}
|
|
@@ -0,0 +1,86 @@
|
|
|
1
|
+
import { RemovalPolicy } from "aws-cdk-lib";
|
|
2
|
+
import { Construct } from 'constructs';
|
|
3
|
+
import { CfnWebACL } from 'aws-cdk-lib/aws-wafv2';
|
|
4
|
+
import { cleanResourceName, tagResource } from '../lib/utils.js';
|
|
5
|
+
export class CloudFrontWebAcl extends Construct {
|
|
6
|
+
acl;
|
|
7
|
+
constructor(scope, id, props) {
|
|
8
|
+
super(scope, id);
|
|
9
|
+
props.name = cleanResourceName(props.name);
|
|
10
|
+
if (!props.managedRules) {
|
|
11
|
+
props.managedRules = [
|
|
12
|
+
'AWSManagedRulesAmazonIpReputationList',
|
|
13
|
+
'AWSManagedRulesAnonymousIpList',
|
|
14
|
+
'AWSManagedRulesCommonRuleSet',
|
|
15
|
+
'AWSManagedRulesKnownBadInputsRuleSet'
|
|
16
|
+
];
|
|
17
|
+
}
|
|
18
|
+
this.acl = new CfnWebACL(this, 'CloudFrontWebAcl', {
|
|
19
|
+
scope: 'CLOUDFRONT',
|
|
20
|
+
name: props.name,
|
|
21
|
+
defaultAction: { allow: {} },
|
|
22
|
+
visibilityConfig: {
|
|
23
|
+
cloudWatchMetricsEnabled: true,
|
|
24
|
+
sampledRequestsEnabled: true,
|
|
25
|
+
metricName: `${props.name}-metrics`
|
|
26
|
+
},
|
|
27
|
+
rules: transformManagedRules(props.managedRules)
|
|
28
|
+
});
|
|
29
|
+
tagResource(this.acl, props.tags);
|
|
30
|
+
if (typeof props.destroy == 'boolean') {
|
|
31
|
+
this.acl.applyRemovalPolicy(props.destroy ? RemovalPolicy.DESTROY : RemovalPolicy.RETAIN);
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
export class ApiGatewayWebAcl extends Construct {
|
|
36
|
+
acl;
|
|
37
|
+
constructor(scope, id, props) {
|
|
38
|
+
super(scope, id);
|
|
39
|
+
props.name = cleanResourceName(props.name);
|
|
40
|
+
if (!props.managedRules) {
|
|
41
|
+
props.managedRules = [
|
|
42
|
+
'AWSManagedRulesAmazonIpReputationList',
|
|
43
|
+
'AWSManagedRulesAnonymousIpList',
|
|
44
|
+
'AWSManagedRulesCommonRuleSet',
|
|
45
|
+
'AWSManagedRulesKnownBadInputsRuleSet',
|
|
46
|
+
'AWSManagedRulesSQLiRuleSet'
|
|
47
|
+
];
|
|
48
|
+
}
|
|
49
|
+
this.acl = new CfnWebACL(this, 'ApiGatewayWebAcl', {
|
|
50
|
+
scope: props.aclScope ?? 'REGIONAL',
|
|
51
|
+
name: props.name,
|
|
52
|
+
defaultAction: { allow: {} },
|
|
53
|
+
visibilityConfig: {
|
|
54
|
+
cloudWatchMetricsEnabled: true,
|
|
55
|
+
sampledRequestsEnabled: true,
|
|
56
|
+
metricName: `${props.name}-metrics`
|
|
57
|
+
},
|
|
58
|
+
rules: transformManagedRules(props.managedRules)
|
|
59
|
+
});
|
|
60
|
+
tagResource(this.acl, props.tags);
|
|
61
|
+
if (typeof props.destroy == 'boolean') {
|
|
62
|
+
this.acl.applyRemovalPolicy(props.destroy ? RemovalPolicy.DESTROY : RemovalPolicy.RETAIN);
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
function transformManagedRules(rules) {
|
|
67
|
+
return rules.map((ruleName, i) => {
|
|
68
|
+
return {
|
|
69
|
+
name: `AWS-${ruleName}`,
|
|
70
|
+
priority: i,
|
|
71
|
+
statement: {
|
|
72
|
+
managedRuleGroupStatement: {
|
|
73
|
+
name: ruleName,
|
|
74
|
+
vendorName: 'AWS'
|
|
75
|
+
}
|
|
76
|
+
},
|
|
77
|
+
overrideAction: { none: {} },
|
|
78
|
+
visibilityConfig: {
|
|
79
|
+
cloudWatchMetricsEnabled: true,
|
|
80
|
+
sampledRequestsEnabled: true,
|
|
81
|
+
metricName: `AWS-${ruleName}`
|
|
82
|
+
}
|
|
83
|
+
};
|
|
84
|
+
});
|
|
85
|
+
}
|
|
86
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"waf.js","sourceRoot":"","sources":["../../src/constructs/waf.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,aAAa,EAAE,MAAM,aAAa,CAAC;AAC5C,OAAO,EAAE,SAAS,EAAE,MAAM,YAAY,CAAC;AACvC,OAAO,EAAE,SAAS,EAAE,MAAM,uBAAuB,CAAC;AAClD,OAAO,EAAE,iBAAiB,EAAE,WAAW,EAAE,MAAM,iBAAiB,CAAC;AAcjE,MAAM,OAAO,gBAAiB,SAAQ,SAAS;IAC7B,GAAG,CAAY;IAE/B,YAAY,KAAgB,EAAE,EAAU,EAAE,KAA4B;QACpE,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,KAAK,CAAC,IAAI,GAAG,iBAAiB,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QAE3C,IAAI,CAAC,KAAK,CAAC,YAAY,EAAE,CAAC;YACxB,KAAK,CAAC,YAAY,GAAG;gBACnB,uCAAuC;gBACvC,gCAAgC;gBAChC,8BAA8B;gBAC9B,sCAAsC;aACvC,CAAC;QACJ,CAAC;QAED,IAAI,CAAC,GAAG,GAAG,IAAI,SAAS,CAAC,IAAI,EAAE,kBAAkB,EAAE;YACjD,KAAK,EAAE,YAAY;YACnB,IAAI,EAAE,KAAK,CAAC,IAAI;YAChB,aAAa,EAAE,EAAE,KAAK,EAAE,EAAE,EAAE;YAC5B,gBAAgB,EAAE;gBAChB,wBAAwB,EAAE,IAAI;gBAC9B,sBAAsB,EAAE,IAAI;gBAC5B,UAAU,EAAE,GAAG,KAAK,CAAC,IAAI,UAAU;aACpC;YACD,KAAK,EAAE,qBAAqB,CAAC,KAAK,CAAC,YAAY,CAAC;SACjD,CAAC,CAAC;QACH,WAAW,CAAC,IAAI,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,CAAC,CAAC;QAClC,IAAI,OAAO,KAAK,CAAC,OAAO,IAAI,SAAS,EAAE,CAAC;YACtC,IAAI,CAAC,GAAG,CAAC,kBAAkB,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC,CAAC,aAAa,CAAC,MAAM,CAAC,CAAC;QAC5F,CAAC;IACH,CAAC;CACF;AAOD,MAAM,OAAO,gBAAiB,SAAQ,SAAS;IAC7B,GAAG,CAAY;IAC/B,YAAY,KAAgB,EAAE,EAAU,EAAE,KAA4B;QACpE,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,KAAK,CAAC,IAAI,GAAG,iBAAiB,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QAE3C,IAAI,CAAC,KAAK,CAAC,YAAY,EAAE,CAAC;YACxB,KAAK,CAAC,YAAY,GAAG;gBACnB,uCAAuC;gBACvC,gCAAgC;gBAChC,8BAA8B;gBAC9B,sCAAsC;gBACtC,4BAA4B;aAC7B,CAAC;QACJ,CAAC;QAED,IAAI,CAAC,GAAG,GAAG,IAAI,SAAS,CAAC,IAAI,EAAE,kBAAkB,EAAE;YACjD,KAAK,EAAE,KAAK,CAAC,QAAQ,IAAI,UAAU;YACnC,IAAI,EAAE,KAAK,CAAC,IAAI;YAChB,aAAa,EAAE,EAAE,KAAK,EAAE,EAAE,EAAE;YAC5B,gBAAgB,EAAE;gBAChB,wBAAwB,EAAE,IAAI;gBAC9B,sBAAsB,EAAE,IAAI;gBAC5B,UAAU,EAAE,GAAG,KAAK,CAAC,IAAI,UAAU;aACpC;YACD,KAAK,EAAE,qBAAqB,CAAC,KAAK,CAAC,YAAY,CAAC;SACjD,CAAC,CAAC;QACH,WAAW,CAAC,IAAI,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,CAAC,CAAC;QAClC,IAAI,OAAO,KAAK,CAAC,OAAO,IAAI,SAAS,EAAE,CAAC;YACtC,IAAI,CAAC,GAAG,CAAC,kBAAkB,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC,CAAC,aAAa,CAAC,MAAM,CAAC,CAAC;QAC5F,CAAC;IACH,CAAC;CACF;AAGD,SAAS,qBAAqB,CAAC,KAAqC;IAClE,OAAO,KAAK,CAAC,GAAG,CAAC,CAAC,QAAQ,EAAE,CAAC,EAAE,EAAE;QAC/B,OAAO;YACL,IAAI,EAAE,OAAO,QAAQ,EAAE;YACvB,QAAQ,EAAE,CAAC;YACX,SAAS,EAAE;gBACT,yBAAyB,EAAE;oBACzB,IAAI,EAAE,QAAQ;oBACd,UAAU,EAAE,KAAK;iBAClB;aACF;YACD,cAAc,EAAE,EAAE,IAAI,EAAE,EAAE,EAAE;YAC5B,gBAAgB,EAAE;gBAChB,wBAAwB,EAAE,IAAI;gBAC9B,sBAAsB,EAAE,IAAI;gBAC5B,UAAU,EAAE,OAAO,QAAQ,EAAE;aAC9B;SACwB,CAAC;IAC9B,CAAC,CAAC,CAAC;AACL,CAAC","sourcesContent":["import { RemovalPolicy } from \"aws-cdk-lib\";\nimport { Construct } from 'constructs';\nimport { CfnWebACL } from 'aws-cdk-lib/aws-wafv2';\nimport { cleanResourceName, tagResource } from '../lib/utils.js';\nimport type { IResourceProps } from '../lib/types.js';\n\nexport type WebAclManagedRule = \n  'AWSManagedRulesAmazonIpReputationList' |\n  'AWSManagedRulesAnonymousIpList' |\n  'AWSManagedRulesCommonRuleSet' |\n  'AWSManagedRulesKnownBadInputsRuleSet' |\n  'AWSManagedRulesSQLiRuleSet';\n\nexport interface CloudFrontWebAclProps extends IResourceProps {\n  name: string;\n  managedRules?: (WebAclManagedRule & string)[];\n}\nexport class CloudFrontWebAcl extends Construct {\n  public readonly acl: CfnWebACL;\n\n  constructor(scope: Construct, id: string, props: CloudFrontWebAclProps) {\n    super(scope, id);\n\n    props.name = cleanResourceName(props.name);\n\n    if (!props.managedRules) {\n      props.managedRules = [\n        'AWSManagedRulesAmazonIpReputationList',\n        'AWSManagedRulesAnonymousIpList',\n        'AWSManagedRulesCommonRuleSet',\n        'AWSManagedRulesKnownBadInputsRuleSet'\n      ];\n    }\n\n    this.acl = new CfnWebACL(this, 'CloudFrontWebAcl', {\n      scope: 'CLOUDFRONT',\n      name: props.name,\n      defaultAction: { allow: {} },\n      visibilityConfig: {\n        cloudWatchMetricsEnabled: true,\n        sampledRequestsEnabled: true,\n        metricName: `${props.name}-metrics`\n      },\n      rules: transformManagedRules(props.managedRules)\n    });\n    tagResource(this.acl, props.tags);\n    if (typeof props.destroy == 'boolean') {\n      this.acl.applyRemovalPolicy(props.destroy ? RemovalPolicy.DESTROY : RemovalPolicy.RETAIN);\n    } \n  }\n}\n\nexport interface ApiGatewayWebAclProps extends IResourceProps {\n  name: string;\n  aclScope?: 'CLOUDFRONT' | 'REGIONAL';\n  managedRules?: (WebAclManagedRule & string)[];\n}\nexport class ApiGatewayWebAcl extends Construct {\n  public readonly acl: CfnWebACL;\n  constructor(scope: Construct, id: string, props: ApiGatewayWebAclProps) {\n    super(scope, id);\n\n    props.name = cleanResourceName(props.name);\n\n    if (!props.managedRules) {\n      props.managedRules = [\n        'AWSManagedRulesAmazonIpReputationList',\n        'AWSManagedRulesAnonymousIpList',\n        'AWSManagedRulesCommonRuleSet',\n        'AWSManagedRulesKnownBadInputsRuleSet',\n        'AWSManagedRulesSQLiRuleSet'\n      ];\n    }\n\n    this.acl = new CfnWebACL(this, 'ApiGatewayWebAcl', {\n      scope: props.aclScope ?? 'REGIONAL',\n      name: props.name,\n      defaultAction: { allow: {} },\n      visibilityConfig: {\n        cloudWatchMetricsEnabled: true,\n        sampledRequestsEnabled: true,\n        metricName: `${props.name}-metrics`\n      },\n      rules: transformManagedRules(props.managedRules)\n    });\n    tagResource(this.acl, props.tags);\n    if (typeof props.destroy == 'boolean') {\n      this.acl.applyRemovalPolicy(props.destroy ? RemovalPolicy.DESTROY : RemovalPolicy.RETAIN);\n    } \n  }\n}\n\n\nfunction transformManagedRules(rules: (WebAclManagedRule | string)[]): CfnWebACL.RuleProperty[] {\n  return rules.map((ruleName, i) => {\n    return {\n      name: `AWS-${ruleName}`,\n      priority: i,\n      statement: {\n        managedRuleGroupStatement: {\n          name: ruleName,\n          vendorName: 'AWS'\n        }\n      },\n      overrideAction: { none: {} },\n      visibilityConfig: { \n        cloudWatchMetricsEnabled: true,\n        sampledRequestsEnabled: true,\n        metricName: `AWS-${ruleName}`\n      }\n    } as CfnWebACL.RuleProperty;\n  });\n}"]}
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
import { IConstruct } from 'constructs';
|
|
2
|
+
export declare function domainAsId(domain: string): string;
|
|
3
|
+
export declare function cleanResourceName(name: string): string;
|
|
4
|
+
export declare function tagResource(resource: IConstruct, tags?: Record<string, string>): void;
|
|
5
|
+
export declare function mergeTags(...tags: (Record<string, string> | null | undefined)[]): Record<string, string>;
|