truemark-cdk-lib 1.3.0 → 1.3.1-alpha.1
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/aws-cdk/lib/extended-construct.d.ts +2 -1
- package/aws-cdk/lib/extended-construct.js +2 -2
- package/aws-cdk/lib/extended-stack.d.ts +2 -1
- package/aws-cdk/lib/extended-stack.js +2 -2
- package/aws-cdk/lib/extended-stage.d.ts +2 -1
- package/aws-cdk/lib/extended-stage.js +2 -2
- package/{aws-website → aws-patterns-website}/lib/website.d.ts +11 -5
- package/aws-patterns-website/lib/website.js +188 -0
- package/aws-patterns-website/lib/website.test.js +60 -0
- package/aws-s3/lib/cloud-front-bucket.d.ts +3 -2
- package/aws-s3/lib/cloud-front-bucket.js +5 -4
- package/index.d.ts +3 -3
- package/index.js +4 -4
- package/package.json +4 -4
- package/aws-website/lib/website.js +0 -187
- package/aws-website/lib/website.test.js +0 -60
- /package/{aws-website → aws-patterns-website}/index.d.ts +0 -0
- /package/{aws-website → aws-patterns-website}/index.js +0 -0
- /package/{aws-website → aws-patterns-website}/lib/index.d.ts +0 -0
- /package/{aws-website → aws-patterns-website}/lib/index.js +0 -0
- /package/{aws-website → aws-patterns-website}/lib/website.test.d.ts +0 -0
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import { Construct } from "constructs";
|
|
2
|
-
import { StandardTagsProps } from "./standard-tags";
|
|
2
|
+
import { StandardTags, StandardTagsProps } from "./standard-tags";
|
|
3
3
|
/**
|
|
4
4
|
* Props for ExtendedConstruct.
|
|
5
5
|
*/
|
|
@@ -10,5 +10,6 @@ export interface ExtendedConstructProps {
|
|
|
10
10
|
readonly standardTags?: StandardTagsProps;
|
|
11
11
|
}
|
|
12
12
|
export declare class ExtendedConstruct extends Construct {
|
|
13
|
+
readonly standardTags: StandardTags;
|
|
13
14
|
constructor(scope: Construct, id: string, props?: ExtendedConstructProps);
|
|
14
15
|
}
|
|
@@ -7,8 +7,8 @@ class ExtendedConstruct extends constructs_1.Construct {
|
|
|
7
7
|
constructor(scope, id, props) {
|
|
8
8
|
super(scope, id);
|
|
9
9
|
// Setup standard tags
|
|
10
|
-
new standard_tags_1.StandardTags(this, props === null || props === void 0 ? void 0 : props.standardTags);
|
|
10
|
+
this.standardTags = new standard_tags_1.StandardTags(this, props === null || props === void 0 ? void 0 : props.standardTags);
|
|
11
11
|
}
|
|
12
12
|
}
|
|
13
13
|
exports.ExtendedConstruct = ExtendedConstruct;
|
|
14
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
14
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXh0ZW5kZWQtY29uc3RydWN0LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiZXh0ZW5kZWQtY29uc3RydWN0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLDJDQUFxQztBQUNyQyxtREFBZ0U7QUFZaEUsTUFBYSxpQkFBa0IsU0FBUSxzQkFBUztJQUc5QyxZQUFZLEtBQWdCLEVBQUUsRUFBVSxFQUFFLEtBQThCO1FBQ3RFLEtBQUssQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDLENBQUM7UUFFakIsc0JBQXNCO1FBQ3RCLElBQUksQ0FBQyxZQUFZLEdBQUcsSUFBSSw0QkFBWSxDQUFDLElBQUksRUFBRSxLQUFLLGFBQUwsS0FBSyx1QkFBTCxLQUFLLENBQUUsWUFBWSxDQUFDLENBQUM7SUFDbEUsQ0FBQztDQUNGO0FBVEQsOENBU0MiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge0NvbnN0cnVjdH0gZnJvbSBcImNvbnN0cnVjdHNcIjtcbmltcG9ydCB7U3RhbmRhcmRUYWdzLCBTdGFuZGFyZFRhZ3NQcm9wc30gZnJvbSBcIi4vc3RhbmRhcmQtdGFnc1wiO1xuXG4vKipcbiAqIFByb3BzIGZvciBFeHRlbmRlZENvbnN0cnVjdC5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBFeHRlbmRlZENvbnN0cnVjdFByb3BzIHtcbiAgLyoqXG4gICAqIFNldHMgc3RhbmRhcmQgdGFncyBmb3IgdGhpcyBDb25zdHJ1Y3QuIFZhbHVlcyBzZXQgaGVyZSB3aWxsIG92ZXJyaWRlIHRob3NlIHNldCBvbiB0aGUgcGFyZW50IHNjb3BlLlxuICAgKi9cbiAgcmVhZG9ubHkgc3RhbmRhcmRUYWdzPzogU3RhbmRhcmRUYWdzUHJvcHM7XG59XG5cbmV4cG9ydCBjbGFzcyBFeHRlbmRlZENvbnN0cnVjdCBleHRlbmRzIENvbnN0cnVjdCB7XG5cbiAgcmVhZG9ubHkgc3RhbmRhcmRUYWdzOiBTdGFuZGFyZFRhZ3M7XG4gIGNvbnN0cnVjdG9yKHNjb3BlOiBDb25zdHJ1Y3QsIGlkOiBzdHJpbmcsIHByb3BzPzogRXh0ZW5kZWRDb25zdHJ1Y3RQcm9wcykge1xuICAgIHN1cGVyKHNjb3BlLCBpZCk7XG5cbiAgICAvLyBTZXR1cCBzdGFuZGFyZCB0YWdzXG4gICAgdGhpcy5zdGFuZGFyZFRhZ3MgPSBuZXcgU3RhbmRhcmRUYWdzKHRoaXMsIHByb3BzPy5zdGFuZGFyZFRhZ3MpO1xuICB9XG59XG4iXX0=
|
|
@@ -3,7 +3,7 @@ import { ParameterStore, ParameterStoreOptions } from "../../aws-ssm";
|
|
|
3
3
|
import { Construct } from "constructs";
|
|
4
4
|
import { StringParameter } from "aws-cdk-lib/aws-ssm";
|
|
5
5
|
import { DashboardRenderingPreference, MonitoringFacade } from "cdk-monitoring-constructs";
|
|
6
|
-
import { StandardTagsProps } from "./standard-tags";
|
|
6
|
+
import { StandardTags, StandardTagsProps } from "./standard-tags";
|
|
7
7
|
/**
|
|
8
8
|
* Options for ExtStack.
|
|
9
9
|
*/
|
|
@@ -89,6 +89,7 @@ export declare class ExtendedStack extends Stack {
|
|
|
89
89
|
protected readonly parameterExports: ParameterStore;
|
|
90
90
|
readonly parameterExportOptions: ParameterStoreOptions;
|
|
91
91
|
readonly monitoringFacade?: MonitoringFacade;
|
|
92
|
+
readonly standardTags: StandardTags;
|
|
92
93
|
constructor(scope: Construct, id: string, props?: ExtendedStackProps);
|
|
93
94
|
/**
|
|
94
95
|
* Helper method to exports a parameter as an SSM Parameter.
|
|
@@ -39,7 +39,7 @@ class ExtendedStack extends aws_cdk_lib_1.Stack {
|
|
|
39
39
|
});
|
|
40
40
|
}
|
|
41
41
|
// Setup standard tags
|
|
42
|
-
new standard_tags_1.StandardTags(this, props === null || props === void 0 ? void 0 : props.standardTags);
|
|
42
|
+
this.standardTags = new standard_tags_1.StandardTags(this, props === null || props === void 0 ? void 0 : props.standardTags);
|
|
43
43
|
}
|
|
44
44
|
/**
|
|
45
45
|
* Helper method to exports a parameter as an SSM Parameter.
|
|
@@ -63,4 +63,4 @@ class ExtendedStack extends aws_cdk_lib_1.Stack {
|
|
|
63
63
|
}
|
|
64
64
|
}
|
|
65
65
|
exports.ExtendedStack = ExtendedStack;
|
|
66
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
66
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"extended-stack.js","sourceRoot":"","sources":["extended-stack.ts"],"names":[],"mappings":";;;AAAA,6CAA0E;AAC1E,2CAAoE;AAGpE,yEAImC;AACnC,mDAGyB;AA6FzB;;GAEG;AACH,MAAa,aAAc,SAAQ,mBAAK;IAOtC,YAAY,KAAgB,EAAE,EAAU,EAAE,KAA0B;;QAClE,KAAK,CAAC,KAAK,EAAE,EAAE,EAAE,KAAK,CAAC,CAAC;QAExB,MAAM,SAAS,SAAG,mBAAK,CAAC,EAAE,CAAC,IAAI,CAAC,0CAAE,SAAS,CAAC;QAE5C,IAAI,CAAC,sBAAsB,GAAG;YAC5B,MAAM,QAAE,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,sBAAsB,mCAAI,CAAC,SAAS,KAAK,SAAS,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,SAAS,EAAE,CAAC,GAAG,IAAI,EAAE,WAAW;YAC7G,MAAM,EAAE,IAAI,CAAC,MAAM;SACpB,CAAC;QACF,IAAI,CAAC,gBAAgB,GAAG,IAAI,wBAAc,CAAC,IAAI,EAAE,kBAAkB,EAAE,IAAI,CAAC,sBAAsB,CAAC,CAAC;QAElG,UAAI,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,sBAAsB,mCAAI,IAAI,EAAE;YACzC,MAAM,gBAAgB,GAAG,IAAI,mDAAuB,CAAC,IAAI,EAAE,kBAAkB,EAAE;gBAC7E,mBAAmB,QAAE,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,mBAAmB,mCAAI,IAAI,CAAC,SAAS;gBACjE,oBAAoB,QAAE,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,oBAAoB,mCAAI,KAAK;gBAC1D,eAAe,QAAE,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,eAAe,mCAAI,KAAK;gBAChD,sBAAsB,QAAE,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,sBAAsB,mCAAI,KAAK;gBAC9D,mBAAmB,QAAE,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,4BAA4B,mCAAI,wDAA4B,CAAC,gBAAgB;aAC1G,CAAC,CAAC;YAEH,IAAI,CAAC,gBAAgB,GAAG,IAAI,4CAAgB,CAAC,IAAI,EAAE,YAAY,EAAE;gBAC/D,qBAAqB,EAAE;oBACrB,SAAS,EAAE,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,eAAe;oBACjC,MAAM,EAAE,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,YAAY;iBAC5B;gBACD,oBAAoB,EAAE;oBACpB,cAAc,QAAE,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,mBAAmB,mCAAI,IAAI;oBAClD,eAAe,EAAE,IAAI,CAAC,SAAS;iBAChC;gBACD,gBAAgB,EAAE,gBAAgB;aACnC,CAAC,CAAC;SACJ;QAED,sBAAsB;QACtB,IAAI,CAAC,YAAY,GAAG,IAAI,4BAAY,CAAC,IAAI,EAAE,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,YAAY,CAAC,CAAC;IAClE,CAAC;IAED;;;;;OAKG;IACH,eAAe,CAAC,IAAY,EAAE,KAAa;QACzC,OAAO,IAAI,CAAC,gBAAgB,CAAC,KAAK,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;IAClD,CAAC;IAED;;;;;OAKG;IACH,eAAe,CAAC,IAAY,EAAE,KAAa;QACzC,OAAO,IAAI,uBAAS,CAAC,IAAI,EAAE,IAAI,EAAE;YAC/B,KAAK;SACN,CAAC,CAAC;IACL,CAAC;CACF;AAjED,sCAiEC","sourcesContent":["import {CfnOutput, Duration, Stack, StackProps, Stage} from \"aws-cdk-lib\";\nimport {ParameterStore, ParameterStoreOptions} from \"../../aws-ssm\";\nimport {Construct} from \"constructs\";\nimport {StringParameter} from \"aws-cdk-lib/aws-ssm\";\nimport {\n  DashboardRenderingPreference,\n  DefaultDashboardFactory,\n  MonitoringFacade\n} from \"cdk-monitoring-constructs\";\nimport {\n  StandardTags,\n  StandardTagsProps\n} from \"./standard-tags\";\n\n/**\n * Options for ExtStack.\n */\nexport interface ExtendedStackOptions {\n\n  /**\n   * Prefix to be used by parameter exports to be stored in the\n   * Systems Manager Parameter Store.\n   *\n   * @default ${stageName}/${stackId}/Exports/\n   */\n  readonly parameterExportsPrefix?: string;\n\n  /**\n   * Flag to determine if a default MonitoringFacade and\n   * DefaultDashboardFactory should be created in this stack.\n   *\n   * @default true\n   */\n  readonly createMonitoringFacade?: boolean;\n\n  /**\n   * Prefix added to each dashboard name.\n   *\n   * @default stackName\n   */\n  readonly dashboardNamePrefix?: string;\n\n  /**\n   * Flag to determine if the default dashboard should be created.\n   *\n   * @default false\n   */\n  readonly createDashboard?: boolean\n\n  /**\n   * Flag to determine if the summary dashboard should be created.\n   *\n   * @default false\n   */\n  readonly createSummaryDashboard?: boolean;\n\n  /**\n   * Flag to determine if the alarm dashboard should be created.\n   *\n   * @default false\n   */\n  readonly createAlarmDashboard?: boolean;\n\n  /**\n   * Rendering preference to dashboards.\n   *\n   * @default DashboardRenderingPreference.INTERACTIVE_ONLY\n   */\n  readonly dashboardRenderingPreference?: DashboardRenderingPreference;\n\n  /**\n   * Default namespace for metrics.\n   */\n  readonly metricNamespace?: string;\n\n  /**\n   * Default metric period\n   */\n  readonly metricPeriod?: Duration;\n\n  /**\n   * Prefix for generated alarms.\n   *\n   * @default stackName\n   */\n  readonly alarmNamePrefix?: string;\n\n  /**\n   * Enables alarm actions.\n   *\n   * @default true\n   */\n  readonly alarmActionsEnabled?: boolean;\n\n  /**\n   * Sets standard tags for this Stack.\n   */\n  readonly standardTags?: StandardTagsProps;\n}\n\n/**\n * Properties for ExtStack.\n */\nexport interface ExtendedStackProps extends ExtendedStackOptions, StackProps {}\n\n/**\n * Extended version of Stack providing functionality for parameter exports and monitoring.\n */\nexport class ExtendedStack extends Stack {\n\n  protected readonly parameterExports: ParameterStore;\n  readonly parameterExportOptions: ParameterStoreOptions;\n  readonly monitoringFacade?: MonitoringFacade;\n  readonly standardTags: StandardTags;\n\n  constructor(scope: Construct, id: string, props?: ExtendedStackProps) {\n    super(scope, id, props);\n\n    const stageName = Stage.of(this)?.stageName;\n\n    this.parameterExportOptions = {\n      prefix: props?.parameterExportsPrefix ?? (stageName === undefined ? \"\" : `/${stageName}`) + `/${id}/Exports/`,\n      region: this.region\n    };\n    this.parameterExports = new ParameterStore(this, \"ParameterExports\", this.parameterExportOptions);\n\n    if (props?.createMonitoringFacade ?? true) {\n      const dashboardFactory = new DefaultDashboardFactory(this, \"DashboardFactory\", {\n        dashboardNamePrefix: props?.dashboardNamePrefix ?? this.stackName,\n        createAlarmDashboard: props?.createAlarmDashboard ?? false,\n        createDashboard: props?.createDashboard ?? false,\n        createSummaryDashboard: props?.createSummaryDashboard ?? false,\n        renderingPreference: props?.dashboardRenderingPreference ?? DashboardRenderingPreference.INTERACTIVE_ONLY\n      });\n\n      this.monitoringFacade = new MonitoringFacade(this, \"Monitoring\", {\n        metricFactoryDefaults: {\n          namespace: props?.metricNamespace,\n          period: props?.metricPeriod\n        },\n        alarmFactoryDefaults: {\n          actionsEnabled: props?.alarmActionsEnabled ?? true,\n          alarmNamePrefix: this.stackName\n        },\n        dashboardFactory: dashboardFactory\n      });\n    }\n\n    // Setup standard tags\n    this.standardTags = new StandardTags(this, props?.standardTags);\n  }\n\n  /**\n   * Helper method to exports a parameter as an SSM Parameter.\n   *\n   * @param name the parameter name\n   * @param value the parameter value\n   */\n  exportParameter(name: string, value: string): StringParameter {\n    return this.parameterExports.write(name, value);\n  }\n\n  /**\n   * Helper method to output a parameter as a CfnOutput.\n   *\n   * @param name the parameter name\n   * @param value the parameter value\n   */\n  outputParameter(name: string, value: string): CfnOutput {\n    return new CfnOutput(this, name, {\n      value\n    });\n  }\n}\n"]}
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import { Stage, StageProps } from "aws-cdk-lib";
|
|
2
|
-
import { StandardTagsProps } from "./standard-tags";
|
|
2
|
+
import { StandardTags, StandardTagsProps } from "./standard-tags";
|
|
3
3
|
import { Construct } from "constructs";
|
|
4
4
|
export interface ExtendedStageOptions {
|
|
5
5
|
readonly standardTags?: StandardTagsProps;
|
|
@@ -7,5 +7,6 @@ export interface ExtendedStageOptions {
|
|
|
7
7
|
export interface ExtendedStageProps extends StageProps, ExtendedStageOptions {
|
|
8
8
|
}
|
|
9
9
|
export declare class ExtendedStage extends Stage {
|
|
10
|
+
readonly standardTags: StandardTags;
|
|
10
11
|
constructor(scope: Construct, id: string, props?: ExtendedStageProps);
|
|
11
12
|
}
|
|
@@ -7,8 +7,8 @@ class ExtendedStage extends aws_cdk_lib_1.Stage {
|
|
|
7
7
|
constructor(scope, id, props) {
|
|
8
8
|
super(scope, id, props);
|
|
9
9
|
// Setup standard tags
|
|
10
|
-
new standard_tags_1.StandardTags(this, props === null || props === void 0 ? void 0 : props.standardTags);
|
|
10
|
+
this.standardTags = new standard_tags_1.StandardTags(this, props === null || props === void 0 ? void 0 : props.standardTags);
|
|
11
11
|
}
|
|
12
12
|
}
|
|
13
13
|
exports.ExtendedStage = ExtendedStage;
|
|
14
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
14
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXh0ZW5kZWQtc3RhZ2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJleHRlbmRlZC1zdGFnZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSw2Q0FBOEM7QUFDOUMsbURBQWdFO0FBU2hFLE1BQWEsYUFBYyxTQUFRLG1CQUFLO0lBR3RDLFlBQVksS0FBZ0IsRUFBRSxFQUFVLEVBQUUsS0FBMEI7UUFDbEUsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLEVBQUUsS0FBSyxDQUFDLENBQUM7UUFFeEIsc0JBQXNCO1FBQ3RCLElBQUksQ0FBQyxZQUFZLEdBQUcsSUFBSSw0QkFBWSxDQUFDLElBQUksRUFBRSxLQUFLLGFBQUwsS0FBSyx1QkFBTCxLQUFLLENBQUUsWUFBWSxDQUFDLENBQUM7SUFDbEUsQ0FBQztDQUNGO0FBVEQsc0NBU0MiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge1N0YWdlLCBTdGFnZVByb3BzfSBmcm9tIFwiYXdzLWNkay1saWJcIjtcbmltcG9ydCB7U3RhbmRhcmRUYWdzLCBTdGFuZGFyZFRhZ3NQcm9wc30gZnJvbSBcIi4vc3RhbmRhcmQtdGFnc1wiO1xuaW1wb3J0IHtDb25zdHJ1Y3R9IGZyb20gXCJjb25zdHJ1Y3RzXCI7XG5cbmV4cG9ydCBpbnRlcmZhY2UgRXh0ZW5kZWRTdGFnZU9wdGlvbnMge1xuICByZWFkb25seSBzdGFuZGFyZFRhZ3M/OiBTdGFuZGFyZFRhZ3NQcm9wcztcbn1cblxuZXhwb3J0IGludGVyZmFjZSBFeHRlbmRlZFN0YWdlUHJvcHMgZXh0ZW5kcyBTdGFnZVByb3BzLCBFeHRlbmRlZFN0YWdlT3B0aW9ucyB7fVxuXG5leHBvcnQgY2xhc3MgRXh0ZW5kZWRTdGFnZSBleHRlbmRzIFN0YWdlIHtcblxuICByZWFkb25seSBzdGFuZGFyZFRhZ3M6IFN0YW5kYXJkVGFncztcbiAgY29uc3RydWN0b3Ioc2NvcGU6IENvbnN0cnVjdCwgaWQ6IHN0cmluZywgcHJvcHM/OiBFeHRlbmRlZFN0YWdlUHJvcHMpIHtcbiAgICBzdXBlcihzY29wZSwgaWQsIHByb3BzKTtcblxuICAgIC8vIFNldHVwIHN0YW5kYXJkIHRhZ3NcbiAgICB0aGlzLnN0YW5kYXJkVGFncyA9IG5ldyBTdGFuZGFyZFRhZ3ModGhpcywgcHJvcHM/LnN0YW5kYXJkVGFncyk7XG4gIH1cbn1cbiJdfQ==
|
|
@@ -5,6 +5,7 @@ import { ARecord } from "aws-cdk-lib/aws-route53";
|
|
|
5
5
|
import { BucketDeployment } from "aws-cdk-lib/aws-s3-deployment";
|
|
6
6
|
import { BundlingOptions, Duration } from "aws-cdk-lib";
|
|
7
7
|
import { DomainNameProps } from "../../aws-route53";
|
|
8
|
+
import { ExtendedConstruct, ExtendedConstructProps } from "../../aws-cdk";
|
|
8
9
|
export declare enum SourceType {
|
|
9
10
|
Custom = "Custom",
|
|
10
11
|
Hugo = "Hugo",
|
|
@@ -15,9 +16,14 @@ export declare enum SourceType {
|
|
|
15
16
|
* Properties for WebsiteDomainName.
|
|
16
17
|
*/
|
|
17
18
|
export interface WebsiteDomainNameProps extends DomainNameProps {
|
|
19
|
+
/**
|
|
20
|
+
* Whether to create a Route53 record for this domain name. Default is true.
|
|
21
|
+
*
|
|
22
|
+
* @default - true
|
|
23
|
+
*/
|
|
18
24
|
readonly createRecord?: boolean;
|
|
19
25
|
}
|
|
20
|
-
export interface WebsiteProps {
|
|
26
|
+
export interface WebsiteProps extends ExtendedConstructProps {
|
|
21
27
|
/**
|
|
22
28
|
* Bucket to use to store website content. If one is not provided, one will be generated.
|
|
23
29
|
*/
|
|
@@ -36,15 +42,15 @@ export interface WebsiteProps {
|
|
|
36
42
|
*/
|
|
37
43
|
readonly domainNames?: WebsiteDomainNameProps[];
|
|
38
44
|
/**
|
|
39
|
-
* Redirect traffic to the first domain in the list of domainNames.
|
|
45
|
+
* Redirect traffic to the first domain in the list of domainNames. Defaults to true.
|
|
40
46
|
*
|
|
41
|
-
* @default true
|
|
47
|
+
* @default - true
|
|
42
48
|
*/
|
|
43
49
|
readonly redirectToApexDomain?: boolean;
|
|
44
50
|
/**
|
|
45
51
|
* Price class for the CloudFront distribution.
|
|
46
52
|
*
|
|
47
|
-
* @default PriceClass.PRICE_CLASS_100
|
|
53
|
+
* @default - PriceClass.PRICE_CLASS_100
|
|
48
54
|
*/
|
|
49
55
|
readonly priceClass?: PriceClass;
|
|
50
56
|
/**
|
|
@@ -81,7 +87,7 @@ export interface WebsiteProps {
|
|
|
81
87
|
*/
|
|
82
88
|
readonly sourceBundlingOptions?: BundlingOptions;
|
|
83
89
|
}
|
|
84
|
-
export declare class Website extends
|
|
90
|
+
export declare class Website extends ExtendedConstruct {
|
|
85
91
|
/**
|
|
86
92
|
* The default bundling options for SourceType.Hugo.
|
|
87
93
|
*/
|
|
@@ -0,0 +1,188 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.Website = exports.SourceType = void 0;
|
|
4
|
+
const aws_s3_1 = require("aws-cdk-lib/aws-s3");
|
|
5
|
+
const aws_cloudfront_1 = require("aws-cdk-lib/aws-cloudfront");
|
|
6
|
+
const aws_cloudfront_origins_1 = require("aws-cdk-lib/aws-cloudfront-origins");
|
|
7
|
+
const aws_certificatemanager_1 = require("aws-cdk-lib/aws-certificatemanager");
|
|
8
|
+
const aws_route53_1 = require("aws-cdk-lib/aws-route53");
|
|
9
|
+
const aws_route53_targets_1 = require("aws-cdk-lib/aws-route53-targets");
|
|
10
|
+
const aws_s3_deployment_1 = require("aws-cdk-lib/aws-s3-deployment");
|
|
11
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
12
|
+
const aws_route53_2 = require("../../aws-route53");
|
|
13
|
+
const aws_cdk_1 = require("../../aws-cdk");
|
|
14
|
+
const truemark_1 = require("../../truemark");
|
|
15
|
+
var SourceType;
|
|
16
|
+
(function (SourceType) {
|
|
17
|
+
SourceType["Custom"] = "Custom";
|
|
18
|
+
SourceType["Hugo"] = "Hugo";
|
|
19
|
+
SourceType["NpmDist"] = "NpmDist";
|
|
20
|
+
SourceType["Static"] = "Static";
|
|
21
|
+
})(SourceType = exports.SourceType || (exports.SourceType = {}));
|
|
22
|
+
class Website extends aws_cdk_1.ExtendedConstruct {
|
|
23
|
+
constructor(scope, id, props) {
|
|
24
|
+
var _a, _b, _c, _d, _e, _f, _g, _h, _j;
|
|
25
|
+
super(scope, id, { standardTags: aws_cdk_1.StandardTags.merge(props.standardTags, truemark_1.LibStandardTags) });
|
|
26
|
+
const domainNames = aws_route53_2.DomainName.fromProps(props.domainNames);
|
|
27
|
+
let certificate = undefined;
|
|
28
|
+
if (props.certificateArn !== undefined) {
|
|
29
|
+
certificate = aws_certificatemanager_1.Certificate.fromCertificateArn(this, "Certificate", props.certificateArn);
|
|
30
|
+
}
|
|
31
|
+
else if (domainNames.length > 0) {
|
|
32
|
+
certificate = new aws_certificatemanager_1.Certificate(this, "Certificate", {
|
|
33
|
+
domainName: domainNames[0].toString(),
|
|
34
|
+
subjectAlternativeNames: aws_route53_2.DomainName.toStrings(domainNames).slice(1),
|
|
35
|
+
validation: aws_certificatemanager_1.CertificateValidation.fromDnsMultiZone(aws_route53_2.DomainName.toZoneMap(this, domainNames))
|
|
36
|
+
});
|
|
37
|
+
}
|
|
38
|
+
this.bucket = (_a = props.bucket) !== null && _a !== void 0 ? _a : new aws_s3_1.Bucket(this, "Bucket", {
|
|
39
|
+
encryption: aws_s3_1.BucketEncryption.S3_MANAGED,
|
|
40
|
+
removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY,
|
|
41
|
+
autoDeleteObjects: true
|
|
42
|
+
});
|
|
43
|
+
this.fallbackBucket = props.fallbackBucket;
|
|
44
|
+
this.originAccessIdentity = new aws_cloudfront_1.OriginAccessIdentity(this, "Access");
|
|
45
|
+
this.bucket.grantRead(this.originAccessIdentity);
|
|
46
|
+
if (this.fallbackBucket) {
|
|
47
|
+
this.fallbackBucket.grantRead(this.originAccessIdentity);
|
|
48
|
+
}
|
|
49
|
+
this.apexDomain = domainNames.length > 0 ? domainNames[0].toString() : "";
|
|
50
|
+
this.viewerRequestFunction = new aws_cloudfront_1.Function(this, "ViewerRequestFunction", {
|
|
51
|
+
code: aws_cloudfront_1.FunctionCode.fromInline(`
|
|
52
|
+
function handler(event) {
|
|
53
|
+
var host = event.request.headers.host.value;
|
|
54
|
+
var uri = event.request.uri;
|
|
55
|
+
var matchApex = MATCH_APEX;
|
|
56
|
+
if (matchApex && host !== "APEX_DOMAIN") {
|
|
57
|
+
return {
|
|
58
|
+
statusCode: 301,
|
|
59
|
+
statusDescription: "Permanently moved",
|
|
60
|
+
headers: {
|
|
61
|
+
"location": { "value": "https://APEX_DOMAIN" + uri }
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
if (uri.endsWith("/")) {
|
|
66
|
+
event.request.uri = uri + "index.html";
|
|
67
|
+
}
|
|
68
|
+
return event.request;
|
|
69
|
+
}`
|
|
70
|
+
.replace(/APEX_DOMAIN/g, this.apexDomain)
|
|
71
|
+
.replace(/MATCH_APEX/g, this.apexDomain !== "" && ((_b = props.redirectToApexDomain) !== null && _b !== void 0 ? _b : true) ? "true" : "false"))
|
|
72
|
+
});
|
|
73
|
+
const bucketOrigin = new aws_cloudfront_origins_1.S3Origin(this.bucket, {
|
|
74
|
+
originAccessIdentity: this.originAccessIdentity
|
|
75
|
+
});
|
|
76
|
+
const fallbackBucketOrigin = this.fallbackBucket === undefined ? undefined : new aws_cloudfront_origins_1.S3Origin(this.fallbackBucket, {
|
|
77
|
+
originAccessIdentity: this.originAccessIdentity
|
|
78
|
+
});
|
|
79
|
+
const defaultOrigin = fallbackBucketOrigin === undefined ? bucketOrigin : new aws_cloudfront_origins_1.OriginGroup({
|
|
80
|
+
primaryOrigin: bucketOrigin,
|
|
81
|
+
fallbackOrigin: fallbackBucketOrigin
|
|
82
|
+
});
|
|
83
|
+
this.distribution = new aws_cloudfront_1.Distribution(this, "Distribution", {
|
|
84
|
+
defaultBehavior: {
|
|
85
|
+
origin: defaultOrigin,
|
|
86
|
+
viewerProtocolPolicy: aws_cloudfront_1.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,
|
|
87
|
+
functionAssociations: [
|
|
88
|
+
{
|
|
89
|
+
function: this.viewerRequestFunction,
|
|
90
|
+
eventType: aws_cloudfront_1.FunctionEventType.VIEWER_REQUEST
|
|
91
|
+
}
|
|
92
|
+
]
|
|
93
|
+
},
|
|
94
|
+
errorResponses: [
|
|
95
|
+
{
|
|
96
|
+
httpStatus: 404,
|
|
97
|
+
responseHttpStatus: 404,
|
|
98
|
+
responsePagePath: "/404.html"
|
|
99
|
+
}
|
|
100
|
+
],
|
|
101
|
+
httpVersion: (_c = props.httpVersion) !== null && _c !== void 0 ? _c : aws_cloudfront_1.HttpVersion.HTTP1_1,
|
|
102
|
+
minimumProtocolVersion: (_d = props.minimumProtocolVersion) !== null && _d !== void 0 ? _d : aws_cloudfront_1.SecurityPolicyProtocol.TLS_V1_2_2021,
|
|
103
|
+
defaultRootObject: "index.html",
|
|
104
|
+
certificate,
|
|
105
|
+
domainNames: aws_route53_2.DomainName.toStrings(domainNames),
|
|
106
|
+
enableIpv6: true,
|
|
107
|
+
priceClass: (_e = props.priceClass) !== null && _e !== void 0 ? _e : aws_cloudfront_1.PriceClass.PRICE_CLASS_100
|
|
108
|
+
});
|
|
109
|
+
this.distributionUrl = `https://${(this.apexDomain != "" ? this.apexDomain : this.distribution.distributionDomainName)}`;
|
|
110
|
+
this.aRecords = [];
|
|
111
|
+
for (let domainNameProps of (_f = props.domainNames) !== null && _f !== void 0 ? _f : []) {
|
|
112
|
+
if ((_g = domainNameProps.createRecord) !== null && _g !== void 0 ? _g : true) {
|
|
113
|
+
let domainName = aws_route53_2.DomainName.findDomainName(domainNameProps, domainNames);
|
|
114
|
+
if (domainName !== undefined) {
|
|
115
|
+
this.aRecords.push(domainName.createARecord(this, aws_route53_1.RecordTarget.fromAlias(new aws_route53_targets_1.CloudFrontTarget(this.distribution))));
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
let bundlingOptions = undefined;
|
|
120
|
+
if (props.sourceType === SourceType.Custom) {
|
|
121
|
+
if (!props.sourceBundlingOptions) {
|
|
122
|
+
throw new Error("sourceBundlingOptions is required if source type is Custom");
|
|
123
|
+
}
|
|
124
|
+
bundlingOptions = props.sourceBundlingOptions;
|
|
125
|
+
}
|
|
126
|
+
if (props.sourceType === SourceType.Static) {
|
|
127
|
+
if (props.sourceBundlingOptions) {
|
|
128
|
+
throw new Error("Cannot use sourceBundlingOptions with source type Static");
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
if (props.sourceType === SourceType.Hugo) {
|
|
132
|
+
bundlingOptions = Website.HUGO_BUNDLING_OPTIONS;
|
|
133
|
+
if (props.sourceBundlingOptions) {
|
|
134
|
+
throw new Error("Cannot use sourceBundlingOptions with source type Hugo");
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
else if (props.sourceType === SourceType.NpmDist) {
|
|
138
|
+
bundlingOptions = Website.NPM_DIST_BUNDLING_OPTIONS;
|
|
139
|
+
if (props.sourceBundlingOptions) {
|
|
140
|
+
throw new Error("Cannot use sourceBundlingOptions with source type NpmDist");
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
const source = aws_s3_deployment_1.Source.asset(props.sourceDirectory, {
|
|
144
|
+
bundling: bundlingOptions
|
|
145
|
+
});
|
|
146
|
+
this.deployment = new aws_s3_deployment_1.BucketDeployment(this, "Assets", {
|
|
147
|
+
sources: [source],
|
|
148
|
+
destinationBucket: this.bucket,
|
|
149
|
+
distribution: this.distribution,
|
|
150
|
+
cacheControl: [aws_s3_deployment_1.CacheControl.maxAge((_h = props.maxAge) !== null && _h !== void 0 ? _h : aws_cdk_lib_1.Duration.minutes(60))],
|
|
151
|
+
prune: false
|
|
152
|
+
});
|
|
153
|
+
if (this.fallbackBucket) {
|
|
154
|
+
this.fallbackDeployment = new aws_s3_deployment_1.BucketDeployment(this, "FallbackAssets", {
|
|
155
|
+
sources: [source],
|
|
156
|
+
destinationBucket: this.fallbackBucket,
|
|
157
|
+
cacheControl: [aws_s3_deployment_1.CacheControl.maxAge((_j = props.maxAge) !== null && _j !== void 0 ? _j : aws_cdk_lib_1.Duration.minutes(60))],
|
|
158
|
+
prune: false
|
|
159
|
+
});
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
exports.Website = Website;
|
|
164
|
+
/**
|
|
165
|
+
* The default bundling options for SourceType.Hugo.
|
|
166
|
+
*/
|
|
167
|
+
Website.HUGO_BUNDLING_OPTIONS = {
|
|
168
|
+
image: aws_cdk_lib_1.DockerImage.fromRegistry("klakegg/hugo:latest-ext"),
|
|
169
|
+
command: [
|
|
170
|
+
"-d", "/asset-output"
|
|
171
|
+
]
|
|
172
|
+
};
|
|
173
|
+
/**
|
|
174
|
+
* The default bundling options for SourceType.NpmDist. It"s expected that "npm run dist" command
|
|
175
|
+
* will place files in ${CDK_BUNDLING_OUTPUT_DIR}. Example: "ng build --output-path=/asset-output"
|
|
176
|
+
*/
|
|
177
|
+
Website.NPM_DIST_BUNDLING_OPTIONS = {
|
|
178
|
+
image: aws_cdk_lib_1.DockerImage.fromRegistry("node:16-alpine"),
|
|
179
|
+
command: [
|
|
180
|
+
// TODO Figure out how to cache .npmcache
|
|
181
|
+
"sh", "-c", [
|
|
182
|
+
"mkdir -p .npmcache",
|
|
183
|
+
"npm ci --cache .npmcache --prefer-offline",
|
|
184
|
+
"npm run dist"
|
|
185
|
+
].join(" && ")
|
|
186
|
+
]
|
|
187
|
+
};
|
|
188
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"website.js","sourceRoot":"","sources":["website.ts"],"names":[],"mappings":";;;AACA,+CAAqE;AACrE,+DAUoC;AACpC,+EAAyE;AACzE,+EAAoG;AACpG,yDAA8D;AAC9D,yEAAiE;AACjE,qEAAqF;AACrF,6CAAkF;AAClF,mDAA8D;AAC9D,2CAAsF;AACtF,6CAA+C;AAE/C,IAAY,UAKX;AALD,WAAY,UAAU;IACpB,+BAAiB,CAAA;IACjB,2BAAa,CAAA;IACb,iCAAmB,CAAA;IACnB,+BAAiB,CAAA;AACnB,CAAC,EALW,UAAU,GAAV,kBAAU,KAAV,kBAAU,QAKrB;AA6FD,MAAa,OAAQ,SAAQ,2BAAiB;IAuC5C,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAmB;;QAC3D,KAAK,CAAC,KAAK,EAAE,EAAE,EAAE,EAAC,YAAY,EAAE,sBAAY,CAAC,KAAK,CAAC,KAAK,CAAC,YAAY,EAAE,0BAAe,CAAC,EAAC,CAAC,CAAC;QAE1F,MAAM,WAAW,GAAG,wBAAU,CAAC,SAAS,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC;QAE5D,IAAI,WAAW,GAA6B,SAAS,CAAC;QACtD,IAAI,KAAK,CAAC,cAAc,KAAK,SAAS,EAAE;YACtC,WAAW,GAAG,oCAAW,CAAC,kBAAkB,CAAC,IAAI,EAAE,aAAa,EAAE,KAAK,CAAC,cAAc,CAAC,CAAA;SACxF;aAAM,IAAI,WAAW,CAAC,MAAM,GAAG,CAAC,EAAE;YACjC,WAAW,GAAG,IAAI,oCAAW,CAAC,IAAI,EAAE,aAAa,EAAE;gBACjD,UAAU,EAAE,WAAW,CAAC,CAAC,CAAC,CAAC,QAAQ,EAAE;gBACrC,uBAAuB,EAAE,wBAAU,CAAC,SAAS,CAAC,WAAW,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC;gBACnE,UAAU,EAAE,8CAAqB,CAAC,gBAAgB,CAAC,wBAAU,CAAC,SAAS,CAAC,IAAI,EAAE,WAAW,CAAC,CAAC;aAC5F,CAAC,CAAC;SACJ;QAED,IAAI,CAAC,MAAM,SAAG,KAAK,CAAC,MAAM,mCAAI,IAAI,eAAM,CAAC,IAAI,EAAE,QAAQ,EAAE;YACvD,UAAU,EAAE,yBAAgB,CAAC,UAAU;YACvC,aAAa,EAAE,2BAAa,CAAC,OAAO;YACpC,iBAAiB,EAAE,IAAI;SACxB,CAAC,CAAC;QACH,IAAI,CAAC,cAAc,GAAG,KAAK,CAAC,cAAc,CAAC;QAE3C,IAAI,CAAC,oBAAoB,GAAG,IAAI,qCAAoB,CAAC,IAAI,EAAE,QAAQ,CAAC,CAAC;QACrE,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,IAAI,CAAC,oBAAoB,CAAC,CAAC;QACjD,IAAI,IAAI,CAAC,cAAc,EAAE;YACvB,IAAI,CAAC,cAAc,CAAC,SAAS,CAAC,IAAI,CAAC,oBAAoB,CAAC,CAAC;SAC1D;QAED,IAAI,CAAC,UAAU,GAAG,WAAW,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;QAE1E,IAAI,CAAC,qBAAqB,GAAG,IAAI,yBAAQ,CAAC,IAAI,EAAE,uBAAuB,EAAE;YACvE,IAAI,EAAE,6BAAY,CAAC,UAAU,CAAC;;;;;;;;;;;;;;;;;;EAkBlC;iBACO,OAAO,CAAC,cAAc,EAAE,IAAI,CAAC,UAAU,CAAC;iBACxC,OAAO,CAAC,aAAa,EAAE,IAAI,CAAC,UAAU,KAAK,EAAE,IAAI,OAAC,KAAK,CAAC,oBAAoB,mCAAE,IAAI,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC;SAC5G,CAAC,CAAC;QAEH,MAAM,YAAY,GAAG,IAAI,iCAAQ,CAAC,IAAI,CAAC,MAAM,EAAE;YAC7C,oBAAoB,EAAE,IAAI,CAAC,oBAAoB;SAChD,CAAC,CAAC;QAEH,MAAM,oBAAoB,GAAG,IAAI,CAAC,cAAc,KAAK,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,IAAI,iCAAQ,CAAC,IAAI,CAAC,cAAc,EAAE;YAC7G,oBAAoB,EAAE,IAAI,CAAC,oBAAoB;SAChD,CAAC,CAAC;QAEH,MAAM,aAAa,GAAG,oBAAoB,KAAK,SAAS,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,IAAI,oCAAW,CAAC;YACxF,aAAa,EAAE,YAAY;YAC3B,cAAc,EAAE,oBAAoB;SACrC,CAAC,CAAC;QAEH,IAAI,CAAC,YAAY,GAAG,IAAI,6BAAY,CAAC,IAAI,EAAE,cAAc,EAAE;YACzD,eAAe,EAAE;gBACf,MAAM,EAAE,aAAa;gBACrB,oBAAoB,EAAE,qCAAoB,CAAC,iBAAiB;gBAC5D,oBAAoB,EAAE;oBACpB;wBACE,QAAQ,EAAE,IAAI,CAAC,qBAAqB;wBACpC,SAAS,EAAE,kCAAiB,CAAC,cAAc;qBAC5C;iBACF;aACF;YACD,cAAc,EAAE;gBACd;oBACE,UAAU,EAAE,GAAG;oBACf,kBAAkB,EAAE,GAAG;oBACvB,gBAAgB,EAAE,WAAW;iBAC9B;aACF;YACD,WAAW,QAAE,KAAK,CAAC,WAAW,mCAAE,4BAAW,CAAC,OAAO;YACnD,sBAAsB,QAAE,KAAK,CAAC,sBAAsB,mCAAE,uCAAsB,CAAC,aAAa;YAC1F,iBAAiB,EAAE,YAAY;YAC/B,WAAW;YACX,WAAW,EAAE,wBAAU,CAAC,SAAS,CAAC,WAAW,CAAC;YAC9C,UAAU,EAAE,IAAI;YAChB,UAAU,QAAE,KAAK,CAAC,UAAU,mCAAE,2BAAU,CAAC,eAAe;SACzD,CAAC,CAAC;QAEH,IAAI,CAAC,eAAe,GAAG,WAAW,CAAC,IAAI,CAAC,UAAU,IAAI,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,sBAAsB,CAAC,EAAE,CAAA;QAExH,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC;QAEnB,KAAK,IAAI,eAAe,UAAI,KAAK,CAAC,WAAW,mCAAI,EAAE,EAAE;YACnD,UAAI,eAAe,CAAC,YAAY,mCAAI,IAAI,EAAE;gBACxC,IAAI,UAAU,GAAG,wBAAU,CAAC,cAAc,CAAC,eAAe,EAAE,WAAW,CAAC,CAAC;gBACzE,IAAI,UAAU,KAAK,SAAS,EAAE;oBAC5B,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC,IAAI,EAC9C,0BAAY,CAAC,SAAS,CAAC,IAAI,sCAAgB,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,CAAC;iBACrE;aACF;SACF;QAED,IAAI,eAAe,GAAgC,SAAS,CAAA;QAE5D,IAAI,KAAK,CAAC,UAAU,KAAK,UAAU,CAAC,MAAM,EAAE;YAC1C,IAAI,CAAC,KAAK,CAAC,qBAAqB,EAAE;gBAChC,MAAM,IAAI,KAAK,CAAC,4DAA4D,CAAC,CAAC;aAC/E;YACD,eAAe,GAAG,KAAK,CAAC,qBAAqB,CAAA;SAC9C;QAED,IAAI,KAAK,CAAC,UAAU,KAAK,UAAU,CAAC,MAAM,EAAE;YAC1C,IAAI,KAAK,CAAC,qBAAqB,EAAE;gBAC/B,MAAM,IAAI,KAAK,CAAC,0DAA0D,CAAC,CAAC;aAC7E;SACF;QAED,IAAI,KAAK,CAAC,UAAU,KAAK,UAAU,CAAC,IAAI,EAAE;YACxC,eAAe,GAAG,OAAO,CAAC,qBAAqB,CAAA;YAC/C,IAAI,KAAK,CAAC,qBAAqB,EAAE;gBAC/B,MAAM,IAAI,KAAK,CAAC,wDAAwD,CAAC,CAAC;aAC3E;SACF;aAAM,IAAI,KAAK,CAAC,UAAU,KAAK,UAAU,CAAC,OAAO,EAAE;YAClD,eAAe,GAAG,OAAO,CAAC,yBAAyB,CAAA;YACnD,IAAI,KAAK,CAAC,qBAAqB,EAAE;gBAC/B,MAAM,IAAI,KAAK,CAAC,2DAA2D,CAAC,CAAC;aAC9E;SACF;QAED,MAAM,MAAM,GAAG,0BAAM,CAAC,KAAK,CAAC,KAAK,CAAC,eAAe,EAAE;YACjD,QAAQ,EAAE,eAAe;SAC1B,CAAC,CAAC;QAEH,IAAI,CAAC,UAAU,GAAG,IAAI,oCAAgB,CAAC,IAAI,EAAE,QAAQ,EAAE;YACrD,OAAO,EAAE,CAAC,MAAM,CAAC;YACjB,iBAAiB,EAAE,IAAI,CAAC,MAAM;YAC9B,YAAY,EAAE,IAAI,CAAC,YAAY;YAC/B,YAAY,EAAE,CAAC,gCAAY,CAAC,MAAM,OAAC,KAAK,CAAC,MAAM,mCAAE,sBAAQ,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC,CAAC;YACvE,KAAK,EAAE,KAAK;SACb,CAAC,CAAC;QAEH,IAAI,IAAI,CAAC,cAAc,EAAE;YACvB,IAAI,CAAC,kBAAkB,GAAG,IAAI,oCAAgB,CAAC,IAAI,EAAE,gBAAgB,EAAE;gBACrE,OAAO,EAAE,CAAC,MAAM,CAAC;gBACjB,iBAAiB,EAAE,IAAI,CAAC,cAAc;gBACtC,YAAY,EAAE,CAAC,gCAAY,CAAC,MAAM,OAAC,KAAK,CAAC,MAAM,mCAAE,sBAAQ,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC,CAAC;gBACvE,KAAK,EAAE,KAAK;aACb,CAAC,CAAA;SACH;IACH,CAAC;;AAnMH,0BAoMC;AAlMC;;GAEG;AACa,6BAAqB,GAAoB;IACvD,KAAK,EAAE,yBAAW,CAAC,YAAY,CAAC,yBAAyB,CAAC;IAC1D,OAAO,EAAE;QACP,IAAI,EAAE,eAAe;KACtB;CACF,CAAA;AAED;;;GAGG;AACa,iCAAyB,GAAoB;IAC3D,KAAK,EAAE,yBAAW,CAAC,YAAY,CAAC,gBAAgB,CAAC;IACjD,OAAO,EAAE;QACP,yCAAyC;QACzC,IAAI,EAAE,IAAI,EAAE;YACV,oBAAoB;YACpB,2CAA2C;YAC3C,cAAc;SACf,CAAC,IAAI,CAAC,MAAM,CAAC;KACf;CACF,CAAA","sourcesContent":["import {Construct} from \"constructs\";\nimport {Bucket, BucketEncryption, IBucket} from \"aws-cdk-lib/aws-s3\";\nimport {\n  Distribution,\n  Function,\n  FunctionCode,\n  FunctionEventType,\n  HttpVersion,\n  OriginAccessIdentity,\n  PriceClass,\n  SecurityPolicyProtocol,\n  ViewerProtocolPolicy\n} from \"aws-cdk-lib/aws-cloudfront\";\nimport {OriginGroup, S3Origin} from \"aws-cdk-lib/aws-cloudfront-origins\";\nimport {Certificate, CertificateValidation, ICertificate} from \"aws-cdk-lib/aws-certificatemanager\";\nimport {ARecord, RecordTarget} from \"aws-cdk-lib/aws-route53\";\nimport {CloudFrontTarget} from \"aws-cdk-lib/aws-route53-targets\";\nimport {BucketDeployment, CacheControl, Source} from \"aws-cdk-lib/aws-s3-deployment\";\nimport {BundlingOptions, DockerImage, Duration, RemovalPolicy} from \"aws-cdk-lib\";\nimport {DomainName, DomainNameProps} from \"../../aws-route53\";\nimport {ExtendedConstruct, ExtendedConstructProps, StandardTags} from \"../../aws-cdk\";\nimport {LibStandardTags} from \"../../truemark\";\n\nexport enum SourceType {\n  Custom = \"Custom\",\n  Hugo = \"Hugo\",\n  NpmDist = \"NpmDist\",\n  Static = \"Static\"\n}\n\n/**\n * Properties for WebsiteDomainName.\n */\nexport interface WebsiteDomainNameProps extends DomainNameProps {\n\n  /**\n   * Whether to create a Route53 record for this domain name. Default is true.\n   *\n   * @default - true\n   */\n  readonly createRecord?: boolean;\n\n}\n\nexport interface WebsiteProps extends ExtendedConstructProps {\n\n  /**\n   * Bucket to use to store website content. If one is not provided, one will be generated.\n   */\n  readonly bucket?: IBucket;\n\n  /**\n   * Bucket to use if the primary bucket fails. If one is not provided a fallback is not setup.\n   */\n  readonly fallbackBucket?: IBucket;\n\n  /**\n   * The ARN to the ACM Certificate to use on the CloudFront distribution. If one is not\n   * provided and domainNames is populated, one will be generated.\n   */\n  readonly certificateArn?: string;\n\n  /**\n   * The domain names to be serviced. The first domain name in the list is treated as the apex domain.\n   */\n  readonly domainNames?: WebsiteDomainNameProps[]\n\n  /**\n   * Redirect traffic to the first domain in the list of domainNames. Defaults to true.\n   *\n   * @default - true\n   */\n  readonly redirectToApexDomain?: boolean;\n\n  /**\n   * Price class for the CloudFront distribution.\n   *\n   * @default - PriceClass.PRICE_CLASS_100\n   */\n  readonly priceClass?: PriceClass\n\n  /**\n   * Minimum protocol version to allow.\n   *\n   * @default SecurityPolicyProtocol.TLS_V1_2_2021,\n   */\n  readonly minimumProtocolVersion?: SecurityPolicyProtocol;\n\n  /**\n   * Minimum HTTP version to allow.\n   *\n   * @default HttpVersion.HTTP1_1\n   */\n  readonly httpVersion?: HttpVersion;\n\n  /**\n   * Cache-control max age.\n   *\n   * @default Duration.minutes(60)\n   */\n  readonly maxAge?: Duration;\n\n  /**\n   * The source type of the website.\n   *\n   * @default SourceType.Static\n   */\n  readonly sourceType?: SourceType;\n\n  /**\n   * The directory where the website sources are located.\n   */\n  readonly sourceDirectory: string;\n\n  /**\n   * Custom bundling options to use to bundle website sources. The sourceType\n   * must be set to SourceType.Custom.\n   */\n  readonly sourceBundlingOptions?: BundlingOptions\n}\n\nexport class Website extends ExtendedConstruct {\n\n  /**\n   * The default bundling options for SourceType.Hugo.\n   */\n  static readonly HUGO_BUNDLING_OPTIONS: BundlingOptions = {\n    image: DockerImage.fromRegistry(\"klakegg/hugo:latest-ext\"),\n    command: [\n      \"-d\", \"/asset-output\"\n    ]\n  }\n\n  /**\n   * The default bundling options for SourceType.NpmDist. It\"s expected that \"npm run dist\" command\n   * will place files in ${CDK_BUNDLING_OUTPUT_DIR}. Example: \"ng build --output-path=/asset-output\"\n   */\n  static readonly NPM_DIST_BUNDLING_OPTIONS: BundlingOptions = {\n    image: DockerImage.fromRegistry(\"node:16-alpine\"),\n    command: [\n      // TODO Figure out how to cache .npmcache\n      \"sh\", \"-c\", [\n        \"mkdir -p .npmcache\",\n        \"npm ci --cache .npmcache --prefer-offline\",\n        \"npm run dist\"\n      ].join(\" && \")\n    ]\n  }\n\n  readonly bucket: IBucket;\n  readonly fallbackBucket?: IBucket;\n  readonly originAccessIdentity: OriginAccessIdentity;\n  readonly apexDomain: string;\n  readonly viewerRequestFunction: Function;\n  readonly distribution: Distribution;\n  readonly distributionUrl: string;\n  readonly aRecords: ARecord[];\n  readonly deployment: BucketDeployment;\n  readonly fallbackDeployment?: BucketDeployment;\n\n  constructor(scope: Construct, id: string, props: WebsiteProps) {\n    super(scope, id, {standardTags: StandardTags.merge(props.standardTags, LibStandardTags)});\n\n    const domainNames = DomainName.fromProps(props.domainNames);\n\n    let certificate: ICertificate | undefined = undefined;\n    if (props.certificateArn !== undefined) {\n      certificate = Certificate.fromCertificateArn(this, \"Certificate\", props.certificateArn)\n    } else if (domainNames.length > 0) {\n      certificate = new Certificate(this, \"Certificate\", {\n        domainName: domainNames[0].toString(),\n        subjectAlternativeNames: DomainName.toStrings(domainNames).slice(1),\n        validation: CertificateValidation.fromDnsMultiZone(DomainName.toZoneMap(this, domainNames))\n      });\n    }\n\n    this.bucket = props.bucket ?? new Bucket(this, \"Bucket\", {\n      encryption: BucketEncryption.S3_MANAGED,\n      removalPolicy: RemovalPolicy.DESTROY,\n      autoDeleteObjects: true\n    });\n    this.fallbackBucket = props.fallbackBucket;\n\n    this.originAccessIdentity = new OriginAccessIdentity(this, \"Access\");\n    this.bucket.grantRead(this.originAccessIdentity);\n    if (this.fallbackBucket) {\n      this.fallbackBucket.grantRead(this.originAccessIdentity);\n    }\n\n    this.apexDomain = domainNames.length > 0 ? domainNames[0].toString() : \"\";\n\n    this.viewerRequestFunction = new Function(this, \"ViewerRequestFunction\", {\n      code: FunctionCode.fromInline(`\nfunction handler(event) {\n  var host = event.request.headers.host.value;\n  var uri = event.request.uri;\n  var matchApex = MATCH_APEX;\n  if (matchApex && host !== \"APEX_DOMAIN\") {\n    return {\n      statusCode: 301,\n      statusDescription: \"Permanently moved\",\n      headers: {\n        \"location\": { \"value\": \"https://APEX_DOMAIN\" + uri }\n      }\n    }\n  }\n  if (uri.endsWith(\"/\")) {\n    event.request.uri = uri + \"index.html\";\n  }\n  return event.request;\n}`\n        .replace(/APEX_DOMAIN/g, this.apexDomain)\n        .replace(/MATCH_APEX/g, this.apexDomain !== \"\" && (props.redirectToApexDomain??true) ? \"true\" : \"false\"))\n    });\n\n    const bucketOrigin = new S3Origin(this.bucket, {\n      originAccessIdentity: this.originAccessIdentity\n    });\n\n    const fallbackBucketOrigin = this.fallbackBucket === undefined ? undefined : new S3Origin(this.fallbackBucket, {\n      originAccessIdentity: this.originAccessIdentity\n    });\n\n    const defaultOrigin = fallbackBucketOrigin === undefined ? bucketOrigin : new OriginGroup({\n      primaryOrigin: bucketOrigin,\n      fallbackOrigin: fallbackBucketOrigin\n    });\n\n    this.distribution = new Distribution(this, \"Distribution\", {\n      defaultBehavior: {\n        origin: defaultOrigin,\n        viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n        functionAssociations: [\n          {\n            function: this.viewerRequestFunction,\n            eventType: FunctionEventType.VIEWER_REQUEST\n          }\n        ]\n      },\n      errorResponses: [\n        {\n          httpStatus: 404,\n          responseHttpStatus: 404,\n          responsePagePath: \"/404.html\"\n        }\n      ],\n      httpVersion: props.httpVersion??HttpVersion.HTTP1_1,\n      minimumProtocolVersion: props.minimumProtocolVersion??SecurityPolicyProtocol.TLS_V1_2_2021,\n      defaultRootObject: \"index.html\",\n      certificate,\n      domainNames: DomainName.toStrings(domainNames),\n      enableIpv6: true,\n      priceClass: props.priceClass??PriceClass.PRICE_CLASS_100\n    });\n\n    this.distributionUrl = `https://${(this.apexDomain != \"\" ? this.apexDomain : this.distribution.distributionDomainName)}`\n\n    this.aRecords = [];\n\n    for (let domainNameProps of props.domainNames ?? []) {\n      if (domainNameProps.createRecord ?? true) {\n        let domainName = DomainName.findDomainName(domainNameProps, domainNames);\n        if (domainName !== undefined) {\n          this.aRecords.push(domainName.createARecord(this,\n            RecordTarget.fromAlias(new CloudFrontTarget(this.distribution))));\n        }\n      }\n    }\n\n    let bundlingOptions: BundlingOptions | undefined = undefined\n\n    if (props.sourceType === SourceType.Custom) {\n      if (!props.sourceBundlingOptions) {\n        throw new Error(\"sourceBundlingOptions is required if source type is Custom\");\n      }\n      bundlingOptions = props.sourceBundlingOptions\n    }\n\n    if (props.sourceType === SourceType.Static) {\n      if (props.sourceBundlingOptions) {\n        throw new Error(\"Cannot use sourceBundlingOptions with source type Static\");\n      }\n    }\n\n    if (props.sourceType === SourceType.Hugo) {\n      bundlingOptions = Website.HUGO_BUNDLING_OPTIONS\n      if (props.sourceBundlingOptions) {\n        throw new Error(\"Cannot use sourceBundlingOptions with source type Hugo\");\n      }\n    } else if (props.sourceType === SourceType.NpmDist) {\n      bundlingOptions = Website.NPM_DIST_BUNDLING_OPTIONS\n      if (props.sourceBundlingOptions) {\n        throw new Error(\"Cannot use sourceBundlingOptions with source type NpmDist\");\n      }\n    }\n\n    const source = Source.asset(props.sourceDirectory, {\n      bundling: bundlingOptions\n    });\n\n    this.deployment = new BucketDeployment(this, \"Assets\", {\n      sources: [source],\n      destinationBucket: this.bucket,\n      distribution: this.distribution,\n      cacheControl: [CacheControl.maxAge(props.maxAge??Duration.minutes(60))],\n      prune: false\n    });\n\n    if (this.fallbackBucket) {\n      this.fallbackDeployment = new BucketDeployment(this, \"FallbackAssets\", {\n        sources: [source],\n        destinationBucket: this.fallbackBucket,\n        cacheControl: [CacheControl.maxAge(props.maxAge??Duration.minutes(60))],\n        prune: false\n      })\n    }\n  }\n}\n"]}
|
|
@@ -0,0 +1,60 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
const assertions_1 = require("aws-cdk-lib/assertions");
|
|
4
|
+
const test_helper_1 = require("../../test-helper");
|
|
5
|
+
const aws_route53_1 = require("aws-cdk-lib/aws-route53");
|
|
6
|
+
const website_1 = require("./website");
|
|
7
|
+
test("Hugo Website Test", () => {
|
|
8
|
+
const stack = test_helper_1.TestHelper.stack();
|
|
9
|
+
new website_1.Website(stack, 'TestWebsite', {
|
|
10
|
+
sourceType: website_1.SourceType.Hugo,
|
|
11
|
+
sourceDirectory: test_helper_1.TestHelper.resolveTestFiles("hugo-website")
|
|
12
|
+
});
|
|
13
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
14
|
+
template.hasResourceProperties("AWS::CloudFront::Distribution", {});
|
|
15
|
+
});
|
|
16
|
+
test("ARecord Created for Website", () => {
|
|
17
|
+
const stack = test_helper_1.TestHelper.stack();
|
|
18
|
+
new website_1.Website(stack, 'TestWebsite', {
|
|
19
|
+
sourceType: website_1.SourceType.Hugo,
|
|
20
|
+
sourceDirectory: test_helper_1.TestHelper.resolveTestFiles("hugo-website"),
|
|
21
|
+
certificateArn: "arn:aws:acm:us-east-1:123456789012:certificate/XXXX",
|
|
22
|
+
domainNames: [
|
|
23
|
+
{
|
|
24
|
+
prefix: "www",
|
|
25
|
+
// TODO In the fugure need to figure out how to mock lookups
|
|
26
|
+
zone: new aws_route53_1.HostedZone(stack, "example-com", { zoneName: "example.com" })
|
|
27
|
+
}
|
|
28
|
+
]
|
|
29
|
+
});
|
|
30
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
31
|
+
// console.log(template.toJSON());
|
|
32
|
+
// TestHelper.logResources(template, "AWS::Route53::RecordSet");
|
|
33
|
+
template.hasResourceProperties("AWS::Route53::RecordSet", {});
|
|
34
|
+
});
|
|
35
|
+
// TODO We need to speed this up
|
|
36
|
+
// test("Npm Dist Website Test", () => {
|
|
37
|
+
// const stack = TestHelper.stack();
|
|
38
|
+
// new Website(stack, "TestWebsite", {
|
|
39
|
+
// sourceType: SourceType.NpmDist,
|
|
40
|
+
// sourceDirectory: TestHelper.resolveTestFiles("angular-website")
|
|
41
|
+
// });
|
|
42
|
+
// const template = Template.fromStack(stack);
|
|
43
|
+
// template.hasResourceProperties("AWS::CloudFront::Distribution", {});
|
|
44
|
+
// });
|
|
45
|
+
// TODO Need to fix
|
|
46
|
+
// test("Website Certificate Test", () => {
|
|
47
|
+
// const stack = TestHelper.stack();
|
|
48
|
+
// new Website(stack, "TestWebsite", {
|
|
49
|
+
// sourceType: SourceType.Hugo,
|
|
50
|
+
// sourceDirectory: TestHelper.resolveTestFiles("hugo-website"),
|
|
51
|
+
// domainNames: [
|
|
52
|
+
// new DomainName("test", "example.com"),
|
|
53
|
+
// new DomainName("test2", "example.com")
|
|
54
|
+
// ]
|
|
55
|
+
// });
|
|
56
|
+
// const template = Template.fromStack(stack);
|
|
57
|
+
// console.log(template.toJSON());
|
|
58
|
+
// // TestHelper.logResources()
|
|
59
|
+
// });
|
|
60
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoid2Vic2l0ZS50ZXN0LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsid2Vic2l0ZS50ZXN0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7O0FBQUEsdURBQWdEO0FBQ2hELG1EQUE2QztBQUM3Qyx5REFBbUQ7QUFDbkQsdUNBQThDO0FBRTlDLElBQUksQ0FBQyxtQkFBbUIsRUFBRSxHQUFHLEVBQUU7SUFDN0IsTUFBTSxLQUFLLEdBQUcsd0JBQVUsQ0FBQyxLQUFLLEVBQUUsQ0FBQztJQUNqQyxJQUFJLGlCQUFPLENBQUMsS0FBSyxFQUFFLGFBQWEsRUFBRTtRQUNoQyxVQUFVLEVBQUUsb0JBQVUsQ0FBQyxJQUFJO1FBQzNCLGVBQWUsRUFBRSx3QkFBVSxDQUFDLGdCQUFnQixDQUFDLGNBQWMsQ0FBQztLQUM3RCxDQUFDLENBQUM7SUFDSCxNQUFNLFFBQVEsR0FBRyxxQkFBUSxDQUFDLFNBQVMsQ0FBQyxLQUFLLENBQUMsQ0FBQztJQUMzQyxRQUFRLENBQUMscUJBQXFCLENBQUMsK0JBQStCLEVBQUUsRUFBRSxDQUFDLENBQUM7QUFDdEUsQ0FBQyxDQUFDLENBQUM7QUFFSCxJQUFJLENBQUMsNkJBQTZCLEVBQUUsR0FBRyxFQUFFO0lBQ3ZDLE1BQU0sS0FBSyxHQUFHLHdCQUFVLENBQUMsS0FBSyxFQUFFLENBQUM7SUFDakMsSUFBSSxpQkFBTyxDQUFDLEtBQUssRUFBRSxhQUFhLEVBQUU7UUFDaEMsVUFBVSxFQUFFLG9CQUFVLENBQUMsSUFBSTtRQUMzQixlQUFlLEVBQUUsd0JBQVUsQ0FBQyxnQkFBZ0IsQ0FBQyxjQUFjLENBQUM7UUFDNUQsY0FBYyxFQUFFLHFEQUFxRDtRQUNyRSxXQUFXLEVBQUU7WUFDWDtnQkFDRSxNQUFNLEVBQUUsS0FBSztnQkFDYiw0REFBNEQ7Z0JBQzVELElBQUksRUFBRSxJQUFJLHdCQUFVLENBQUMsS0FBSyxFQUFFLGFBQWEsRUFBRSxFQUFDLFFBQVEsRUFBRSxhQUFhLEVBQUMsQ0FBQzthQUN0RTtTQUNGO0tBQ0YsQ0FBQyxDQUFDO0lBQ0gsTUFBTSxRQUFRLEdBQUcscUJBQVEsQ0FBQyxTQUFTLENBQUMsS0FBSyxDQUFDLENBQUM7SUFDM0Msa0NBQWtDO0lBQ2xDLGdFQUFnRTtJQUNoRSxRQUFRLENBQUMscUJBQXFCLENBQUMseUJBQXlCLEVBQUUsRUFBRSxDQUFDLENBQUM7QUFDaEUsQ0FBQyxDQUFDLENBQUM7QUFFSCxnQ0FBZ0M7QUFDaEMsd0NBQXdDO0FBQ3hDLHNDQUFzQztBQUN0Qyx3Q0FBd0M7QUFDeEMsc0NBQXNDO0FBQ3RDLHNFQUFzRTtBQUN0RSxRQUFRO0FBQ1IsZ0RBQWdEO0FBQ2hELHlFQUF5RTtBQUN6RSxNQUFNO0FBRU4sbUJBQW1CO0FBQ25CLDJDQUEyQztBQUMzQyxzQ0FBc0M7QUFDdEMsd0NBQXdDO0FBQ3hDLG1DQUFtQztBQUNuQyxvRUFBb0U7QUFDcEUscUJBQXFCO0FBQ3JCLCtDQUErQztBQUMvQywrQ0FBK0M7QUFDL0MsUUFBUTtBQUNSLFFBQVE7QUFDUixnREFBZ0Q7QUFDaEQsb0NBQW9DO0FBQ3BDLGlDQUFpQztBQUNqQyxNQUFNIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtUZW1wbGF0ZX0gZnJvbSBcImF3cy1jZGstbGliL2Fzc2VydGlvbnNcIjtcbmltcG9ydCB7VGVzdEhlbHBlcn0gZnJvbSBcIi4uLy4uL3Rlc3QtaGVscGVyXCI7XG5pbXBvcnQge0hvc3RlZFpvbmV9IGZyb20gXCJhd3MtY2RrLWxpYi9hd3Mtcm91dGU1M1wiO1xuaW1wb3J0IHtTb3VyY2VUeXBlLCBXZWJzaXRlfSBmcm9tIFwiLi93ZWJzaXRlXCI7XG5cbnRlc3QoXCJIdWdvIFdlYnNpdGUgVGVzdFwiLCAoKSA9PiB7XG4gIGNvbnN0IHN0YWNrID0gVGVzdEhlbHBlci5zdGFjaygpO1xuICBuZXcgV2Vic2l0ZShzdGFjaywgJ1Rlc3RXZWJzaXRlJywge1xuICAgIHNvdXJjZVR5cGU6IFNvdXJjZVR5cGUuSHVnbyxcbiAgICBzb3VyY2VEaXJlY3Rvcnk6IFRlc3RIZWxwZXIucmVzb2x2ZVRlc3RGaWxlcyhcImh1Z28td2Vic2l0ZVwiKVxuICB9KTtcbiAgY29uc3QgdGVtcGxhdGUgPSBUZW1wbGF0ZS5mcm9tU3RhY2soc3RhY2spO1xuICB0ZW1wbGF0ZS5oYXNSZXNvdXJjZVByb3BlcnRpZXMoXCJBV1M6OkNsb3VkRnJvbnQ6OkRpc3RyaWJ1dGlvblwiLCB7fSk7XG59KTtcblxudGVzdChcIkFSZWNvcmQgQ3JlYXRlZCBmb3IgV2Vic2l0ZVwiLCAoKSA9PiB7XG4gIGNvbnN0IHN0YWNrID0gVGVzdEhlbHBlci5zdGFjaygpO1xuICBuZXcgV2Vic2l0ZShzdGFjaywgJ1Rlc3RXZWJzaXRlJywge1xuICAgIHNvdXJjZVR5cGU6IFNvdXJjZVR5cGUuSHVnbyxcbiAgICBzb3VyY2VEaXJlY3Rvcnk6IFRlc3RIZWxwZXIucmVzb2x2ZVRlc3RGaWxlcyhcImh1Z28td2Vic2l0ZVwiKSxcbiAgICBjZXJ0aWZpY2F0ZUFybjogXCJhcm46YXdzOmFjbTp1cy1lYXN0LTE6MTIzNDU2Nzg5MDEyOmNlcnRpZmljYXRlL1hYWFhcIixcbiAgICBkb21haW5OYW1lczogW1xuICAgICAge1xuICAgICAgICBwcmVmaXg6IFwid3d3XCIsXG4gICAgICAgIC8vIFRPRE8gSW4gdGhlIGZ1Z3VyZSBuZWVkIHRvIGZpZ3VyZSBvdXQgaG93IHRvIG1vY2sgbG9va3Vwc1xuICAgICAgICB6b25lOiBuZXcgSG9zdGVkWm9uZShzdGFjaywgXCJleGFtcGxlLWNvbVwiLCB7em9uZU5hbWU6IFwiZXhhbXBsZS5jb21cIn0pXG4gICAgICB9XG4gICAgXVxuICB9KTtcbiAgY29uc3QgdGVtcGxhdGUgPSBUZW1wbGF0ZS5mcm9tU3RhY2soc3RhY2spO1xuICAvLyBjb25zb2xlLmxvZyh0ZW1wbGF0ZS50b0pTT04oKSk7XG4gIC8vIFRlc3RIZWxwZXIubG9nUmVzb3VyY2VzKHRlbXBsYXRlLCBcIkFXUzo6Um91dGU1Mzo6UmVjb3JkU2V0XCIpO1xuICB0ZW1wbGF0ZS5oYXNSZXNvdXJjZVByb3BlcnRpZXMoXCJBV1M6OlJvdXRlNTM6OlJlY29yZFNldFwiLCB7fSk7XG59KTtcblxuLy8gVE9ETyBXZSBuZWVkIHRvIHNwZWVkIHRoaXMgdXBcbi8vIHRlc3QoXCJOcG0gRGlzdCBXZWJzaXRlIFRlc3RcIiwgKCkgPT4ge1xuLy8gICBjb25zdCBzdGFjayA9IFRlc3RIZWxwZXIuc3RhY2soKTtcbi8vICAgbmV3IFdlYnNpdGUoc3RhY2ssIFwiVGVzdFdlYnNpdGVcIiwge1xuLy8gICAgIHNvdXJjZVR5cGU6IFNvdXJjZVR5cGUuTnBtRGlzdCxcbi8vICAgICBzb3VyY2VEaXJlY3Rvcnk6IFRlc3RIZWxwZXIucmVzb2x2ZVRlc3RGaWxlcyhcImFuZ3VsYXItd2Vic2l0ZVwiKVxuLy8gICB9KTtcbi8vICAgY29uc3QgdGVtcGxhdGUgPSBUZW1wbGF0ZS5mcm9tU3RhY2soc3RhY2spO1xuLy8gICB0ZW1wbGF0ZS5oYXNSZXNvdXJjZVByb3BlcnRpZXMoXCJBV1M6OkNsb3VkRnJvbnQ6OkRpc3RyaWJ1dGlvblwiLCB7fSk7XG4vLyB9KTtcblxuLy8gVE9ETyBOZWVkIHRvIGZpeFxuLy8gdGVzdChcIldlYnNpdGUgQ2VydGlmaWNhdGUgVGVzdFwiLCAoKSA9PiB7XG4vLyAgIGNvbnN0IHN0YWNrID0gVGVzdEhlbHBlci5zdGFjaygpO1xuLy8gICBuZXcgV2Vic2l0ZShzdGFjaywgXCJUZXN0V2Vic2l0ZVwiLCB7XG4vLyAgICAgc291cmNlVHlwZTogU291cmNlVHlwZS5IdWdvLFxuLy8gICAgIHNvdXJjZURpcmVjdG9yeTogVGVzdEhlbHBlci5yZXNvbHZlVGVzdEZpbGVzKFwiaHVnby13ZWJzaXRlXCIpLFxuLy8gICAgIGRvbWFpbk5hbWVzOiBbXG4vLyAgICAgICBuZXcgRG9tYWluTmFtZShcInRlc3RcIiwgXCJleGFtcGxlLmNvbVwiKSxcbi8vICAgICAgIG5ldyBEb21haW5OYW1lKFwidGVzdDJcIiwgXCJleGFtcGxlLmNvbVwiKVxuLy8gICAgIF1cbi8vICAgfSk7XG4vLyAgIGNvbnN0IHRlbXBsYXRlID0gVGVtcGxhdGUuZnJvbVN0YWNrKHN0YWNrKTtcbi8vICAgY29uc29sZS5sb2codGVtcGxhdGUudG9KU09OKCkpO1xuLy8gICAvLyBUZXN0SGVscGVyLmxvZ1Jlc291cmNlcygpXG4vLyB9KTtcbiJdfQ==
|
|
@@ -3,10 +3,11 @@ import { Bucket } from "aws-cdk-lib/aws-s3";
|
|
|
3
3
|
import { OriginAccessIdentity } from "aws-cdk-lib/aws-cloudfront";
|
|
4
4
|
import { BucketDeployment, ISource } from "aws-cdk-lib/aws-s3-deployment";
|
|
5
5
|
import { Duration, RemovalPolicy } from "aws-cdk-lib";
|
|
6
|
+
import { ExtendedConstruct, ExtendedConstructProps } from "../../aws-cdk";
|
|
6
7
|
/**
|
|
7
8
|
* Properties for CloudFrontBucket.
|
|
8
9
|
*/
|
|
9
|
-
export interface CloudFrontBucketProps {
|
|
10
|
+
export interface CloudFrontBucketProps extends ExtendedConstructProps {
|
|
10
11
|
/**
|
|
11
12
|
* Policy to apply when the bucket is removed from this stack.
|
|
12
13
|
* @default RemovalPolicy.RETAIN
|
|
@@ -35,7 +36,7 @@ export interface CloudFrontBucketProps {
|
|
|
35
36
|
/**
|
|
36
37
|
* Simple Construct for creating buckets that will be accessed directly by CloudFront as an Origin.
|
|
37
38
|
*/
|
|
38
|
-
export declare class CloudFrontBucket extends
|
|
39
|
+
export declare class CloudFrontBucket extends ExtendedConstruct {
|
|
39
40
|
readonly bucket: Bucket;
|
|
40
41
|
readonly bucketName: string;
|
|
41
42
|
readonly bucketArn: string;
|
|
@@ -1,18 +1,19 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
3
|
exports.CloudFrontBucket = void 0;
|
|
4
|
-
const constructs_1 = require("constructs");
|
|
5
4
|
const aws_s3_1 = require("aws-cdk-lib/aws-s3");
|
|
6
5
|
const aws_cloudfront_1 = require("aws-cdk-lib/aws-cloudfront");
|
|
7
6
|
const aws_s3_deployment_1 = require("aws-cdk-lib/aws-s3-deployment");
|
|
8
7
|
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
8
|
+
const aws_cdk_1 = require("../../aws-cdk");
|
|
9
|
+
const truemark_1 = require("../../truemark");
|
|
9
10
|
/**
|
|
10
11
|
* Simple Construct for creating buckets that will be accessed directly by CloudFront as an Origin.
|
|
11
12
|
*/
|
|
12
|
-
class CloudFrontBucket extends
|
|
13
|
+
class CloudFrontBucket extends aws_cdk_1.ExtendedConstruct {
|
|
13
14
|
constructor(scope, id, props) {
|
|
14
15
|
var _a, _b, _c, _d;
|
|
15
|
-
super(scope, id);
|
|
16
|
+
super(scope, id, { standardTags: aws_cdk_1.StandardTags.merge(props.standardTags, truemark_1.LibStandardTags) });
|
|
16
17
|
const removalPolicy = (_a = props.removalPolicy) !== null && _a !== void 0 ? _a : aws_cdk_lib_1.RemovalPolicy.RETAIN;
|
|
17
18
|
const autoDeleteObjects = ((_b = props.autoDeleteObjects) !== null && _b !== void 0 ? _b : false) && removalPolicy === aws_cdk_lib_1.RemovalPolicy.DESTROY;
|
|
18
19
|
this.bucket = new aws_s3_1.Bucket(this, "Default", {
|
|
@@ -77,4 +78,4 @@ class CloudFrontBucket extends constructs_1.Construct {
|
|
|
77
78
|
}
|
|
78
79
|
}
|
|
79
80
|
exports.CloudFrontBucket = CloudFrontBucket;
|
|
80
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
81
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloud-front-bucket.js","sourceRoot":"","sources":["cloud-front-bucket.ts"],"names":[],"mappings":"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","sourcesContent":["import {Construct} from \"constructs\";\nimport {BlockPublicAccess, Bucket, BucketEncryption, IBucket} from \"aws-cdk-lib/aws-s3\";\nimport {OriginAccessIdentity} from \"aws-cdk-lib/aws-cloudfront\";\nimport {BucketDeployment, CacheControl, ISource, Source} from \"aws-cdk-lib/aws-s3-deployment\";\nimport {Duration, RemovalPolicy} from \"aws-cdk-lib\";\nimport {ExtendedConstruct, ExtendedConstructProps, StandardTags} from \"../../aws-cdk\";\nimport {LibStandardTags} from \"../../truemark\";\n\n/**\n * Properties for CloudFrontBucket.\n */\nexport interface CloudFrontBucketProps extends ExtendedConstructProps {\n\n  /**\n   * Policy to apply when the bucket is removed from this stack.\n   * @default RemovalPolicy.RETAIN\n   */\n  readonly removalPolicy?: RemovalPolicy;\n\n  /**\n   * Whether all objects should be automatically deleted when the bucket is removed from the stack or when the stack is deleted.\n   * Requires the removalPolicy to be set to RemovalPolicy.DESTROY. Default is false.\n   *\n   * @default false\n   */\n  readonly autoDeleteObjects?: boolean;\n\n  /**\n   * Whether this bucket should have versioning turned on or not. Default is false.\n   *\n   * @default false\n   */\n  readonly versioned?: boolean;\n\n  /**\n   * Whether this bucket should have transfer acceleration turned on or not. Default is false.\n   *\n   * @default false\n   */\n  readonly transferAcceleration?: boolean;\n}\n\n/**\n * Simple Construct for creating buckets that will be accessed directly by CloudFront as an Origin.\n */\nexport class CloudFrontBucket extends ExtendedConstruct {\n\n  readonly bucket: Bucket;\n  readonly bucketName: string;\n  readonly bucketArn: string;\n  readonly originAccessIdentity: OriginAccessIdentity;\n  readonly originAccessIdentityId: string;\n\n  constructor(scope: Construct, id: string, props: CloudFrontBucketProps) {\n    super(scope, id, {standardTags: StandardTags.merge(props.standardTags, LibStandardTags)});\n\n    const removalPolicy = props.removalPolicy ?? RemovalPolicy.RETAIN;\n    const autoDeleteObjects = (props.autoDeleteObjects ?? false) && removalPolicy === RemovalPolicy.DESTROY;\n\n    this.bucket = new Bucket(this, \"Default\", {\n      encryption: BucketEncryption.S3_MANAGED, // CloudFront cannot use KMS with S3\n      blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n      removalPolicy,\n      autoDeleteObjects,\n      versioned: props.versioned ?? false,\n      transferAcceleration: props.transferAcceleration ?? false\n    });\n    this.bucketName = this.bucket.bucketName;\n    this.bucketArn = this.bucket.bucketArn;\n    this.originAccessIdentity = new OriginAccessIdentity(this, \"Access\", {\n      comment: `S3 bucket ${this.bucket.bucketName}`\n    });\n    this.originAccessIdentityId = this.originAccessIdentity.originAccessIdentityId;\n    this.bucket.grantRead(this.originAccessIdentity);\n  }\n\n  /**\n   * Helper method to deploy local assets to the created bucket. Ths function assumes\n   * CloudFront invalidation requests will be sent for mutable files to serve new content.\n   * For more complicated deployments, use BucketDeployment directly.\n   *\n   * @param path the path to the local assets\n   * @param maxAge the length of time to browsers will cache files; default is Duration.minutes(15)\n   * @param sMaxAge the length of time CloudFront will cache files; default is Duration.days(7)\n   * @param prune true to prune old files; default is false\n   */\n  deployPath(path: string, maxAge?: Duration, sMaxAge?: Duration, prune?: boolean): BucketDeployment {\n    return new BucketDeployment(this, \"Deploy\", {\n      sources: [Source.asset(path)],\n      destinationBucket: this.bucket,\n      prune: prune ?? false,\n      cacheControl: [\n        CacheControl.setPublic(),\n        CacheControl.maxAge(maxAge ?? Duration.minutes(15)),\n        CacheControl.sMaxAge(sMaxAge ?? Duration.days(7))\n      ]\n    });\n  }\n\n  /**\n   * Helper method to assets to the created bucket. This function assumes CloudFront invalidation\n   * requests will be sent for mutable files to serve new content.\n   * For more complicated deployments, use BucketDeployment directly.\n   *\n   * @param source the source to deploy\n   * @param maxAge the length of time to browsers will cache files; default is Duration.minutes(15)\n   * @param sMaxAge the length of time CloudFront will cache files; default is Duration.days(7)\n   * @param prune true to prune old files; default is false\n   */\n  deploySource(source: ISource, maxAge?: Duration, sMaxAge?: Duration, prune?: boolean): BucketDeployment {\n    return new BucketDeployment(this, \"Deploy\", {\n      sources: [source],\n      destinationBucket: this.bucket,\n      prune: prune ?? false,\n      cacheControl: [\n        CacheControl.setPublic(),\n        CacheControl.maxAge(maxAge ?? Duration.minutes(15)),\n        CacheControl.sMaxAge(sMaxAge ?? Duration.days(7))\n      ]\n    })\n  }\n}\n"]}
|
package/index.d.ts
CHANGED
|
@@ -14,7 +14,7 @@ export * as aws_ses from "./aws-ses";
|
|
|
14
14
|
export * as aws_sqs from "./aws-sqs";
|
|
15
15
|
export * as aws_ssm from "./aws-ssm";
|
|
16
16
|
export * as aws_vpc from "./aws-vpc";
|
|
17
|
-
export * as aws_website from "./aws-website";
|
|
18
|
-
export * as helpers from "./helpers";
|
|
19
|
-
export * as wordpress from "./aws-patterns-wordpress";
|
|
20
17
|
export * as domain_redirect from "./aws-patterns-domain-redirect";
|
|
18
|
+
export * as website from "./aws-patterns-website";
|
|
19
|
+
export * as wordpress from "./aws-patterns-wordpress";
|
|
20
|
+
export * as helpers from "./helpers";
|
package/index.js
CHANGED
|
@@ -16,8 +16,8 @@ exports.aws_ses = require("./aws-ses");
|
|
|
16
16
|
exports.aws_sqs = require("./aws-sqs");
|
|
17
17
|
exports.aws_ssm = require("./aws-ssm");
|
|
18
18
|
exports.aws_vpc = require("./aws-vpc");
|
|
19
|
-
exports.aws_website = require("./aws-website");
|
|
20
|
-
exports.helpers = require("./helpers");
|
|
21
|
-
exports.wordpress = require("./aws-patterns-wordpress");
|
|
22
19
|
exports.domain_redirect = require("./aws-patterns-domain-redirect");
|
|
23
|
-
|
|
20
|
+
exports.website = require("./aws-patterns-website");
|
|
21
|
+
exports.wordpress = require("./aws-patterns-wordpress");
|
|
22
|
+
exports.helpers = require("./helpers");
|
|
23
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOztBQUFBLHlEQUF1RDtBQUN2RCx1Q0FBcUM7QUFDckMsdURBQXFEO0FBQ3JELHFEQUFtRDtBQUNuRCxxREFBbUQ7QUFDbkQseURBQXVEO0FBQ3ZELGlEQUErQztBQUMvQyx1Q0FBcUM7QUFDckMsNkNBQTJDO0FBQzNDLHFEQUFtRDtBQUNuRCwrQ0FBNkM7QUFDN0MscUNBQW1DO0FBQ25DLHVDQUFxQztBQUNyQyx1Q0FBcUM7QUFDckMsdUNBQXFDO0FBQ3JDLHVDQUFxQztBQUNyQyxvRUFBa0U7QUFDbEUsb0RBQWtEO0FBQ2xELHdEQUFzRDtBQUN0RCx1Q0FBcUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBhcyBhd3NfYXBpZ2F0ZXdheXYyIGZyb20gXCIuL2F3cy1hcGlnYXRld2F5djJcIjtcbmV4cG9ydCAqIGFzIGF3c19jZGsgZnJvbSBcIi4vYXdzLWNka1wiO1xuZXhwb3J0ICogYXMgYXdzX2NlbnRlcmdhdWdlIGZyb20gXCIuL2F3cy1jZW50ZXJnYXVnZVwiO1xuZXhwb3J0ICogYXMgYXdzX2Nsb3VkZnJvbnQgZnJvbSBcIi4vYXdzLWNsb3VkZnJvbnRcIjtcbmV4cG9ydCAqIGFzIGF3c19jbG91ZHdhdGNoIGZyb20gXCIuL2F3cy1jbG91ZHdhdGNoXCI7XG5leHBvcnQgKiBhcyBhd3NfY29kZXBpcGVsaW5lIGZyb20gXCIuL2F3cy1jb2RlcGlwZWxpbmVcIjtcbmV4cG9ydCAqIGFzIGF3c19keW5hbW9kYiBmcm9tIFwiLi9hd3MtZHluYW1vZGJcIjtcbmV4cG9ydCAqIGFzIGF3c19lY3MgZnJvbSBcIi4vYXdzLWVjc1wiO1xuZXhwb3J0ICogYXMgYXdzX2xhbWJkYSBmcm9tIFwiLi9hd3MtbGFtYmRhXCI7XG5leHBvcnQgKiBhcyBhd3NfbW9uaXRvcmluZyBmcm9tIFwiLi9hd3MtbW9uaXRvcmluZ1wiO1xuZXhwb3J0ICogYXMgYXdzX3JvdXRlNTMgZnJvbSBcIi4vYXdzLXJvdXRlNTNcIjtcbmV4cG9ydCAqIGFzIGF3c19zMyBmcm9tIFwiLi9hd3MtczNcIjtcbmV4cG9ydCAqIGFzIGF3c19zZXMgZnJvbSBcIi4vYXdzLXNlc1wiO1xuZXhwb3J0ICogYXMgYXdzX3NxcyBmcm9tIFwiLi9hd3Mtc3FzXCI7XG5leHBvcnQgKiBhcyBhd3Nfc3NtIGZyb20gXCIuL2F3cy1zc21cIjtcbmV4cG9ydCAqIGFzIGF3c192cGMgZnJvbSBcIi4vYXdzLXZwY1wiO1xuZXhwb3J0ICogYXMgZG9tYWluX3JlZGlyZWN0IGZyb20gXCIuL2F3cy1wYXR0ZXJucy1kb21haW4tcmVkaXJlY3RcIjtcbmV4cG9ydCAqIGFzIHdlYnNpdGUgZnJvbSBcIi4vYXdzLXBhdHRlcm5zLXdlYnNpdGVcIjtcbmV4cG9ydCAqIGFzIHdvcmRwcmVzcyBmcm9tIFwiLi9hd3MtcGF0dGVybnMtd29yZHByZXNzXCI7XG5leHBvcnQgKiBhcyBoZWxwZXJzIGZyb20gXCIuL2hlbHBlcnNcIjtcbiJdfQ==
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "truemark-cdk-lib",
|
|
3
|
-
"version": "1.3.
|
|
3
|
+
"version": "1.3.1-alpha.1+a50134f",
|
|
4
4
|
"main": "index.js",
|
|
5
5
|
"types": "index.d.ts",
|
|
6
6
|
"scripts": {
|
|
@@ -55,8 +55,9 @@
|
|
|
55
55
|
"./aws-ecs": "./aws-ecs/index.js",
|
|
56
56
|
"./aws-lambda": "./aws-lambda/index.js",
|
|
57
57
|
"./aws-monitoring": "./aws-monitoring/index.js",
|
|
58
|
-
"./aws-patterns-wordpress": "./aws-patterns-wordpress/index.js",
|
|
59
58
|
"./aws-patterns-domain-redirect": "./aws-patterns-domain-redirect/index.js",
|
|
59
|
+
"./aws-website": "./aws-patterns-website/index.js",
|
|
60
|
+
"./aws-patterns-wordpress": "./aws-patterns-wordpress/index.js",
|
|
60
61
|
"./aws-route53": "./aws-route53/index.js",
|
|
61
62
|
"./aws-s3": "./aws-s3/index.js",
|
|
62
63
|
"./aws-ses": "./aws-ses/index.js",
|
|
@@ -64,11 +65,10 @@
|
|
|
64
65
|
"./aws-ssm": "./aws-ssm/index.js",
|
|
65
66
|
"./aws-tags": "./aws-tags/index.js",
|
|
66
67
|
"./aws-vpc": "./aws-vpc/index.js",
|
|
67
|
-
"./aws-website": "./aws-website/index.js",
|
|
68
68
|
"./helpers": "./helpers/index.js"
|
|
69
69
|
},
|
|
70
70
|
"publishConfig": {
|
|
71
71
|
"access": "public"
|
|
72
72
|
},
|
|
73
|
-
"gitHead": "
|
|
73
|
+
"gitHead": "a50134fb888ff111a9f3815c2ad13dd109f4a208"
|
|
74
74
|
}
|
|
@@ -1,187 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.Website = exports.SourceType = void 0;
|
|
4
|
-
const constructs_1 = require("constructs");
|
|
5
|
-
const aws_s3_1 = require("aws-cdk-lib/aws-s3");
|
|
6
|
-
const aws_cloudfront_1 = require("aws-cdk-lib/aws-cloudfront");
|
|
7
|
-
const aws_cloudfront_origins_1 = require("aws-cdk-lib/aws-cloudfront-origins");
|
|
8
|
-
const aws_certificatemanager_1 = require("aws-cdk-lib/aws-certificatemanager");
|
|
9
|
-
const aws_route53_1 = require("aws-cdk-lib/aws-route53");
|
|
10
|
-
const aws_route53_targets_1 = require("aws-cdk-lib/aws-route53-targets");
|
|
11
|
-
const aws_s3_deployment_1 = require("aws-cdk-lib/aws-s3-deployment");
|
|
12
|
-
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
13
|
-
const aws_route53_2 = require("../../aws-route53");
|
|
14
|
-
var SourceType;
|
|
15
|
-
(function (SourceType) {
|
|
16
|
-
SourceType["Custom"] = "Custom";
|
|
17
|
-
SourceType["Hugo"] = "Hugo";
|
|
18
|
-
SourceType["NpmDist"] = "NpmDist";
|
|
19
|
-
SourceType["Static"] = "Static";
|
|
20
|
-
})(SourceType = exports.SourceType || (exports.SourceType = {}));
|
|
21
|
-
class Website extends constructs_1.Construct {
|
|
22
|
-
constructor(scope, id, props) {
|
|
23
|
-
var _a, _b, _c, _d, _e, _f, _g, _h, _j;
|
|
24
|
-
super(scope, id);
|
|
25
|
-
const domainNames = aws_route53_2.DomainName.fromProps(props.domainNames);
|
|
26
|
-
let certificate = undefined;
|
|
27
|
-
if (props.certificateArn !== undefined) {
|
|
28
|
-
certificate = aws_certificatemanager_1.Certificate.fromCertificateArn(this, "Certificate", props.certificateArn);
|
|
29
|
-
}
|
|
30
|
-
else if (domainNames.length > 0) {
|
|
31
|
-
certificate = new aws_certificatemanager_1.Certificate(this, "Certificate", {
|
|
32
|
-
domainName: domainNames[0].toString(),
|
|
33
|
-
subjectAlternativeNames: aws_route53_2.DomainName.toStrings(domainNames).slice(1),
|
|
34
|
-
validation: aws_certificatemanager_1.CertificateValidation.fromDnsMultiZone(aws_route53_2.DomainName.toZoneMap(this, domainNames))
|
|
35
|
-
});
|
|
36
|
-
}
|
|
37
|
-
this.bucket = (_a = props.bucket) !== null && _a !== void 0 ? _a : new aws_s3_1.Bucket(this, "Bucket", {
|
|
38
|
-
encryption: aws_s3_1.BucketEncryption.S3_MANAGED,
|
|
39
|
-
removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY,
|
|
40
|
-
autoDeleteObjects: true
|
|
41
|
-
});
|
|
42
|
-
this.fallbackBucket = props.fallbackBucket;
|
|
43
|
-
this.originAccessIdentity = new aws_cloudfront_1.OriginAccessIdentity(this, "Access");
|
|
44
|
-
this.bucket.grantRead(this.originAccessIdentity);
|
|
45
|
-
if (this.fallbackBucket) {
|
|
46
|
-
this.fallbackBucket.grantRead(this.originAccessIdentity);
|
|
47
|
-
}
|
|
48
|
-
this.apexDomain = domainNames.length > 0 ? domainNames[0].toString() : "";
|
|
49
|
-
this.viewerRequestFunction = new aws_cloudfront_1.Function(this, "ViewerRequestFunction", {
|
|
50
|
-
code: aws_cloudfront_1.FunctionCode.fromInline(`
|
|
51
|
-
function handler(event) {
|
|
52
|
-
var host = event.request.headers.host.value;
|
|
53
|
-
var uri = event.request.uri;
|
|
54
|
-
var matchApex = MATCH_APEX;
|
|
55
|
-
if (matchApex && host !== "APEX_DOMAIN") {
|
|
56
|
-
return {
|
|
57
|
-
statusCode: 301,
|
|
58
|
-
statusDescription: "Permanently moved",
|
|
59
|
-
headers: {
|
|
60
|
-
"location": { "value": "https://APEX_DOMAIN" + uri }
|
|
61
|
-
}
|
|
62
|
-
}
|
|
63
|
-
}
|
|
64
|
-
if (uri.endsWith("/")) {
|
|
65
|
-
event.request.uri = uri + "index.html";
|
|
66
|
-
}
|
|
67
|
-
return event.request;
|
|
68
|
-
}`
|
|
69
|
-
.replace(/APEX_DOMAIN/g, this.apexDomain)
|
|
70
|
-
.replace(/MATCH_APEX/g, this.apexDomain !== "" && ((_b = props.redirectToApexDomain) !== null && _b !== void 0 ? _b : true) ? "true" : "false"))
|
|
71
|
-
});
|
|
72
|
-
const bucketOrigin = new aws_cloudfront_origins_1.S3Origin(this.bucket, {
|
|
73
|
-
originAccessIdentity: this.originAccessIdentity
|
|
74
|
-
});
|
|
75
|
-
const fallbackBucketOrigin = this.fallbackBucket === undefined ? undefined : new aws_cloudfront_origins_1.S3Origin(this.fallbackBucket, {
|
|
76
|
-
originAccessIdentity: this.originAccessIdentity
|
|
77
|
-
});
|
|
78
|
-
const defaultOrigin = fallbackBucketOrigin === undefined ? bucketOrigin : new aws_cloudfront_origins_1.OriginGroup({
|
|
79
|
-
primaryOrigin: bucketOrigin,
|
|
80
|
-
fallbackOrigin: fallbackBucketOrigin
|
|
81
|
-
});
|
|
82
|
-
this.distribution = new aws_cloudfront_1.Distribution(this, "Distribution", {
|
|
83
|
-
defaultBehavior: {
|
|
84
|
-
origin: defaultOrigin,
|
|
85
|
-
viewerProtocolPolicy: aws_cloudfront_1.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,
|
|
86
|
-
functionAssociations: [
|
|
87
|
-
{
|
|
88
|
-
function: this.viewerRequestFunction,
|
|
89
|
-
eventType: aws_cloudfront_1.FunctionEventType.VIEWER_REQUEST
|
|
90
|
-
}
|
|
91
|
-
]
|
|
92
|
-
},
|
|
93
|
-
errorResponses: [
|
|
94
|
-
{
|
|
95
|
-
httpStatus: 404,
|
|
96
|
-
responseHttpStatus: 404,
|
|
97
|
-
responsePagePath: "/404.html"
|
|
98
|
-
}
|
|
99
|
-
],
|
|
100
|
-
httpVersion: (_c = props.httpVersion) !== null && _c !== void 0 ? _c : aws_cloudfront_1.HttpVersion.HTTP1_1,
|
|
101
|
-
minimumProtocolVersion: (_d = props.minimumProtocolVersion) !== null && _d !== void 0 ? _d : aws_cloudfront_1.SecurityPolicyProtocol.TLS_V1_2_2021,
|
|
102
|
-
defaultRootObject: "index.html",
|
|
103
|
-
certificate,
|
|
104
|
-
domainNames: aws_route53_2.DomainName.toStrings(domainNames),
|
|
105
|
-
enableIpv6: true,
|
|
106
|
-
priceClass: (_e = props.priceClass) !== null && _e !== void 0 ? _e : aws_cloudfront_1.PriceClass.PRICE_CLASS_100
|
|
107
|
-
});
|
|
108
|
-
this.distributionUrl = `https://${(this.apexDomain != "" ? this.apexDomain : this.distribution.distributionDomainName)}`;
|
|
109
|
-
this.aRecords = [];
|
|
110
|
-
for (let domainNameProps of (_f = props.domainNames) !== null && _f !== void 0 ? _f : []) {
|
|
111
|
-
if ((_g = domainNameProps.createRecord) !== null && _g !== void 0 ? _g : true) {
|
|
112
|
-
let domainName = aws_route53_2.DomainName.findDomainName(domainNameProps, domainNames);
|
|
113
|
-
if (domainName !== undefined) {
|
|
114
|
-
this.aRecords.push(domainName.createARecord(this, aws_route53_1.RecordTarget.fromAlias(new aws_route53_targets_1.CloudFrontTarget(this.distribution))));
|
|
115
|
-
}
|
|
116
|
-
}
|
|
117
|
-
}
|
|
118
|
-
let bundlingOptions = undefined;
|
|
119
|
-
if (props.sourceType === SourceType.Custom) {
|
|
120
|
-
if (!props.sourceBundlingOptions) {
|
|
121
|
-
throw new Error("sourceBundlingOptions is required if source type is Custom");
|
|
122
|
-
}
|
|
123
|
-
bundlingOptions = props.sourceBundlingOptions;
|
|
124
|
-
}
|
|
125
|
-
if (props.sourceType === SourceType.Static) {
|
|
126
|
-
if (props.sourceBundlingOptions) {
|
|
127
|
-
throw new Error("Cannot use sourceBundlingOptions with source type Static");
|
|
128
|
-
}
|
|
129
|
-
}
|
|
130
|
-
if (props.sourceType === SourceType.Hugo) {
|
|
131
|
-
bundlingOptions = Website.HUGO_BUNDLING_OPTIONS;
|
|
132
|
-
if (props.sourceBundlingOptions) {
|
|
133
|
-
throw new Error("Cannot use sourceBundlingOptions with source type Hugo");
|
|
134
|
-
}
|
|
135
|
-
}
|
|
136
|
-
else if (props.sourceType === SourceType.NpmDist) {
|
|
137
|
-
bundlingOptions = Website.NPM_DIST_BUNDLING_OPTIONS;
|
|
138
|
-
if (props.sourceBundlingOptions) {
|
|
139
|
-
throw new Error("Cannot use sourceBundlingOptions with source type NpmDist");
|
|
140
|
-
}
|
|
141
|
-
}
|
|
142
|
-
const source = aws_s3_deployment_1.Source.asset(props.sourceDirectory, {
|
|
143
|
-
bundling: bundlingOptions
|
|
144
|
-
});
|
|
145
|
-
this.deployment = new aws_s3_deployment_1.BucketDeployment(this, "Assets", {
|
|
146
|
-
sources: [source],
|
|
147
|
-
destinationBucket: this.bucket,
|
|
148
|
-
distribution: this.distribution,
|
|
149
|
-
cacheControl: [aws_s3_deployment_1.CacheControl.maxAge((_h = props.maxAge) !== null && _h !== void 0 ? _h : aws_cdk_lib_1.Duration.minutes(60))],
|
|
150
|
-
prune: false
|
|
151
|
-
});
|
|
152
|
-
if (this.fallbackBucket) {
|
|
153
|
-
this.fallbackDeployment = new aws_s3_deployment_1.BucketDeployment(this, "FallbackAssets", {
|
|
154
|
-
sources: [source],
|
|
155
|
-
destinationBucket: this.fallbackBucket,
|
|
156
|
-
cacheControl: [aws_s3_deployment_1.CacheControl.maxAge((_j = props.maxAge) !== null && _j !== void 0 ? _j : aws_cdk_lib_1.Duration.minutes(60))],
|
|
157
|
-
prune: false
|
|
158
|
-
});
|
|
159
|
-
}
|
|
160
|
-
}
|
|
161
|
-
}
|
|
162
|
-
exports.Website = Website;
|
|
163
|
-
/**
|
|
164
|
-
* The default bundling options for SourceType.Hugo.
|
|
165
|
-
*/
|
|
166
|
-
Website.HUGO_BUNDLING_OPTIONS = {
|
|
167
|
-
image: aws_cdk_lib_1.DockerImage.fromRegistry("klakegg/hugo:latest-ext"),
|
|
168
|
-
command: [
|
|
169
|
-
"-d", "/asset-output"
|
|
170
|
-
]
|
|
171
|
-
};
|
|
172
|
-
/**
|
|
173
|
-
* The default bundling options for SourceType.NpmDist. It"s expected that "npm run dist" command
|
|
174
|
-
* will place files in ${CDK_BUNDLING_OUTPUT_DIR}. Example: "ng build --output-path=/asset-output"
|
|
175
|
-
*/
|
|
176
|
-
Website.NPM_DIST_BUNDLING_OPTIONS = {
|
|
177
|
-
image: aws_cdk_lib_1.DockerImage.fromRegistry("node:16-alpine"),
|
|
178
|
-
command: [
|
|
179
|
-
// TODO Figure out how to cache .npmcache
|
|
180
|
-
"sh", "-c", [
|
|
181
|
-
"mkdir -p .npmcache",
|
|
182
|
-
"npm ci --cache .npmcache --prefer-offline",
|
|
183
|
-
"npm run dist"
|
|
184
|
-
].join(" && ")
|
|
185
|
-
]
|
|
186
|
-
};
|
|
187
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"website.js","sourceRoot":"","sources":["website.ts"],"names":[],"mappings":"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","sourcesContent":["import {Construct} from \"constructs\";\nimport {Bucket, BucketEncryption, IBucket} from \"aws-cdk-lib/aws-s3\";\nimport {\n  Distribution,\n  Function,\n  FunctionCode,\n  FunctionEventType,\n  HttpVersion,\n  OriginAccessIdentity,\n  PriceClass,\n  SecurityPolicyProtocol,\n  ViewerProtocolPolicy\n} from \"aws-cdk-lib/aws-cloudfront\";\nimport {OriginGroup, S3Origin} from \"aws-cdk-lib/aws-cloudfront-origins\";\nimport {Certificate, CertificateValidation, ICertificate} from \"aws-cdk-lib/aws-certificatemanager\";\nimport {ARecord, RecordTarget} from \"aws-cdk-lib/aws-route53\";\nimport {CloudFrontTarget} from \"aws-cdk-lib/aws-route53-targets\";\nimport {BucketDeployment, CacheControl, Source} from \"aws-cdk-lib/aws-s3-deployment\";\nimport {BundlingOptions, DockerImage, Duration, RemovalPolicy} from \"aws-cdk-lib\";\nimport {DomainName, DomainNameProps} from \"../../aws-route53\";\n\nexport enum SourceType {\n  Custom = \"Custom\",\n  Hugo = \"Hugo\",\n  NpmDist = \"NpmDist\",\n  Static = \"Static\"\n}\n\n/**\n * Properties for WebsiteDomainName.\n */\nexport interface WebsiteDomainNameProps extends DomainNameProps {\n\n  readonly createRecord?: boolean;\n\n}\n\nexport interface WebsiteProps {\n\n  /**\n   * Bucket to use to store website content. If one is not provided, one will be generated.\n   */\n  readonly bucket?: IBucket;\n\n  /**\n   * Bucket to use if the primary bucket fails. If one is not provided a fallback is not setup.\n   */\n  readonly fallbackBucket?: IBucket;\n\n  /**\n   * The ARN to the ACM Certificate to use on the CloudFront distribution. If one is not\n   * provided and domainNames is populated, one will be generated.\n   */\n  readonly certificateArn?: string;\n\n  /**\n   * The domain names to be serviced. The first domain name in the list is treated as the apex domain.\n   */\n  readonly domainNames?: WebsiteDomainNameProps[]\n\n  /**\n   * Redirect traffic to the first domain in the list of domainNames.\n   *\n   * @default true\n   */\n  readonly redirectToApexDomain?: boolean;\n\n  /**\n   * Price class for the CloudFront distribution.\n   *\n   * @default PriceClass.PRICE_CLASS_100\n   */\n  readonly priceClass?: PriceClass\n\n  /**\n   * Minimum protocol version to allow.\n   *\n   * @default SecurityPolicyProtocol.TLS_V1_2_2021,\n   */\n  readonly minimumProtocolVersion?: SecurityPolicyProtocol;\n\n  /**\n   * Minimum HTTP version to allow.\n   *\n   * @default HttpVersion.HTTP1_1\n   */\n  readonly httpVersion?: HttpVersion;\n\n  /**\n   * Cache-control max age.\n   *\n   * @default Duration.minutes(60)\n   */\n  readonly maxAge?: Duration;\n\n  /**\n   * The source type of the website.\n   *\n   * @default SourceType.Static\n   */\n  readonly sourceType?: SourceType;\n\n  /**\n   * The directory where the website sources are located.\n   */\n  readonly sourceDirectory: string;\n\n  /**\n   * Custom bundling options to use to bundle website sources. The sourceType\n   * must be set to SourceType.Custom.\n   */\n  readonly sourceBundlingOptions?: BundlingOptions\n}\n\nexport class Website extends Construct {\n\n  /**\n   * The default bundling options for SourceType.Hugo.\n   */\n  static readonly HUGO_BUNDLING_OPTIONS: BundlingOptions = {\n    image: DockerImage.fromRegistry(\"klakegg/hugo:latest-ext\"),\n    command: [\n      \"-d\", \"/asset-output\"\n    ]\n  }\n\n  /**\n   * The default bundling options for SourceType.NpmDist. It\"s expected that \"npm run dist\" command\n   * will place files in ${CDK_BUNDLING_OUTPUT_DIR}. Example: \"ng build --output-path=/asset-output\"\n   */\n  static readonly NPM_DIST_BUNDLING_OPTIONS: BundlingOptions = {\n    image: DockerImage.fromRegistry(\"node:16-alpine\"),\n    command: [\n      // TODO Figure out how to cache .npmcache\n      \"sh\", \"-c\", [\n        \"mkdir -p .npmcache\",\n        \"npm ci --cache .npmcache --prefer-offline\",\n        \"npm run dist\"\n      ].join(\" && \")\n    ]\n  }\n\n  readonly bucket: IBucket;\n  readonly fallbackBucket?: IBucket;\n  readonly originAccessIdentity: OriginAccessIdentity;\n  readonly apexDomain: string;\n  readonly viewerRequestFunction: Function;\n  readonly distribution: Distribution;\n  readonly distributionUrl: string;\n  readonly aRecords: ARecord[];\n  readonly deployment: BucketDeployment;\n  readonly fallbackDeployment?: BucketDeployment;\n\n  constructor(scope: Construct, id: string, props: WebsiteProps) {\n    super(scope, id);\n\n    const domainNames = DomainName.fromProps(props.domainNames);\n\n    let certificate: ICertificate | undefined = undefined;\n    if (props.certificateArn !== undefined) {\n      certificate = Certificate.fromCertificateArn(this, \"Certificate\", props.certificateArn)\n    } else if (domainNames.length > 0) {\n      certificate = new Certificate(this, \"Certificate\", {\n        domainName: domainNames[0].toString(),\n        subjectAlternativeNames: DomainName.toStrings(domainNames).slice(1),\n        validation: CertificateValidation.fromDnsMultiZone(DomainName.toZoneMap(this, domainNames))\n      });\n    }\n\n    this.bucket = props.bucket ?? new Bucket(this, \"Bucket\", {\n      encryption: BucketEncryption.S3_MANAGED,\n      removalPolicy: RemovalPolicy.DESTROY,\n      autoDeleteObjects: true\n    });\n    this.fallbackBucket = props.fallbackBucket;\n\n    this.originAccessIdentity = new OriginAccessIdentity(this, \"Access\");\n    this.bucket.grantRead(this.originAccessIdentity);\n    if (this.fallbackBucket) {\n      this.fallbackBucket.grantRead(this.originAccessIdentity);\n    }\n\n    this.apexDomain = domainNames.length > 0 ? domainNames[0].toString() : \"\";\n\n    this.viewerRequestFunction = new Function(this, \"ViewerRequestFunction\", {\n      code: FunctionCode.fromInline(`\nfunction handler(event) {\n  var host = event.request.headers.host.value;\n  var uri = event.request.uri;\n  var matchApex = MATCH_APEX;\n  if (matchApex && host !== \"APEX_DOMAIN\") {\n    return {\n      statusCode: 301,\n      statusDescription: \"Permanently moved\",\n      headers: {\n        \"location\": { \"value\": \"https://APEX_DOMAIN\" + uri }\n      }\n    }\n  }\n  if (uri.endsWith(\"/\")) {\n    event.request.uri = uri + \"index.html\";\n  }\n  return event.request;\n}`\n        .replace(/APEX_DOMAIN/g, this.apexDomain)\n        .replace(/MATCH_APEX/g, this.apexDomain !== \"\" && (props.redirectToApexDomain??true) ? \"true\" : \"false\"))\n    });\n\n    const bucketOrigin = new S3Origin(this.bucket, {\n      originAccessIdentity: this.originAccessIdentity\n    });\n\n    const fallbackBucketOrigin = this.fallbackBucket === undefined ? undefined : new S3Origin(this.fallbackBucket, {\n      originAccessIdentity: this.originAccessIdentity\n    });\n\n    const defaultOrigin = fallbackBucketOrigin === undefined ? bucketOrigin : new OriginGroup({\n      primaryOrigin: bucketOrigin,\n      fallbackOrigin: fallbackBucketOrigin\n    });\n\n    this.distribution = new Distribution(this, \"Distribution\", {\n      defaultBehavior: {\n        origin: defaultOrigin,\n        viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n        functionAssociations: [\n          {\n            function: this.viewerRequestFunction,\n            eventType: FunctionEventType.VIEWER_REQUEST\n          }\n        ]\n      },\n      errorResponses: [\n        {\n          httpStatus: 404,\n          responseHttpStatus: 404,\n          responsePagePath: \"/404.html\"\n        }\n      ],\n      httpVersion: props.httpVersion??HttpVersion.HTTP1_1,\n      minimumProtocolVersion: props.minimumProtocolVersion??SecurityPolicyProtocol.TLS_V1_2_2021,\n      defaultRootObject: \"index.html\",\n      certificate,\n      domainNames: DomainName.toStrings(domainNames),\n      enableIpv6: true,\n      priceClass: props.priceClass??PriceClass.PRICE_CLASS_100\n    });\n\n    this.distributionUrl = `https://${(this.apexDomain != \"\" ? this.apexDomain : this.distribution.distributionDomainName)}`\n\n    this.aRecords = [];\n\n    for (let domainNameProps of props.domainNames ?? []) {\n      if (domainNameProps.createRecord ?? true) {\n        let domainName = DomainName.findDomainName(domainNameProps, domainNames);\n        if (domainName !== undefined) {\n          this.aRecords.push(domainName.createARecord(this,\n            RecordTarget.fromAlias(new CloudFrontTarget(this.distribution))));\n        }\n      }\n    }\n\n    let bundlingOptions: BundlingOptions | undefined = undefined\n\n    if (props.sourceType === SourceType.Custom) {\n      if (!props.sourceBundlingOptions) {\n        throw new Error(\"sourceBundlingOptions is required if source type is Custom\");\n      }\n      bundlingOptions = props.sourceBundlingOptions\n    }\n\n    if (props.sourceType === SourceType.Static) {\n      if (props.sourceBundlingOptions) {\n        throw new Error(\"Cannot use sourceBundlingOptions with source type Static\");\n      }\n    }\n\n    if (props.sourceType === SourceType.Hugo) {\n      bundlingOptions = Website.HUGO_BUNDLING_OPTIONS\n      if (props.sourceBundlingOptions) {\n        throw new Error(\"Cannot use sourceBundlingOptions with source type Hugo\");\n      }\n    } else if (props.sourceType === SourceType.NpmDist) {\n      bundlingOptions = Website.NPM_DIST_BUNDLING_OPTIONS\n      if (props.sourceBundlingOptions) {\n        throw new Error(\"Cannot use sourceBundlingOptions with source type NpmDist\");\n      }\n    }\n\n    const source = Source.asset(props.sourceDirectory, {\n      bundling: bundlingOptions\n    });\n\n    this.deployment = new BucketDeployment(this, \"Assets\", {\n      sources: [source],\n      destinationBucket: this.bucket,\n      distribution: this.distribution,\n      cacheControl: [CacheControl.maxAge(props.maxAge??Duration.minutes(60))],\n      prune: false\n    });\n\n    if (this.fallbackBucket) {\n      this.fallbackDeployment = new BucketDeployment(this, \"FallbackAssets\", {\n        sources: [source],\n        destinationBucket: this.fallbackBucket,\n        cacheControl: [CacheControl.maxAge(props.maxAge??Duration.minutes(60))],\n        prune: false\n      })\n    }\n  }\n}\n"]}
|
|
@@ -1,60 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
const index_1 = require("../index");
|
|
4
|
-
const assertions_1 = require("aws-cdk-lib/assertions");
|
|
5
|
-
const test_helper_1 = require("../../test-helper");
|
|
6
|
-
const aws_route53_1 = require("aws-cdk-lib/aws-route53");
|
|
7
|
-
test("Hugo Website Test", () => {
|
|
8
|
-
const stack = test_helper_1.TestHelper.stack();
|
|
9
|
-
new index_1.Website(stack, 'TestWebsite', {
|
|
10
|
-
sourceType: index_1.SourceType.Hugo,
|
|
11
|
-
sourceDirectory: test_helper_1.TestHelper.resolveTestFiles("hugo-website")
|
|
12
|
-
});
|
|
13
|
-
const template = assertions_1.Template.fromStack(stack);
|
|
14
|
-
template.hasResourceProperties("AWS::CloudFront::Distribution", {});
|
|
15
|
-
});
|
|
16
|
-
test("ARecord Created for Website", () => {
|
|
17
|
-
const stack = test_helper_1.TestHelper.stack();
|
|
18
|
-
new index_1.Website(stack, 'TestWebsite', {
|
|
19
|
-
sourceType: index_1.SourceType.Hugo,
|
|
20
|
-
sourceDirectory: test_helper_1.TestHelper.resolveTestFiles("hugo-website"),
|
|
21
|
-
certificateArn: "arn:aws:acm:us-east-1:123456789012:certificate/XXXX",
|
|
22
|
-
domainNames: [
|
|
23
|
-
{
|
|
24
|
-
prefix: "www",
|
|
25
|
-
// TODO In the fugure need to figure out how to mock lookups
|
|
26
|
-
zone: new aws_route53_1.HostedZone(stack, "example-com", { zoneName: "example.com" })
|
|
27
|
-
}
|
|
28
|
-
]
|
|
29
|
-
});
|
|
30
|
-
const template = assertions_1.Template.fromStack(stack);
|
|
31
|
-
// console.log(template.toJSON());
|
|
32
|
-
// TestHelper.logResources(template, "AWS::Route53::RecordSet");
|
|
33
|
-
template.hasResourceProperties("AWS::Route53::RecordSet", {});
|
|
34
|
-
});
|
|
35
|
-
// TODO We need to speed this up
|
|
36
|
-
// test("Npm Dist Website Test", () => {
|
|
37
|
-
// const stack = TestHelper.stack();
|
|
38
|
-
// new Website(stack, "TestWebsite", {
|
|
39
|
-
// sourceType: SourceType.NpmDist,
|
|
40
|
-
// sourceDirectory: TestHelper.resolveTestFiles("angular-website")
|
|
41
|
-
// });
|
|
42
|
-
// const template = Template.fromStack(stack);
|
|
43
|
-
// template.hasResourceProperties("AWS::CloudFront::Distribution", {});
|
|
44
|
-
// });
|
|
45
|
-
// TODO Need to fix
|
|
46
|
-
// test("Website Certificate Test", () => {
|
|
47
|
-
// const stack = TestHelper.stack();
|
|
48
|
-
// new Website(stack, "TestWebsite", {
|
|
49
|
-
// sourceType: SourceType.Hugo,
|
|
50
|
-
// sourceDirectory: TestHelper.resolveTestFiles("hugo-website"),
|
|
51
|
-
// domainNames: [
|
|
52
|
-
// new DomainName("test", "example.com"),
|
|
53
|
-
// new DomainName("test2", "example.com")
|
|
54
|
-
// ]
|
|
55
|
-
// });
|
|
56
|
-
// const template = Template.fromStack(stack);
|
|
57
|
-
// console.log(template.toJSON());
|
|
58
|
-
// // TestHelper.logResources()
|
|
59
|
-
// });
|
|
60
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoid2Vic2l0ZS50ZXN0LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsid2Vic2l0ZS50ZXN0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7O0FBQUEsb0NBQTZDO0FBQzdDLHVEQUFnRDtBQUNoRCxtREFBNkM7QUFDN0MseURBQW1EO0FBRW5ELElBQUksQ0FBQyxtQkFBbUIsRUFBRSxHQUFHLEVBQUU7SUFDN0IsTUFBTSxLQUFLLEdBQUcsd0JBQVUsQ0FBQyxLQUFLLEVBQUUsQ0FBQztJQUNqQyxJQUFJLGVBQU8sQ0FBQyxLQUFLLEVBQUUsYUFBYSxFQUFFO1FBQ2hDLFVBQVUsRUFBRSxrQkFBVSxDQUFDLElBQUk7UUFDM0IsZUFBZSxFQUFFLHdCQUFVLENBQUMsZ0JBQWdCLENBQUMsY0FBYyxDQUFDO0tBQzdELENBQUMsQ0FBQztJQUNILE1BQU0sUUFBUSxHQUFHLHFCQUFRLENBQUMsU0FBUyxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQzNDLFFBQVEsQ0FBQyxxQkFBcUIsQ0FBQywrQkFBK0IsRUFBRSxFQUFFLENBQUMsQ0FBQztBQUN0RSxDQUFDLENBQUMsQ0FBQztBQUVILElBQUksQ0FBQyw2QkFBNkIsRUFBRSxHQUFHLEVBQUU7SUFDdkMsTUFBTSxLQUFLLEdBQUcsd0JBQVUsQ0FBQyxLQUFLLEVBQUUsQ0FBQztJQUNqQyxJQUFJLGVBQU8sQ0FBQyxLQUFLLEVBQUUsYUFBYSxFQUFFO1FBQ2hDLFVBQVUsRUFBRSxrQkFBVSxDQUFDLElBQUk7UUFDM0IsZUFBZSxFQUFFLHdCQUFVLENBQUMsZ0JBQWdCLENBQUMsY0FBYyxDQUFDO1FBQzVELGNBQWMsRUFBRSxxREFBcUQ7UUFDckUsV0FBVyxFQUFFO1lBQ1g7Z0JBQ0UsTUFBTSxFQUFFLEtBQUs7Z0JBQ2IsNERBQTREO2dCQUM1RCxJQUFJLEVBQUUsSUFBSSx3QkFBVSxDQUFDLEtBQUssRUFBRSxhQUFhLEVBQUUsRUFBQyxRQUFRLEVBQUUsYUFBYSxFQUFDLENBQUM7YUFDdEU7U0FDRjtLQUNGLENBQUMsQ0FBQztJQUNILE1BQU0sUUFBUSxHQUFHLHFCQUFRLENBQUMsU0FBUyxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQzNDLGtDQUFrQztJQUNsQyxnRUFBZ0U7SUFDaEUsUUFBUSxDQUFDLHFCQUFxQixDQUFDLHlCQUF5QixFQUFFLEVBQUUsQ0FBQyxDQUFDO0FBQ2hFLENBQUMsQ0FBQyxDQUFDO0FBRUgsZ0NBQWdDO0FBQ2hDLHdDQUF3QztBQUN4QyxzQ0FBc0M7QUFDdEMsd0NBQXdDO0FBQ3hDLHNDQUFzQztBQUN0QyxzRUFBc0U7QUFDdEUsUUFBUTtBQUNSLGdEQUFnRDtBQUNoRCx5RUFBeUU7QUFDekUsTUFBTTtBQUVOLG1CQUFtQjtBQUNuQiwyQ0FBMkM7QUFDM0Msc0NBQXNDO0FBQ3RDLHdDQUF3QztBQUN4QyxtQ0FBbUM7QUFDbkMsb0VBQW9FO0FBQ3BFLHFCQUFxQjtBQUNyQiwrQ0FBK0M7QUFDL0MsK0NBQStDO0FBQy9DLFFBQVE7QUFDUixRQUFRO0FBQ1IsZ0RBQWdEO0FBQ2hELG9DQUFvQztBQUNwQyxpQ0FBaUM7QUFDakMsTUFBTSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7U291cmNlVHlwZSwgV2Vic2l0ZX0gZnJvbSBcIi4uL2luZGV4XCI7XG5pbXBvcnQge1RlbXBsYXRlfSBmcm9tIFwiYXdzLWNkay1saWIvYXNzZXJ0aW9uc1wiO1xuaW1wb3J0IHtUZXN0SGVscGVyfSBmcm9tIFwiLi4vLi4vdGVzdC1oZWxwZXJcIjtcbmltcG9ydCB7SG9zdGVkWm9uZX0gZnJvbSBcImF3cy1jZGstbGliL2F3cy1yb3V0ZTUzXCI7XG5cbnRlc3QoXCJIdWdvIFdlYnNpdGUgVGVzdFwiLCAoKSA9PiB7XG4gIGNvbnN0IHN0YWNrID0gVGVzdEhlbHBlci5zdGFjaygpO1xuICBuZXcgV2Vic2l0ZShzdGFjaywgJ1Rlc3RXZWJzaXRlJywge1xuICAgIHNvdXJjZVR5cGU6IFNvdXJjZVR5cGUuSHVnbyxcbiAgICBzb3VyY2VEaXJlY3Rvcnk6IFRlc3RIZWxwZXIucmVzb2x2ZVRlc3RGaWxlcyhcImh1Z28td2Vic2l0ZVwiKVxuICB9KTtcbiAgY29uc3QgdGVtcGxhdGUgPSBUZW1wbGF0ZS5mcm9tU3RhY2soc3RhY2spO1xuICB0ZW1wbGF0ZS5oYXNSZXNvdXJjZVByb3BlcnRpZXMoXCJBV1M6OkNsb3VkRnJvbnQ6OkRpc3RyaWJ1dGlvblwiLCB7fSk7XG59KTtcblxudGVzdChcIkFSZWNvcmQgQ3JlYXRlZCBmb3IgV2Vic2l0ZVwiLCAoKSA9PiB7XG4gIGNvbnN0IHN0YWNrID0gVGVzdEhlbHBlci5zdGFjaygpO1xuICBuZXcgV2Vic2l0ZShzdGFjaywgJ1Rlc3RXZWJzaXRlJywge1xuICAgIHNvdXJjZVR5cGU6IFNvdXJjZVR5cGUuSHVnbyxcbiAgICBzb3VyY2VEaXJlY3Rvcnk6IFRlc3RIZWxwZXIucmVzb2x2ZVRlc3RGaWxlcyhcImh1Z28td2Vic2l0ZVwiKSxcbiAgICBjZXJ0aWZpY2F0ZUFybjogXCJhcm46YXdzOmFjbTp1cy1lYXN0LTE6MTIzNDU2Nzg5MDEyOmNlcnRpZmljYXRlL1hYWFhcIixcbiAgICBkb21haW5OYW1lczogW1xuICAgICAge1xuICAgICAgICBwcmVmaXg6IFwid3d3XCIsXG4gICAgICAgIC8vIFRPRE8gSW4gdGhlIGZ1Z3VyZSBuZWVkIHRvIGZpZ3VyZSBvdXQgaG93IHRvIG1vY2sgbG9va3Vwc1xuICAgICAgICB6b25lOiBuZXcgSG9zdGVkWm9uZShzdGFjaywgXCJleGFtcGxlLWNvbVwiLCB7em9uZU5hbWU6IFwiZXhhbXBsZS5jb21cIn0pXG4gICAgICB9XG4gICAgXVxuICB9KTtcbiAgY29uc3QgdGVtcGxhdGUgPSBUZW1wbGF0ZS5mcm9tU3RhY2soc3RhY2spO1xuICAvLyBjb25zb2xlLmxvZyh0ZW1wbGF0ZS50b0pTT04oKSk7XG4gIC8vIFRlc3RIZWxwZXIubG9nUmVzb3VyY2VzKHRlbXBsYXRlLCBcIkFXUzo6Um91dGU1Mzo6UmVjb3JkU2V0XCIpO1xuICB0ZW1wbGF0ZS5oYXNSZXNvdXJjZVByb3BlcnRpZXMoXCJBV1M6OlJvdXRlNTM6OlJlY29yZFNldFwiLCB7fSk7XG59KTtcblxuLy8gVE9ETyBXZSBuZWVkIHRvIHNwZWVkIHRoaXMgdXBcbi8vIHRlc3QoXCJOcG0gRGlzdCBXZWJzaXRlIFRlc3RcIiwgKCkgPT4ge1xuLy8gICBjb25zdCBzdGFjayA9IFRlc3RIZWxwZXIuc3RhY2soKTtcbi8vICAgbmV3IFdlYnNpdGUoc3RhY2ssIFwiVGVzdFdlYnNpdGVcIiwge1xuLy8gICAgIHNvdXJjZVR5cGU6IFNvdXJjZVR5cGUuTnBtRGlzdCxcbi8vICAgICBzb3VyY2VEaXJlY3Rvcnk6IFRlc3RIZWxwZXIucmVzb2x2ZVRlc3RGaWxlcyhcImFuZ3VsYXItd2Vic2l0ZVwiKVxuLy8gICB9KTtcbi8vICAgY29uc3QgdGVtcGxhdGUgPSBUZW1wbGF0ZS5mcm9tU3RhY2soc3RhY2spO1xuLy8gICB0ZW1wbGF0ZS5oYXNSZXNvdXJjZVByb3BlcnRpZXMoXCJBV1M6OkNsb3VkRnJvbnQ6OkRpc3RyaWJ1dGlvblwiLCB7fSk7XG4vLyB9KTtcblxuLy8gVE9ETyBOZWVkIHRvIGZpeFxuLy8gdGVzdChcIldlYnNpdGUgQ2VydGlmaWNhdGUgVGVzdFwiLCAoKSA9PiB7XG4vLyAgIGNvbnN0IHN0YWNrID0gVGVzdEhlbHBlci5zdGFjaygpO1xuLy8gICBuZXcgV2Vic2l0ZShzdGFjaywgXCJUZXN0V2Vic2l0ZVwiLCB7XG4vLyAgICAgc291cmNlVHlwZTogU291cmNlVHlwZS5IdWdvLFxuLy8gICAgIHNvdXJjZURpcmVjdG9yeTogVGVzdEhlbHBlci5yZXNvbHZlVGVzdEZpbGVzKFwiaHVnby13ZWJzaXRlXCIpLFxuLy8gICAgIGRvbWFpbk5hbWVzOiBbXG4vLyAgICAgICBuZXcgRG9tYWluTmFtZShcInRlc3RcIiwgXCJleGFtcGxlLmNvbVwiKSxcbi8vICAgICAgIG5ldyBEb21haW5OYW1lKFwidGVzdDJcIiwgXCJleGFtcGxlLmNvbVwiKVxuLy8gICAgIF1cbi8vICAgfSk7XG4vLyAgIGNvbnN0IHRlbXBsYXRlID0gVGVtcGxhdGUuZnJvbVN0YWNrKHN0YWNrKTtcbi8vICAgY29uc29sZS5sb2codGVtcGxhdGUudG9KU09OKCkpO1xuLy8gICAvLyBUZXN0SGVscGVyLmxvZ1Jlc291cmNlcygpXG4vLyB9KTtcbiJdfQ==
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|