@pwrdrvr/microapps-cdk 0.2.9 → 0.2.12
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/.gitattributes +18 -18
- package/.jsii +1264 -165
- package/API.md +511 -8
- package/README.md +29 -0
- package/changelog.md +2 -0
- package/lib/MicroApps.d.ts +126 -105
- package/lib/MicroApps.d.ts.map +1 -0
- package/lib/MicroApps.js +32 -30
- package/lib/MicroAppsAPIGwy.d.ts +28 -61
- package/lib/MicroAppsAPIGwy.d.ts.map +1 -0
- package/lib/MicroAppsAPIGwy.js +14 -23
- package/lib/MicroAppsCF.d.ts +85 -104
- package/lib/MicroAppsCF.d.ts.map +1 -0
- package/lib/MicroAppsCF.js +58 -41
- package/lib/MicroAppsEdgeToOrigin.d.ts +108 -0
- package/lib/MicroAppsEdgeToOrigin.d.ts.map +1 -0
- package/lib/MicroAppsEdgeToOrigin.js +139 -0
- package/lib/MicroAppsS3.d.ts +19 -69
- package/lib/MicroAppsS3.d.ts.map +1 -0
- package/lib/MicroAppsS3.js +5 -32
- package/lib/MicroAppsSvcs.d.ts +58 -86
- package/lib/MicroAppsSvcs.d.ts.map +1 -0
- package/lib/MicroAppsSvcs.js +36 -95
- package/lib/MicroAppsTable.d.ts +60 -0
- package/lib/MicroAppsTable.d.ts.map +1 -0
- package/lib/MicroAppsTable.js +42 -0
- package/lib/index.d.ts +3 -0
- package/lib/index.d.ts.map +1 -0
- package/lib/index.js +3 -1
- package/lib/microapps-deployer/index.js +174 -113
- package/lib/microapps-deployer/index.js.map +3 -3
- package/lib/microapps-edge-to-origin/index.js +97 -0
- package/lib/microapps-edge-to-origin/index.js.map +7 -0
- package/lib/microapps-router/index.js +54 -69
- package/lib/microapps-router/index.js.map +3 -3
- package/lib/microapps-router/templates/appFrame.html +0 -0
- package/lib/utils/ReverseDomain.d.ts +1 -0
- package/lib/utils/ReverseDomain.d.ts.map +1 -0
- package/package.json +29 -24
- package/releasetag.txt +1 -0
- package/version.txt +1 -0
- package/patches/@aws-cdk+aws-apigatewayv2-alpha+2.8.0-alpha.0.patch +0 -39
package/lib/MicroAppsCF.js
CHANGED
|
@@ -12,14 +12,9 @@ const r53targets = require("aws-cdk-lib/aws-route53-targets");
|
|
|
12
12
|
const constructs_1 = require("constructs");
|
|
13
13
|
const ReverseDomain_1 = require("./utils/ReverseDomain");
|
|
14
14
|
/**
|
|
15
|
-
*
|
|
16
|
-
*
|
|
17
|
-
* @experimental
|
|
15
|
+
* Create a new MicroApps CloudFront Distribution.
|
|
18
16
|
*/
|
|
19
17
|
class MicroAppsCF extends constructs_1.Construct {
|
|
20
|
-
/**
|
|
21
|
-
* @experimental
|
|
22
|
-
*/
|
|
23
18
|
constructor(scope, id, props) {
|
|
24
19
|
super(scope, id);
|
|
25
20
|
if (props === undefined) {
|
|
@@ -29,7 +24,7 @@ class MicroAppsCF extends constructs_1.Construct {
|
|
|
29
24
|
(props.r53Zone !== undefined && props.domainNameEdge === undefined)) {
|
|
30
25
|
throw new Error('If either of r53Zone or domainNameEdge are set then the other must be set');
|
|
31
26
|
}
|
|
32
|
-
const { domainNameEdge, domainNameOrigin, httpApi, removalPolicy, certEdge, assetNameRoot, assetNameSuffix, r53Zone, bucketLogs, bucketAppsOrigin, rootPathPrefix, createAPIPathRoute = true, } = props;
|
|
27
|
+
const { domainNameEdge, domainNameOrigin, httpApi, removalPolicy, certEdge, assetNameRoot, assetNameSuffix, r53Zone, bucketLogs, bucketAppsOrigin, rootPathPrefix, createAPIPathRoute = true, createNextDataPathRoute = true, edgeToOriginLambdas, } = props;
|
|
33
28
|
const apigwyOriginRequestPolicy = MicroAppsCF.createAPIOriginPolicy(this, {
|
|
34
29
|
assetNameRoot,
|
|
35
30
|
assetNameSuffix,
|
|
@@ -46,6 +41,9 @@ class MicroAppsCF extends constructs_1.Construct {
|
|
|
46
41
|
httpOriginFQDN = `${httpApi.apiId}.execute-api.${aws_cdk_lib_1.Aws.REGION}.amazonaws.com`;
|
|
47
42
|
}
|
|
48
43
|
//
|
|
44
|
+
// Get the Edge to Origin Lambdas
|
|
45
|
+
//
|
|
46
|
+
//
|
|
49
47
|
// CloudFront Distro
|
|
50
48
|
//
|
|
51
49
|
const apiGwyOrigin = new cforigins.HttpOrigin(httpOriginFQDN, {
|
|
@@ -64,6 +62,7 @@ class MicroAppsCF extends constructs_1.Construct {
|
|
|
64
62
|
originRequestPolicy: apigwyOriginRequestPolicy,
|
|
65
63
|
origin: apiGwyOrigin,
|
|
66
64
|
viewerProtocolPolicy: cf.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,
|
|
65
|
+
edgeLambdas: edgeToOriginLambdas,
|
|
67
66
|
},
|
|
68
67
|
enableIpv6: true,
|
|
69
68
|
priceClass: cf.PriceClass.PRICE_CLASS_100,
|
|
@@ -83,6 +82,8 @@ class MicroAppsCF extends constructs_1.Construct {
|
|
|
83
82
|
apigwyOriginRequestPolicy: apigwyOriginRequestPolicy,
|
|
84
83
|
rootPathPrefix,
|
|
85
84
|
createAPIPathRoute,
|
|
85
|
+
createNextDataPathRoute,
|
|
86
|
+
apigwyEdgeFunctions: edgeToOriginLambdas,
|
|
86
87
|
});
|
|
87
88
|
//
|
|
88
89
|
// Create the edge name for the CloudFront distro
|
|
@@ -100,7 +101,7 @@ class MicroAppsCF extends constructs_1.Construct {
|
|
|
100
101
|
}
|
|
101
102
|
}
|
|
102
103
|
/**
|
|
103
|
-
*
|
|
104
|
+
* Create or get the origin request policy
|
|
104
105
|
*
|
|
105
106
|
* If a custom domain name is NOT used for the origin then a policy
|
|
106
107
|
* will be created.
|
|
@@ -109,39 +110,49 @@ class MicroAppsCF extends constructs_1.Construct {
|
|
|
109
110
|
* policy will be returned. This policy passes the Host header to the
|
|
110
111
|
* origin, which is fine when using a custom domain name on the origin.
|
|
111
112
|
*
|
|
112
|
-
* @
|
|
113
|
+
* @param _scope
|
|
114
|
+
* @param _props
|
|
113
115
|
*/
|
|
114
|
-
static createAPIOriginPolicy(
|
|
115
|
-
const { assetNameRoot, assetNameSuffix, domainNameEdge } = props;
|
|
116
|
-
let apigwyOriginRequestPolicy = cf.OriginRequestPolicy.ALL_VIEWER;
|
|
117
|
-
if (domainNameEdge === undefined) {
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
116
|
+
static createAPIOriginPolicy(_scope, _props) {
|
|
117
|
+
// const { assetNameRoot, assetNameSuffix, domainNameEdge } = props;
|
|
118
|
+
// let apigwyOriginRequestPolicy: cf.IOriginRequestPolicy = cf.OriginRequestPolicy.ALL_VIEWER;
|
|
119
|
+
// if (domainNameEdge === undefined) {
|
|
120
|
+
// // When not using a custom domain name we must limit down the origin policy to
|
|
121
|
+
// // prevent it from passing the Host header (distribution_id.cloudfront.net) to
|
|
122
|
+
// // apigwy which will then reject it with a 403 because it does not match the
|
|
123
|
+
// // execute-api name that apigwy is expecting.
|
|
124
|
+
// //
|
|
125
|
+
// // 2021-12-28 - There is a bug in the name generation that causes the same asset
|
|
126
|
+
// // in different stacks to have the same generated name. We have to make the id
|
|
127
|
+
// // in all cases to ensure the generated name is unique.
|
|
128
|
+
// apigwyOriginRequestPolicy = new cf.OriginRequestPolicy(
|
|
129
|
+
// scope,
|
|
130
|
+
// `apigwy-origin-policy-${Stack.of(scope).stackName}`,
|
|
131
|
+
// {
|
|
132
|
+
// comment: assetNameRoot ? `${assetNameRoot}-apigwy${assetNameSuffix}` : undefined,
|
|
133
|
+
// originRequestPolicyName: assetNameRoot
|
|
134
|
+
// ? `${assetNameRoot}-apigwy${assetNameSuffix}`
|
|
135
|
+
// : undefined,
|
|
136
|
+
// cookieBehavior: cf.OriginRequestCookieBehavior.all(),
|
|
137
|
+
// queryStringBehavior: cf.OriginRequestQueryStringBehavior.all(),
|
|
138
|
+
// // TODO: If signing is enabled this should forward all signature headers
|
|
139
|
+
// // TODO: If set to "cfront.OriginRequestHeaderBehavior.all()" then
|
|
140
|
+
// // `replaceHostHeader` must be set to true to prevent API Gateway from rejecting
|
|
141
|
+
// // the request
|
|
142
|
+
// // headerBehavior: cf.OriginRequestHeaderBehavior.allowList('user-agent', 'referer'),
|
|
143
|
+
// headerBehavior: cf.OriginRequestHeaderBehavior.all(),
|
|
144
|
+
// },
|
|
145
|
+
// );
|
|
146
|
+
// }
|
|
147
|
+
return cf.OriginRequestPolicy.ALL_VIEWER;
|
|
137
148
|
}
|
|
138
149
|
/**
|
|
139
|
-
*
|
|
140
|
-
*
|
|
141
|
-
* @
|
|
150
|
+
* Add API Gateway and S3 routes to an existing CloudFront Distribution
|
|
151
|
+
* @param _scope
|
|
152
|
+
* @param props
|
|
142
153
|
*/
|
|
143
154
|
static addRoutes(_scope, props) {
|
|
144
|
-
const { apiGwyOrigin, bucketAppsOrigin, distro, apigwyOriginRequestPolicy, rootPathPrefix = '', createAPIPathRoute = true, } = props;
|
|
155
|
+
const { apiGwyOrigin, bucketAppsOrigin, distro, apigwyOriginRequestPolicy, rootPathPrefix = '', createAPIPathRoute = true, createNextDataPathRoute = true, } = props;
|
|
145
156
|
//
|
|
146
157
|
// Add Behaviors
|
|
147
158
|
//
|
|
@@ -159,6 +170,7 @@ class MicroAppsCF extends constructs_1.Construct {
|
|
|
159
170
|
compress: true,
|
|
160
171
|
originRequestPolicy: apigwyOriginRequestPolicy,
|
|
161
172
|
viewerProtocolPolicy: cf.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,
|
|
173
|
+
edgeLambdas: props.apigwyEdgeFunctions,
|
|
162
174
|
};
|
|
163
175
|
//
|
|
164
176
|
// If a route specifically has `/api/` in it, send it to API Gateway
|
|
@@ -169,6 +181,14 @@ class MicroAppsCF extends constructs_1.Construct {
|
|
|
169
181
|
distro.addBehavior(path_1.posix.join(rootPathPrefix, '/*/*/api/*'), apiGwyOrigin, apiGwyBehaviorOptions);
|
|
170
182
|
}
|
|
171
183
|
//
|
|
184
|
+
// If a route specifically has `/_next/data/` in it, send it to API Gateway
|
|
185
|
+
// This is needed to catch routes that have periods in the API path data,
|
|
186
|
+
// such as: /release/0.0.0/_next/data/app.json
|
|
187
|
+
//
|
|
188
|
+
if (createNextDataPathRoute) {
|
|
189
|
+
distro.addBehavior(path_1.posix.join(rootPathPrefix, '/*/*/_next/data/*'), apiGwyOrigin, apiGwyBehaviorOptions);
|
|
190
|
+
}
|
|
191
|
+
//
|
|
172
192
|
// All static assets are assumed to have a dot in them
|
|
173
193
|
//
|
|
174
194
|
distro.addBehavior(path_1.posix.join(rootPathPrefix, '/*/*/*.*'), bucketAppsOrigin, s3BehaviorOptions);
|
|
@@ -179,14 +199,11 @@ class MicroAppsCF extends constructs_1.Construct {
|
|
|
179
199
|
//
|
|
180
200
|
distro.addBehavior(path_1.posix.join(rootPathPrefix, '/*'), apiGwyOrigin, apiGwyBehaviorOptions);
|
|
181
201
|
}
|
|
182
|
-
/**
|
|
183
|
-
* @experimental
|
|
184
|
-
*/
|
|
185
202
|
get cloudFrontDistro() {
|
|
186
203
|
return this._cloudFrontDistro;
|
|
187
204
|
}
|
|
188
205
|
}
|
|
189
206
|
exports.MicroAppsCF = MicroAppsCF;
|
|
190
207
|
_a = JSII_RTTI_SYMBOL_1;
|
|
191
|
-
MicroAppsCF[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroAppsCF", version: "0.2.
|
|
192
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"MicroAppsCF.js","sourceRoot":"","sources":["../src/MicroAppsCF.ts"],"names":[],"mappings":"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","sourcesContent":["import { posix as posixPath } from 'path';\nimport * as apigwy from '@aws-cdk/aws-apigatewayv2-alpha';\nimport { Aws, RemovalPolicy, Stack } from 'aws-cdk-lib';\n// import * as apigwycfn from 'aws-cdk-lib/aws-apigatewayv2';\nimport * as acm from 'aws-cdk-lib/aws-certificatemanager';\nimport * as cf from 'aws-cdk-lib/aws-cloudfront';\nimport * as cforigins from 'aws-cdk-lib/aws-cloudfront-origins';\nimport * as r53 from 'aws-cdk-lib/aws-route53';\nimport * as r53targets from 'aws-cdk-lib/aws-route53-targets';\nimport * as s3 from 'aws-cdk-lib/aws-s3';\nimport { Construct } from 'constructs';\nimport { reverseDomain } from './utils/ReverseDomain';\n\n                                            \nexport interface IMicroAppsCF {\n  readonly cloudFrontDistro: cf.Distribution;\n}\n\n                                                                 \nexport interface MicroAppsCFProps {\n                                                                                                                                                                                                 \n  readonly removalPolicy?: RemovalPolicy;\n\n                                                           \n  readonly bucketAppsOrigin: cforigins.S3Origin;\n\n                                              \n  readonly bucketLogs?: s3.IBucket;\n\n                                                                                                                    \n  readonly domainNameEdge?: string;\n\n                                                                                                                                              \n  readonly domainNameOrigin?: string;\n\n                                                 \n  readonly httpApi: apigwy.HttpApi;\n\n                                                                                                                   \n  readonly assetNameRoot?: string;\n\n                                                                                            \n  readonly assetNameSuffix?: string;\n\n                                                                  \n  readonly certEdge?: acm.ICertificate;\n\n                                                                                 \n  readonly r53Zone?: r53.IHostedZone;\n\n                                                                                               \n  readonly rootPathPrefix?: string;\n\n                                                                                                                                                                                                                                                                                                                                                                    \n  readonly createAPIPathRoute?: boolean;\n}\n\n                                                  \nexport interface CreateAPIOriginPolicyOptions {\n                                                                                                                   \n  readonly assetNameRoot?: string;\n\n                                                                                            \n  readonly assetNameSuffix?: string;\n\n                                                                                                                                                                                   \n  readonly domainNameEdge?: string;\n}\n\n                                  \nexport interface AddRoutesOptions {\n                                                            \n  readonly apiGwyOrigin: cf.IOrigin;\n\n                                                              \n  readonly bucketAppsOrigin: cforigins.S3Origin;\n\n                                                                         \n  readonly distro: cf.Distribution;\n\n                                                             \n  readonly apigwyOriginRequestPolicy: cf.IOriginRequestPolicy;\n\n                                                                                               \n  readonly rootPathPrefix?: string;\n\n                                                                                                                                                                                                                                                                                                                                                                    \n  readonly createAPIPathRoute?: boolean;\n}\n\n                                                          \nexport class MicroAppsCF extends Construct implements IMicroAppsCF {\n                                                                                                                                                                                                                                                                                                                                                                                                                                  \n  public static createAPIOriginPolicy(\n    scope: Construct,\n    props: CreateAPIOriginPolicyOptions,\n  ): cf.IOriginRequestPolicy {\n    const { assetNameRoot, assetNameSuffix, domainNameEdge } = props;\n\n    let apigwyOriginRequestPolicy: cf.IOriginRequestPolicy = cf.OriginRequestPolicy.ALL_VIEWER;\n    if (domainNameEdge === undefined) {\n      // When not using a custom domain name we must limit down the origin policy to\n      // prevent it from passing the Host header (distribution_id.cloudfront.net) to\n      // apigwy which will then reject it with a 403 because it does not match the\n      // execute-api name that apigwy is expecting.\n      //\n      // 2021-12-28 - There is a bug in the name generation that causes the same asset\n      // in different stacks to have the same generated name.  We have to make the id\n      // in all cases to ensure the generated name is unique.\n      apigwyOriginRequestPolicy = new cf.OriginRequestPolicy(\n        scope,\n        `apigwy-origin-policy-${Stack.of(scope).stackName}`,\n        {\n          comment: assetNameRoot ? `${assetNameRoot}-apigwy${assetNameSuffix}` : undefined,\n\n          originRequestPolicyName: assetNameRoot\n            ? `${assetNameRoot}-apigwy${assetNameSuffix}`\n            : undefined,\n          cookieBehavior: cf.OriginRequestCookieBehavior.all(),\n          queryStringBehavior: cf.OriginRequestQueryStringBehavior.all(),\n          headerBehavior: cf.OriginRequestHeaderBehavior.allowList('user-agent', 'referer'),\n        },\n      );\n    }\n\n    return apigwyOriginRequestPolicy;\n  }\n\n                                                                                                                          \n  public static addRoutes(_scope: Construct, props: AddRoutesOptions) {\n    const {\n      apiGwyOrigin,\n      bucketAppsOrigin,\n      distro,\n      apigwyOriginRequestPolicy,\n      rootPathPrefix = '',\n      createAPIPathRoute = true,\n    } = props;\n\n    //\n    // Add Behaviors\n    //\n    const s3BehaviorOptions: cf.AddBehaviorOptions = {\n      allowedMethods: cf.AllowedMethods.ALLOW_GET_HEAD_OPTIONS,\n      cachePolicy: cf.CachePolicy.CACHING_OPTIMIZED,\n      compress: true,\n      originRequestPolicy: cf.OriginRequestPolicy.CORS_S3_ORIGIN,\n      viewerProtocolPolicy: cf.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n    };\n    const apiGwyBehaviorOptions: cf.AddBehaviorOptions = {\n      allowedMethods: cf.AllowedMethods.ALLOW_ALL,\n      // TODO: Caching needs to be set by the app response\n      cachePolicy: cf.CachePolicy.CACHING_DISABLED,\n      compress: true,\n      originRequestPolicy: apigwyOriginRequestPolicy,\n      viewerProtocolPolicy: cf.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n    };\n\n    //\n    // If a route specifically has `/api/` in it, send it to API Gateway\n    // This is needed to catch routes that have periods in the API path data,\n    // such as: /release/0.0.0/api/update/default/release/0.0.0\n    //\n    if (createAPIPathRoute) {\n      distro.addBehavior(\n        posixPath.join(rootPathPrefix, '/*/*/api/*'),\n        apiGwyOrigin,\n        apiGwyBehaviorOptions,\n      );\n    }\n\n    //\n    // All static assets are assumed to have a dot in them\n    //\n    distro.addBehavior(\n      posixPath.join(rootPathPrefix, '/*/*/*.*'),\n      bucketAppsOrigin,\n      s3BehaviorOptions,\n    );\n\n    //\n    // Everything that isn't a static asset is going to API Gateway\n    // There is no trailing slash because Serverless Next.js wants\n    // go load pages at /release/0.0.3 (with no trailing slash).\n    //\n    distro.addBehavior(posixPath.join(rootPathPrefix, '/*'), apiGwyOrigin, apiGwyBehaviorOptions);\n  }\n\n  private _cloudFrontDistro: cf.Distribution;\n  public get cloudFrontDistro(): cf.Distribution {\n    return this._cloudFrontDistro;\n  }\n\n  constructor(scope: Construct, id: string, props: MicroAppsCFProps) {\n    super(scope, id);\n\n    if (props === undefined) {\n      throw new Error('props must be set');\n    }\n\n    if (\n      (props.r53Zone === undefined && props.domainNameEdge !== undefined) ||\n      (props.r53Zone !== undefined && props.domainNameEdge === undefined)\n    ) {\n      throw new Error('If either of r53Zone or domainNameEdge are set then the other must be set');\n    }\n\n    const {\n      domainNameEdge,\n      domainNameOrigin,\n      httpApi,\n      removalPolicy,\n      certEdge,\n      assetNameRoot,\n      assetNameSuffix,\n      r53Zone,\n      bucketLogs,\n      bucketAppsOrigin,\n      rootPathPrefix,\n      createAPIPathRoute = true,\n    } = props;\n\n    const apigwyOriginRequestPolicy = MicroAppsCF.createAPIOriginPolicy(this, {\n      assetNameRoot,\n      assetNameSuffix,\n      domainNameEdge,\n    });\n\n    //\n    // Determine URL of the origin FQDN\n    //\n    let httpOriginFQDN: string = 'invalid.pwrdrvr.com';\n    if (domainNameOrigin !== undefined) {\n      httpOriginFQDN = domainNameOrigin;\n    } else {\n      httpOriginFQDN = `${httpApi.apiId}.execute-api.${Aws.REGION}.amazonaws.com`;\n    }\n\n    //\n    // CloudFront Distro\n    //\n    const apiGwyOrigin = new cforigins.HttpOrigin(httpOriginFQDN, {\n      protocolPolicy: cf.OriginProtocolPolicy.HTTPS_ONLY,\n      originSslProtocols: [cf.OriginSslPolicy.TLS_V1_2],\n    });\n    this._cloudFrontDistro = new cf.Distribution(this, 'cft', {\n      comment: assetNameRoot ? `${assetNameRoot}${assetNameSuffix}` : domainNameEdge,\n      domainNames: domainNameEdge !== undefined ? [domainNameEdge] : undefined,\n      certificate: certEdge,\n      httpVersion: cf.HttpVersion.HTTP2,\n      defaultBehavior: {\n        allowedMethods: cf.AllowedMethods.ALLOW_ALL,\n        cachePolicy: cf.CachePolicy.CACHING_DISABLED,\n        compress: true,\n        originRequestPolicy: apigwyOriginRequestPolicy,\n        origin: apiGwyOrigin,\n        viewerProtocolPolicy: cf.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n      },\n      enableIpv6: true,\n      priceClass: cf.PriceClass.PRICE_CLASS_100,\n      logBucket: bucketLogs,\n      logFilePrefix: props.domainNameEdge\n        ? `${reverseDomain(props.domainNameEdge)}/cloudfront-raw/`\n        : undefined,\n    });\n    if (removalPolicy !== undefined) {\n      this._cloudFrontDistro.applyRemovalPolicy(removalPolicy);\n    }\n\n    // Add routes to the CloudFront Distribution\n    MicroAppsCF.addRoutes(scope, {\n      apiGwyOrigin,\n      bucketAppsOrigin,\n      distro: this._cloudFrontDistro,\n      apigwyOriginRequestPolicy: apigwyOriginRequestPolicy,\n      rootPathPrefix,\n      createAPIPathRoute,\n    });\n\n    //\n    // Create the edge name for the CloudFront distro\n    //\n\n    if (r53Zone !== undefined) {\n      const rrAppsEdge = new r53.RecordSet(this, 'edge-arecord', {\n        recordName: domainNameEdge,\n        recordType: r53.RecordType.A,\n        target: r53.RecordTarget.fromAlias(new r53targets.CloudFrontTarget(this._cloudFrontDistro)),\n        zone: r53Zone,\n      });\n      if (removalPolicy !== undefined) {\n        rrAppsEdge.applyRemovalPolicy(removalPolicy);\n      }\n    }\n  }\n}\n"]}
|
|
208
|
+
MicroAppsCF[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroAppsCF", version: "0.2.12" };
|
|
209
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"MicroAppsCF.js","sourceRoot":"","sources":["../src/MicroAppsCF.ts"],"names":[],"mappings":"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","sourcesContent":["import { posix as posixPath } from 'path';\nimport * as apigwy from '@aws-cdk/aws-apigatewayv2-alpha';\nimport { Aws, RemovalPolicy } from 'aws-cdk-lib';\nimport * as acm from 'aws-cdk-lib/aws-certificatemanager';\nimport * as cf from 'aws-cdk-lib/aws-cloudfront';\nimport * as cforigins from 'aws-cdk-lib/aws-cloudfront-origins';\nimport * as r53 from 'aws-cdk-lib/aws-route53';\nimport * as r53targets from 'aws-cdk-lib/aws-route53-targets';\nimport * as s3 from 'aws-cdk-lib/aws-s3';\nimport { Construct } from 'constructs';\nimport { reverseDomain } from './utils/ReverseDomain';\n\n/**\n * Represents a MicroApps CloudFront\n */\nexport interface IMicroAppsCF {\n  /**\n   * The CloudFront distribution\n   */\n  readonly cloudFrontDistro: cf.Distribution;\n}\n\n/**\n * Properties to initialize an instance of `MicroAppsCF`.\n */\nexport interface MicroAppsCFProps {\n  /**\n   * RemovalPolicy override for child resources\n   *\n   * Note: if set to DESTROY the S3 buckes will have `autoDeleteObjects` set to `true`\n   *\n   * @default - per resource default\n   */\n  readonly removalPolicy?: RemovalPolicy;\n\n  /**\n   * S3 bucket origin for deployed applications\n   */\n  readonly bucketAppsOrigin: cforigins.S3Origin;\n\n  /**\n   * S3 bucket for CloudFront logs\n   */\n  readonly bucketLogs?: s3.IBucket;\n\n  /**\n   * CloudFront Distribution domain name\n   *\n   * @example apps.pwrdrvr.com\n   * @default auto-assigned\n   */\n  readonly domainNameEdge?: string;\n\n  /**\n   * API Gateway custom origin domain name\n   *\n   * @example apps.pwrdrvr.com\n   * @default - retrieved from httpApi, if possible\n   */\n  readonly domainNameOrigin?: string;\n\n  /**\n   * API Gateway v2 HTTP API for apps\n   */\n  readonly httpApi: apigwy.HttpApi;\n\n  /**\n   * Optional asset name root\n   *\n   * @example microapps\n   * @default - resource names auto assigned\n   */\n  readonly assetNameRoot?: string;\n\n  /**\n   * Optional asset name suffix\n   *\n   * @example -dev-pr-12\n   * @default none\n   */\n  readonly assetNameSuffix?: string;\n\n  /**\n   * ACM Certificate that covers `domainNameEdge` name\n   */\n  readonly certEdge?: acm.ICertificate;\n\n  /**\n   * Route53 zone in which to create optional `domainNameEdge` record\n   */\n  readonly r53Zone?: r53.IHostedZone;\n\n  /**\n   * Path prefix on the root of the CloudFront distribution\n   *\n   * @example dev/\n   */\n  readonly rootPathPrefix?: string;\n\n  /**\n   * Create an extra Behavior (Route) for /api/ that allows\n   * API routes to have a period in them.\n   *\n   * When false API routes with a period in the path will get routed to S3.\n   *\n   * When true API routes that contain /api/ in the path will get routed to API Gateway\n   * even if they have a period in the path.\n   *\n   * @default true\n   */\n  readonly createAPIPathRoute?: boolean;\n\n  /**\n   * Create an extra Behavior (Route) for /_next/data/\n   * This route is used by Next.js to load data from the API Gateway\n   * on `getServerSideProps` calls.  The requests can end in `.json`,\n   * which would cause them to be routed to S3 if this route is not created.\n   *\n   * When false API routes with a period in the path will get routed to S3.\n   *\n   * When true API routes that contain /_next/data/ in the path will get routed to API Gateway\n   * even if they have a period in the path.\n   *\n   * @default true\n   */\n  readonly createNextDataPathRoute?: boolean;\n\n  /**\n   * Configuration of the edge to origin lambda functions\n   *\n   * @defaunt - no edge to API Gateway origin functions added\n   */\n  readonly edgeToOriginLambdas?: cf.EdgeLambda[];\n}\n\n/**\n * Options for the `CreateAPIOriginPolicy`\n */\nexport interface CreateAPIOriginPolicyOptions {\n  /**\n   * Optional asset name root\n   *\n   * @example microapps\n   * @default - resource names auto assigned\n   */\n  readonly assetNameRoot?: string;\n\n  /**\n   * Optional asset name suffix\n   *\n   * @example -dev-pr-12\n   * @default none\n   */\n  readonly assetNameSuffix?: string;\n\n  /**\n   * Edge domain name used by CloudFront - If set a custom\n   * OriginRequestPolicy will be created that prevents\n   * the Host header from being passed to the origin.\n   */\n  readonly domainNameEdge?: string;\n}\n\n/**\n * Options for `AddRoutes`\n */\nexport interface AddRoutesOptions {\n  /**\n   * API Gateway CloudFront Origin for API calls\n   */\n  readonly apiGwyOrigin: cf.IOrigin;\n\n  /**\n   * S3 Bucket CloudFront Origin for static assets\n   */\n  readonly bucketAppsOrigin: cforigins.S3Origin;\n\n  /**\n   * CloudFront Distribution to add the Behaviors (Routes) to\n   */\n  readonly distro: cf.Distribution;\n\n  /**\n   * Origin Request policy for API Gateway Origin\n   */\n  readonly apigwyOriginRequestPolicy: cf.IOriginRequestPolicy;\n\n  /**\n   * Path prefix on the root of the CloudFront distribution\n   *\n   * @example dev/\n   */\n  readonly rootPathPrefix?: string;\n\n  /**\n   * Create an extra Behavior (Route) for /api/ that allows\n   * API routes to have a period in them.\n   *\n   * When false API routes with a period in the path will get routed to S3.\n   *\n   * When true API routes that contain /api/ in the path will get routed to API Gateway\n   * even if they have a period in the path.\n   *\n   * @default true\n   */\n  readonly createAPIPathRoute?: boolean;\n\n  /**\n   * Create an extra Behavior (Route) for /_next/data/\n   * This route is used by Next.js to load data from the API Gateway\n   * on `getServerSideProps` calls.  The requests can end in `.json`,\n   * which would cause them to be routed to S3 if this route is not created.\n   *\n   * When false API routes with a period in the path will get routed to S3.\n   *\n   * When true API routes that contain /_next/data/ in the path will get routed to API Gateway\n   * even if they have a period in the path.\n   *\n   * @default true\n   */\n  readonly createNextDataPathRoute?: boolean;\n\n  /**\n   * Edge lambdas to associate with the API Gateway routes\n   */\n  readonly apigwyEdgeFunctions?: cf.EdgeLambda[];\n}\n\n/**\n * Create a new MicroApps CloudFront Distribution.\n */\nexport class MicroAppsCF extends Construct implements IMicroAppsCF {\n  /**\n   * Create or get the origin request policy\n   *\n   * If a custom domain name is NOT used for the origin then a policy\n   * will be created.\n   *\n   * If a custom domain name IS used for the origin then the ALL_VIEWER\n   * policy will be returned.  This policy passes the Host header to the\n   * origin, which is fine when using a custom domain name on the origin.\n   *\n   * @param _scope\n   * @param _props\n   */\n  public static createAPIOriginPolicy(\n    _scope: Construct,\n    _props: CreateAPIOriginPolicyOptions,\n  ): cf.IOriginRequestPolicy {\n    // const { assetNameRoot, assetNameSuffix, domainNameEdge } = props;\n\n    // let apigwyOriginRequestPolicy: cf.IOriginRequestPolicy = cf.OriginRequestPolicy.ALL_VIEWER;\n    // if (domainNameEdge === undefined) {\n    //   // When not using a custom domain name we must limit down the origin policy to\n    //   // prevent it from passing the Host header (distribution_id.cloudfront.net) to\n    //   // apigwy which will then reject it with a 403 because it does not match the\n    //   // execute-api name that apigwy is expecting.\n    //   //\n    //   // 2021-12-28 - There is a bug in the name generation that causes the same asset\n    //   // in different stacks to have the same generated name.  We have to make the id\n    //   // in all cases to ensure the generated name is unique.\n    //   apigwyOriginRequestPolicy = new cf.OriginRequestPolicy(\n    //     scope,\n    //     `apigwy-origin-policy-${Stack.of(scope).stackName}`,\n    //     {\n    //       comment: assetNameRoot ? `${assetNameRoot}-apigwy${assetNameSuffix}` : undefined,\n\n    //       originRequestPolicyName: assetNameRoot\n    //         ? `${assetNameRoot}-apigwy${assetNameSuffix}`\n    //         : undefined,\n    //       cookieBehavior: cf.OriginRequestCookieBehavior.all(),\n    //       queryStringBehavior: cf.OriginRequestQueryStringBehavior.all(),\n    //       // TODO: If signing is enabled this should forward all signature headers\n    //       // TODO: If set to \"cfront.OriginRequestHeaderBehavior.all()\" then\n    //       // `replaceHostHeader` must be set to true to prevent API Gateway from rejecting\n    //       // the request\n    //       // headerBehavior: cf.OriginRequestHeaderBehavior.allowList('user-agent', 'referer'),\n    //       headerBehavior: cf.OriginRequestHeaderBehavior.all(),\n    //     },\n    //   );\n    // }\n\n    return cf.OriginRequestPolicy.ALL_VIEWER;\n  }\n\n  /**\n   * Add API Gateway and S3 routes to an existing CloudFront Distribution\n   * @param _scope\n   * @param props\n   */\n  public static addRoutes(_scope: Construct, props: AddRoutesOptions) {\n    const {\n      apiGwyOrigin,\n      bucketAppsOrigin,\n      distro,\n      apigwyOriginRequestPolicy,\n      rootPathPrefix = '',\n      createAPIPathRoute = true,\n      createNextDataPathRoute = true,\n    } = props;\n\n    //\n    // Add Behaviors\n    //\n    const s3BehaviorOptions: cf.AddBehaviorOptions = {\n      allowedMethods: cf.AllowedMethods.ALLOW_GET_HEAD_OPTIONS,\n      cachePolicy: cf.CachePolicy.CACHING_OPTIMIZED,\n      compress: true,\n      originRequestPolicy: cf.OriginRequestPolicy.CORS_S3_ORIGIN,\n      viewerProtocolPolicy: cf.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n    };\n    const apiGwyBehaviorOptions: cf.AddBehaviorOptions = {\n      allowedMethods: cf.AllowedMethods.ALLOW_ALL,\n      // TODO: Caching needs to be set by the app response\n      cachePolicy: cf.CachePolicy.CACHING_DISABLED,\n      compress: true,\n      originRequestPolicy: apigwyOriginRequestPolicy,\n      viewerProtocolPolicy: cf.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n      edgeLambdas: props.apigwyEdgeFunctions,\n    };\n\n    //\n    // If a route specifically has `/api/` in it, send it to API Gateway\n    // This is needed to catch routes that have periods in the API path data,\n    // such as: /release/0.0.0/api/update/default/release/0.0.0\n    //\n    if (createAPIPathRoute) {\n      distro.addBehavior(\n        posixPath.join(rootPathPrefix, '/*/*/api/*'),\n        apiGwyOrigin,\n        apiGwyBehaviorOptions,\n      );\n    }\n\n    //\n    // If a route specifically has `/_next/data/` in it, send it to API Gateway\n    // This is needed to catch routes that have periods in the API path data,\n    // such as: /release/0.0.0/_next/data/app.json\n    //\n    if (createNextDataPathRoute) {\n      distro.addBehavior(\n        posixPath.join(rootPathPrefix, '/*/*/_next/data/*'),\n        apiGwyOrigin,\n        apiGwyBehaviorOptions,\n      );\n    }\n\n    //\n    // All static assets are assumed to have a dot in them\n    //\n    distro.addBehavior(\n      posixPath.join(rootPathPrefix, '/*/*/*.*'),\n      bucketAppsOrigin,\n      s3BehaviorOptions,\n    );\n\n    //\n    // Everything that isn't a static asset is going to API Gateway\n    // There is no trailing slash because Serverless Next.js wants\n    // go load pages at /release/0.0.3 (with no trailing slash).\n    //\n    distro.addBehavior(posixPath.join(rootPathPrefix, '/*'), apiGwyOrigin, apiGwyBehaviorOptions);\n  }\n\n  private _cloudFrontDistro: cf.Distribution;\n  public get cloudFrontDistro(): cf.Distribution {\n    return this._cloudFrontDistro;\n  }\n\n  constructor(scope: Construct, id: string, props: MicroAppsCFProps) {\n    super(scope, id);\n\n    if (props === undefined) {\n      throw new Error('props must be set');\n    }\n\n    if (\n      (props.r53Zone === undefined && props.domainNameEdge !== undefined) ||\n      (props.r53Zone !== undefined && props.domainNameEdge === undefined)\n    ) {\n      throw new Error('If either of r53Zone or domainNameEdge are set then the other must be set');\n    }\n\n    const {\n      domainNameEdge,\n      domainNameOrigin,\n      httpApi,\n      removalPolicy,\n      certEdge,\n      assetNameRoot,\n      assetNameSuffix,\n      r53Zone,\n      bucketLogs,\n      bucketAppsOrigin,\n      rootPathPrefix,\n      createAPIPathRoute = true,\n      createNextDataPathRoute = true,\n      edgeToOriginLambdas,\n    } = props;\n\n    const apigwyOriginRequestPolicy = MicroAppsCF.createAPIOriginPolicy(this, {\n      assetNameRoot,\n      assetNameSuffix,\n      domainNameEdge,\n    });\n\n    //\n    // Determine URL of the origin FQDN\n    //\n    let httpOriginFQDN: string = 'invalid.pwrdrvr.com';\n    if (domainNameOrigin !== undefined) {\n      httpOriginFQDN = domainNameOrigin;\n    } else {\n      httpOriginFQDN = `${httpApi.apiId}.execute-api.${Aws.REGION}.amazonaws.com`;\n    }\n\n    //\n    // Get the Edge to Origin Lambdas\n    //\n\n    //\n    // CloudFront Distro\n    //\n    const apiGwyOrigin = new cforigins.HttpOrigin(httpOriginFQDN, {\n      protocolPolicy: cf.OriginProtocolPolicy.HTTPS_ONLY,\n      originSslProtocols: [cf.OriginSslPolicy.TLS_V1_2],\n    });\n    this._cloudFrontDistro = new cf.Distribution(this, 'cft', {\n      comment: assetNameRoot ? `${assetNameRoot}${assetNameSuffix}` : domainNameEdge,\n      domainNames: domainNameEdge !== undefined ? [domainNameEdge] : undefined,\n      certificate: certEdge,\n      httpVersion: cf.HttpVersion.HTTP2,\n      defaultBehavior: {\n        allowedMethods: cf.AllowedMethods.ALLOW_ALL,\n        cachePolicy: cf.CachePolicy.CACHING_DISABLED,\n        compress: true,\n        originRequestPolicy: apigwyOriginRequestPolicy,\n        origin: apiGwyOrigin,\n        viewerProtocolPolicy: cf.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n        edgeLambdas: edgeToOriginLambdas,\n      },\n      enableIpv6: true,\n      priceClass: cf.PriceClass.PRICE_CLASS_100,\n      logBucket: bucketLogs,\n      logFilePrefix: props.domainNameEdge\n        ? `${reverseDomain(props.domainNameEdge)}/cloudfront-raw/`\n        : undefined,\n    });\n    if (removalPolicy !== undefined) {\n      this._cloudFrontDistro.applyRemovalPolicy(removalPolicy);\n    }\n\n    // Add routes to the CloudFront Distribution\n    MicroAppsCF.addRoutes(scope, {\n      apiGwyOrigin,\n      bucketAppsOrigin,\n      distro: this._cloudFrontDistro,\n      apigwyOriginRequestPolicy: apigwyOriginRequestPolicy,\n      rootPathPrefix,\n      createAPIPathRoute,\n      createNextDataPathRoute,\n      apigwyEdgeFunctions: edgeToOriginLambdas,\n    });\n\n    //\n    // Create the edge name for the CloudFront distro\n    //\n\n    if (r53Zone !== undefined) {\n      const rrAppsEdge = new r53.RecordSet(this, 'edge-arecord', {\n        recordName: domainNameEdge,\n        recordType: r53.RecordType.A,\n        target: r53.RecordTarget.fromAlias(new r53targets.CloudFrontTarget(this._cloudFrontDistro)),\n        zone: r53Zone,\n      });\n      if (removalPolicy !== undefined) {\n        rrAppsEdge.applyRemovalPolicy(removalPolicy);\n      }\n    }\n  }\n}\n"]}
|
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
import { RemovalPolicy } from 'aws-cdk-lib';
|
|
2
|
+
import * as cf from 'aws-cdk-lib/aws-cloudfront';
|
|
3
|
+
import * as lambda from 'aws-cdk-lib/aws-lambda';
|
|
4
|
+
import { Construct } from 'constructs';
|
|
5
|
+
/**
|
|
6
|
+
* Represents a MicroApps Edge to Origin Function
|
|
7
|
+
*/
|
|
8
|
+
export interface IMicroAppsEdgeToOrigin {
|
|
9
|
+
/**
|
|
10
|
+
* The edge to origin function for API Gateway Request Origin Edge Lambda
|
|
11
|
+
*
|
|
12
|
+
* The generated `config.yml` is included in the Lambda's code.
|
|
13
|
+
*/
|
|
14
|
+
readonly edgeToOriginFunction: lambda.Function | cf.experimental.EdgeFunction;
|
|
15
|
+
/**
|
|
16
|
+
* Configuration of the edge to origin lambda functions
|
|
17
|
+
*/
|
|
18
|
+
readonly edgeToOriginLambdas: cf.EdgeLambda[];
|
|
19
|
+
}
|
|
20
|
+
/**
|
|
21
|
+
* Properties to initialize an instance of `MicroAppsEdgeToOrigin`.
|
|
22
|
+
*/
|
|
23
|
+
export interface MicroAppsEdgeToOriginProps {
|
|
24
|
+
/**
|
|
25
|
+
* RemovalPolicy override for child resources
|
|
26
|
+
*
|
|
27
|
+
* @default - per resource default
|
|
28
|
+
*/
|
|
29
|
+
readonly removalPolicy?: RemovalPolicy;
|
|
30
|
+
/**
|
|
31
|
+
* Optional asset name root
|
|
32
|
+
*
|
|
33
|
+
* @example microapps
|
|
34
|
+
* @default - resource names auto assigned
|
|
35
|
+
*/
|
|
36
|
+
readonly assetNameRoot?: string;
|
|
37
|
+
/**
|
|
38
|
+
* Optional asset name suffix
|
|
39
|
+
*
|
|
40
|
+
* @example -dev-pr-12
|
|
41
|
+
* @default none
|
|
42
|
+
*/
|
|
43
|
+
readonly assetNameSuffix?: string;
|
|
44
|
+
/**
|
|
45
|
+
* Adds an X-Forwarded-Host-Header when calling API Gateway
|
|
46
|
+
*
|
|
47
|
+
* Can only be trusted if `signingMode` is enabled, which restricts
|
|
48
|
+
* access to API Gateway to only IAM signed requests.
|
|
49
|
+
*
|
|
50
|
+
* Note: if true, creates OriginRequest Lambda @ Edge function for API Gateway Origin
|
|
51
|
+
* @default true
|
|
52
|
+
*/
|
|
53
|
+
readonly addXForwardedHostHeader?: boolean;
|
|
54
|
+
/**
|
|
55
|
+
* Replaces Host header (which will be the Edge domain name) with the Origin domain name
|
|
56
|
+
* when enabled. This is necessary when API Gateway has not been configured
|
|
57
|
+
* with a custom domain name that matches the exact domain name used by the CloudFront
|
|
58
|
+
* Distribution AND when the OriginRequestPolicy.HeadersBehavior is set
|
|
59
|
+
* to pass all headers to the origin.
|
|
60
|
+
*
|
|
61
|
+
* Note: if true, creates OriginRequest Lambda @ Edge function for API Gateway Origin
|
|
62
|
+
* @default true
|
|
63
|
+
*/
|
|
64
|
+
readonly replaceHostHeader?: boolean;
|
|
65
|
+
/**
|
|
66
|
+
* Requires IAM auth on the API Gateway origin if not set to 'none'.
|
|
67
|
+
*
|
|
68
|
+
* 'sign' - Uses request headers for auth.
|
|
69
|
+
* 'presign' - Uses query string for auth.
|
|
70
|
+
*
|
|
71
|
+
* If enabled,
|
|
72
|
+
*
|
|
73
|
+
* Note: if 'sign' or 'presign', creates OriginRequest Lambda @ Edge function for API Gateway Origin
|
|
74
|
+
* @default 'sign'
|
|
75
|
+
*/
|
|
76
|
+
readonly signingMode?: 'sign' | 'presign' | 'none';
|
|
77
|
+
/**
|
|
78
|
+
* Origin region that API Gateway will be deployed to, used
|
|
79
|
+
* for the config.yml on the Edge function to sign requests for
|
|
80
|
+
* the correct region
|
|
81
|
+
*
|
|
82
|
+
* @default undefined
|
|
83
|
+
*/
|
|
84
|
+
readonly originRegion?: string;
|
|
85
|
+
}
|
|
86
|
+
export interface GenerateEdgeToOriginConfigOptions {
|
|
87
|
+
readonly originRegion: string;
|
|
88
|
+
readonly signingMode: 'sign' | 'presign' | '';
|
|
89
|
+
readonly addXForwardedHostHeader: boolean;
|
|
90
|
+
readonly replaceHostHeader: boolean;
|
|
91
|
+
}
|
|
92
|
+
/**
|
|
93
|
+
* Create a new MicroApps Edge to Origin Function w/ `config.yml`
|
|
94
|
+
*/
|
|
95
|
+
export declare class MicroAppsEdgeToOrigin extends Construct implements IMicroAppsEdgeToOrigin {
|
|
96
|
+
/**
|
|
97
|
+
* Generate the yaml config for the edge lambda
|
|
98
|
+
* @param props
|
|
99
|
+
* @returns
|
|
100
|
+
*/
|
|
101
|
+
static generateEdgeToOriginConfig(props: GenerateEdgeToOriginConfigOptions): string;
|
|
102
|
+
private _edgeToOriginFunction;
|
|
103
|
+
get edgeToOriginFunction(): lambda.Function | cf.experimental.EdgeFunction;
|
|
104
|
+
private _edgeToOriginLambdas;
|
|
105
|
+
get edgeToOriginLambdas(): cf.EdgeLambda[];
|
|
106
|
+
constructor(scope: Construct, id: string, props: MicroAppsEdgeToOriginProps);
|
|
107
|
+
}
|
|
108
|
+
//# sourceMappingURL=MicroAppsEdgeToOrigin.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"MicroAppsEdgeToOrigin.d.ts","sourceRoot":"","sources":["../src/MicroAppsEdgeToOrigin.ts"],"names":[],"mappings":"AAGA,OAAO,EAAiB,aAAa,EAAE,MAAM,aAAa,CAAC;AAC3D,OAAO,KAAK,EAAE,MAAM,4BAA4B,CAAC;AAEjD,OAAO,KAAK,MAAM,MAAM,wBAAwB,CAAC;AAGjD,OAAO,EAAE,SAAS,EAAE,MAAM,YAAY,CAAC;AAEvC;;GAEG;AACH,MAAM,WAAW,sBAAsB;IACrC;;;;OAIG;IACH,QAAQ,CAAC,oBAAoB,EAAE,MAAM,CAAC,QAAQ,GAAG,EAAE,CAAC,YAAY,CAAC,YAAY,CAAC;IAE9E;;OAEG;IACH,QAAQ,CAAC,mBAAmB,EAAE,EAAE,CAAC,UAAU,EAAE,CAAC;CAC/C;AAED;;GAEG;AACH,MAAM,WAAW,0BAA0B;IACzC;;;;OAIG;IACH,QAAQ,CAAC,aAAa,CAAC,EAAE,aAAa,CAAC;IAEvC;;;;;OAKG;IACH,QAAQ,CAAC,aAAa,CAAC,EAAE,MAAM,CAAC;IAEhC;;;;;OAKG;IACH,QAAQ,CAAC,eAAe,CAAC,EAAE,MAAM,CAAC;IAElC;;;;;;;;OAQG;IACH,QAAQ,CAAC,uBAAuB,CAAC,EAAE,OAAO,CAAC;IAE3C;;;;;;;;;OASG;IACH,QAAQ,CAAC,iBAAiB,CAAC,EAAE,OAAO,CAAC;IAErC;;;;;;;;;;OAUG;IACH,QAAQ,CAAC,WAAW,CAAC,EAAE,MAAM,GAAG,SAAS,GAAG,MAAM,CAAC;IAEnD;;;;;;OAMG;IACH,QAAQ,CAAC,YAAY,CAAC,EAAE,MAAM,CAAC;CAChC;AAED,MAAM,WAAW,iCAAiC;IAChD,QAAQ,CAAC,YAAY,EAAE,MAAM,CAAC;IAC9B,QAAQ,CAAC,WAAW,EAAE,MAAM,GAAG,SAAS,GAAG,EAAE,CAAC;IAC9C,QAAQ,CAAC,uBAAuB,EAAE,OAAO,CAAC;IAC1C,QAAQ,CAAC,iBAAiB,EAAE,OAAO,CAAC;CACrC;AAED;;GAEG;AACH,qBAAa,qBAAsB,SAAQ,SAAU,YAAW,sBAAsB;IACpF;;;;OAIG;WACW,0BAA0B,CAAC,KAAK,EAAE,iCAAiC;IAOjF,OAAO,CAAC,qBAAqB,CAAiD;IAC9E,IAAW,oBAAoB,IAAI,MAAM,CAAC,QAAQ,GAAG,EAAE,CAAC,YAAY,CAAC,YAAY,CAEhF;IAED,OAAO,CAAC,oBAAoB,CAAkB;IAC9C,IAAW,mBAAmB,IAAI,EAAE,CAAC,UAAU,EAAE,CAEhD;gBAEW,KAAK,EAAE,SAAS,EAAE,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE,0BAA0B;CA4I5E"}
|
|
@@ -0,0 +1,139 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var _a;
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.MicroAppsEdgeToOrigin = void 0;
|
|
5
|
+
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
6
|
+
const fs_1 = require("fs");
|
|
7
|
+
const os = require("os");
|
|
8
|
+
const path = require("path");
|
|
9
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
10
|
+
const cf = require("aws-cdk-lib/aws-cloudfront");
|
|
11
|
+
const iam = require("aws-cdk-lib/aws-iam");
|
|
12
|
+
const lambda = require("aws-cdk-lib/aws-lambda");
|
|
13
|
+
const lambdaNodejs = require("aws-cdk-lib/aws-lambda-nodejs");
|
|
14
|
+
const logs = require("aws-cdk-lib/aws-logs");
|
|
15
|
+
const constructs_1 = require("constructs");
|
|
16
|
+
/**
|
|
17
|
+
* Create a new MicroApps Edge to Origin Function w/ `config.yml`
|
|
18
|
+
*/
|
|
19
|
+
class MicroAppsEdgeToOrigin extends constructs_1.Construct {
|
|
20
|
+
constructor(scope, id, props) {
|
|
21
|
+
super(scope, id);
|
|
22
|
+
if (props === undefined) {
|
|
23
|
+
throw new Error('props must be set');
|
|
24
|
+
}
|
|
25
|
+
const { removalPolicy, assetNameRoot, assetNameSuffix, signingMode = 'sign', addXForwardedHostHeader = true, replaceHostHeader = true, originRegion, } = props;
|
|
26
|
+
// Create the edge function config file from the construct options
|
|
27
|
+
const edgeToOriginConfigYaml = MicroAppsEdgeToOrigin.generateEdgeToOriginConfig({
|
|
28
|
+
originRegion: originRegion || aws_cdk_lib_1.Aws.REGION,
|
|
29
|
+
addXForwardedHostHeader,
|
|
30
|
+
replaceHostHeader,
|
|
31
|
+
signingMode: signingMode === 'none' ? '' : signingMode,
|
|
32
|
+
});
|
|
33
|
+
//
|
|
34
|
+
// Create the Edge to Origin Function
|
|
35
|
+
//
|
|
36
|
+
const edgeToOriginFuncProps = {
|
|
37
|
+
functionName: assetNameRoot ? `${assetNameRoot}-edge-to-origin${assetNameSuffix}` : undefined,
|
|
38
|
+
memorySize: 1769,
|
|
39
|
+
logRetention: logs.RetentionDays.ONE_MONTH,
|
|
40
|
+
runtime: lambda.Runtime.NODEJS_14_X,
|
|
41
|
+
timeout: aws_cdk_lib_1.Duration.seconds(5),
|
|
42
|
+
initialPolicy: [
|
|
43
|
+
// This can't have a reference to the httpApi because it would mean
|
|
44
|
+
// the parent stack (this stack) has to be created before the us-east-1
|
|
45
|
+
// child stack for the Edge Lambda Function.
|
|
46
|
+
// That's why we use a tag-based policy to allow the Edge Function
|
|
47
|
+
// to invoke any API Gateway API that we apply a tag to
|
|
48
|
+
// We allow the edge function to sign for all regions since
|
|
49
|
+
// we may use custom closest region in the future.
|
|
50
|
+
new iam.PolicyStatement({
|
|
51
|
+
actions: ['execute-api:Invoke'],
|
|
52
|
+
resources: [`arn:aws:execute-api:*:${aws_cdk_lib_1.Aws.ACCOUNT_ID}:*/*/*/*`],
|
|
53
|
+
}),
|
|
54
|
+
],
|
|
55
|
+
...(removalPolicy ? { removalPolicy } : {}),
|
|
56
|
+
};
|
|
57
|
+
if (process.env.NODE_ENV === 'test' &&
|
|
58
|
+
fs_1.existsSync(path.join(__dirname, '..', '..', 'microapps-edge-to-origin', 'dist', 'index.js'))) {
|
|
59
|
+
// Emit the config file from the construct options
|
|
60
|
+
fs_1.writeFileSync(path.join(__dirname, '..', '..', 'microapps-edge-to-origin', 'dist', 'config.yml'), edgeToOriginConfigYaml);
|
|
61
|
+
// copyFileSync(
|
|
62
|
+
// path.join(__dirname, '..', '..', '..', 'configs', 'microapps-edge-to-origin', 'config.yml'),
|
|
63
|
+
// path.join(__dirname, '..', '..', 'microapps-edge-to-origin', 'dist', 'config.yml'),
|
|
64
|
+
// );
|
|
65
|
+
// This is for tests run under jest
|
|
66
|
+
// This is also for anytime when the edge function has already been bundled
|
|
67
|
+
this._edgeToOriginFunction = new cf.experimental.EdgeFunction(this, 'edge-to-apigwy-func', {
|
|
68
|
+
code: lambda.Code.fromAsset(path.join(__dirname, '..', '..', 'microapps-edge-to-origin', 'dist')),
|
|
69
|
+
handler: 'index.handler',
|
|
70
|
+
...edgeToOriginFuncProps,
|
|
71
|
+
});
|
|
72
|
+
}
|
|
73
|
+
else if (fs_1.existsSync(path.join(__dirname, 'microapps-edge-to-origin', 'index.js'))) {
|
|
74
|
+
// Emit the config file from the construct options
|
|
75
|
+
fs_1.writeFileSync(path.join(__dirname, 'microapps-edge-to-origin', 'config.yml'), edgeToOriginConfigYaml);
|
|
76
|
+
// This is for built apps packaged with the CDK construct
|
|
77
|
+
this._edgeToOriginFunction = new cf.experimental.EdgeFunction(this, 'edge-to-apigwy-func', {
|
|
78
|
+
code: lambda.Code.fromAsset(path.join(__dirname, 'microapps-edge-to-origin')),
|
|
79
|
+
handler: 'index.handler',
|
|
80
|
+
...edgeToOriginFuncProps,
|
|
81
|
+
});
|
|
82
|
+
}
|
|
83
|
+
else {
|
|
84
|
+
// Emit the config file from the construct options
|
|
85
|
+
fs_1.writeFileSync(path.join(__dirname, '..', '..', 'microapps-edge-to-origin', 'config.yml'), edgeToOriginConfigYaml);
|
|
86
|
+
// This builds the function for distribution with the CDK Construct
|
|
87
|
+
// and will be used during local builds and PR builds of microapps-core
|
|
88
|
+
// if the microapps-edge-to-origin function is not already bundled.
|
|
89
|
+
// This will fail to deploy in any region other than us-east-1
|
|
90
|
+
// We cannot use NodejsFunction because it will not create in us-east-1
|
|
91
|
+
this._edgeToOriginFunction = new lambdaNodejs.NodejsFunction(this, 'edge-to-apigwy-func', {
|
|
92
|
+
entry: path.join(__dirname, '..', '..', 'microapps-edge-to-origin', 'src', 'index.ts'),
|
|
93
|
+
handler: 'handler',
|
|
94
|
+
bundling: {
|
|
95
|
+
minify: true,
|
|
96
|
+
sourceMap: true,
|
|
97
|
+
commandHooks: {
|
|
98
|
+
beforeInstall: () => [],
|
|
99
|
+
beforeBundling: () => [],
|
|
100
|
+
afterBundling: (_inputDir, outputDir) => {
|
|
101
|
+
return [
|
|
102
|
+
`${os.platform() === 'win32' ? 'copy' : 'cp'} ${path.join(__dirname, '..', '..', '..', 'configs', 'microapps-edge-to-origin', 'config.yml')} ${outputDir}`,
|
|
103
|
+
];
|
|
104
|
+
},
|
|
105
|
+
},
|
|
106
|
+
},
|
|
107
|
+
...edgeToOriginFuncProps,
|
|
108
|
+
});
|
|
109
|
+
}
|
|
110
|
+
this._edgeToOriginLambdas = [
|
|
111
|
+
{
|
|
112
|
+
eventType: cf.LambdaEdgeEventType.ORIGIN_REQUEST,
|
|
113
|
+
functionVersion: this._edgeToOriginFunction.currentVersion,
|
|
114
|
+
includeBody: true,
|
|
115
|
+
},
|
|
116
|
+
];
|
|
117
|
+
}
|
|
118
|
+
/**
|
|
119
|
+
* Generate the yaml config for the edge lambda
|
|
120
|
+
* @param props
|
|
121
|
+
* @returns
|
|
122
|
+
*/
|
|
123
|
+
static generateEdgeToOriginConfig(props) {
|
|
124
|
+
return `originRegion: ${props.originRegion}
|
|
125
|
+
${props.signingMode === '' ? '' : `signingMode: ${props.signingMode}`}
|
|
126
|
+
addXForwardedHostHeader: ${props.addXForwardedHostHeader}
|
|
127
|
+
replaceHostHeader: ${props.replaceHostHeader}`;
|
|
128
|
+
}
|
|
129
|
+
get edgeToOriginFunction() {
|
|
130
|
+
return this._edgeToOriginFunction;
|
|
131
|
+
}
|
|
132
|
+
get edgeToOriginLambdas() {
|
|
133
|
+
return this._edgeToOriginLambdas;
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
exports.MicroAppsEdgeToOrigin = MicroAppsEdgeToOrigin;
|
|
137
|
+
_a = JSII_RTTI_SYMBOL_1;
|
|
138
|
+
MicroAppsEdgeToOrigin[_a] = { fqn: "@pwrdrvr/microapps-cdk.MicroAppsEdgeToOrigin", version: "0.2.12" };
|
|
139
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"MicroAppsEdgeToOrigin.js","sourceRoot":"","sources":["../src/MicroAppsEdgeToOrigin.ts"],"names":[],"mappings":"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","sourcesContent":["import { existsSync, writeFileSync } from 'fs';\nimport * as os from 'os';\nimport * as path from 'path';\nimport { Aws, Duration, RemovalPolicy } from 'aws-cdk-lib';\nimport * as cf from 'aws-cdk-lib/aws-cloudfront';\nimport * as iam from 'aws-cdk-lib/aws-iam';\nimport * as lambda from 'aws-cdk-lib/aws-lambda';\nimport * as lambdaNodejs from 'aws-cdk-lib/aws-lambda-nodejs';\nimport * as logs from 'aws-cdk-lib/aws-logs';\nimport { Construct } from 'constructs';\n\n/**\n * Represents a MicroApps Edge to Origin Function\n */\nexport interface IMicroAppsEdgeToOrigin {\n  /**\n   * The edge to origin function for API Gateway Request Origin Edge Lambda\n   *\n   * The generated `config.yml` is included in the Lambda's code.\n   */\n  readonly edgeToOriginFunction: lambda.Function | cf.experimental.EdgeFunction;\n\n  /**\n   * Configuration of the edge to origin lambda functions\n   */\n  readonly edgeToOriginLambdas: cf.EdgeLambda[];\n}\n\n/**\n * Properties to initialize an instance of `MicroAppsEdgeToOrigin`.\n */\nexport interface MicroAppsEdgeToOriginProps {\n  /**\n   * RemovalPolicy override for child resources\n   *\n   * @default - per resource default\n   */\n  readonly removalPolicy?: RemovalPolicy;\n\n  /**\n   * Optional asset name root\n   *\n   * @example microapps\n   * @default - resource names auto assigned\n   */\n  readonly assetNameRoot?: string;\n\n  /**\n   * Optional asset name suffix\n   *\n   * @example -dev-pr-12\n   * @default none\n   */\n  readonly assetNameSuffix?: string;\n\n  /**\n   * Adds an X-Forwarded-Host-Header when calling API Gateway\n   *\n   * Can only be trusted if `signingMode` is enabled, which restricts\n   * access to API Gateway to only IAM signed requests.\n   *\n   * Note: if true, creates OriginRequest Lambda @ Edge function for API Gateway Origin\n   * @default true\n   */\n  readonly addXForwardedHostHeader?: boolean;\n\n  /**\n   * Replaces Host header (which will be the Edge domain name) with the Origin domain name\n   * when enabled.  This is necessary when API Gateway has not been configured\n   * with a custom domain name that matches the exact domain name used by the CloudFront\n   * Distribution AND when the OriginRequestPolicy.HeadersBehavior is set\n   * to pass all headers to the origin.\n   *\n   * Note: if true, creates OriginRequest Lambda @ Edge function for API Gateway Origin\n   * @default true\n   */\n  readonly replaceHostHeader?: boolean;\n\n  /**\n   * Requires IAM auth on the API Gateway origin if not set to 'none'.\n   *\n   * 'sign' - Uses request headers for auth.\n   * 'presign' - Uses query string for auth.\n   *\n   * If enabled,\n   *\n   * Note: if 'sign' or 'presign', creates OriginRequest Lambda @ Edge function for API Gateway Origin\n   * @default 'sign'\n   */\n  readonly signingMode?: 'sign' | 'presign' | 'none';\n\n  /**\n   * Origin region that API Gateway will be deployed to, used\n   * for the config.yml on the Edge function to sign requests for\n   * the correct region\n   *\n   * @default undefined\n   */\n  readonly originRegion?: string;\n}\n\nexport interface GenerateEdgeToOriginConfigOptions {\n  readonly originRegion: string;\n  readonly signingMode: 'sign' | 'presign' | '';\n  readonly addXForwardedHostHeader: boolean;\n  readonly replaceHostHeader: boolean;\n}\n\n/**\n * Create a new MicroApps Edge to Origin Function w/ `config.yml`\n */\nexport class MicroAppsEdgeToOrigin extends Construct implements IMicroAppsEdgeToOrigin {\n  /**\n   * Generate the yaml config for the edge lambda\n   * @param props\n   * @returns\n   */\n  public static generateEdgeToOriginConfig(props: GenerateEdgeToOriginConfigOptions) {\n    return `originRegion: ${props.originRegion}\n${props.signingMode === '' ? '' : `signingMode: ${props.signingMode}`}\naddXForwardedHostHeader: ${props.addXForwardedHostHeader}\nreplaceHostHeader: ${props.replaceHostHeader}`;\n  }\n\n  private _edgeToOriginFunction: lambda.Function | cf.experimental.EdgeFunction;\n  public get edgeToOriginFunction(): lambda.Function | cf.experimental.EdgeFunction {\n    return this._edgeToOriginFunction;\n  }\n\n  private _edgeToOriginLambdas: cf.EdgeLambda[];\n  public get edgeToOriginLambdas(): cf.EdgeLambda[] {\n    return this._edgeToOriginLambdas;\n  }\n\n  constructor(scope: Construct, id: string, props: MicroAppsEdgeToOriginProps) {\n    super(scope, id);\n\n    if (props === undefined) {\n      throw new Error('props must be set');\n    }\n\n    const {\n      removalPolicy,\n      assetNameRoot,\n      assetNameSuffix,\n      signingMode = 'sign',\n      addXForwardedHostHeader = true,\n      replaceHostHeader = true,\n      originRegion,\n    } = props;\n\n    // Create the edge function config file from the construct options\n    const edgeToOriginConfigYaml = MicroAppsEdgeToOrigin.generateEdgeToOriginConfig({\n      originRegion: originRegion || Aws.REGION,\n      addXForwardedHostHeader,\n      replaceHostHeader,\n      signingMode: signingMode === 'none' ? '' : signingMode,\n    });\n\n    //\n    // Create the Edge to Origin Function\n    //\n    const edgeToOriginFuncProps: Omit<lambda.FunctionProps, 'handler' | 'code'> = {\n      functionName: assetNameRoot ? `${assetNameRoot}-edge-to-origin${assetNameSuffix}` : undefined,\n      memorySize: 1769,\n      logRetention: logs.RetentionDays.ONE_MONTH,\n      runtime: lambda.Runtime.NODEJS_14_X,\n      timeout: Duration.seconds(5),\n      initialPolicy: [\n        // This can't have a reference to the httpApi because it would mean\n        // the parent stack (this stack) has to be created before the us-east-1\n        // child stack for the Edge Lambda Function.\n        // That's why we use a tag-based policy to allow the Edge Function\n        // to invoke any API Gateway API that we apply a tag to\n        // We allow the edge function to sign for all regions since\n        // we may use custom closest region in the future.\n        new iam.PolicyStatement({\n          actions: ['execute-api:Invoke'],\n          resources: [`arn:aws:execute-api:*:${Aws.ACCOUNT_ID}:*/*/*/*`],\n          // Unfortunately, API Gateway access cannot be restricted using\n          // tags on the target resource\n          // https://docs.aws.amazon.com/IAM/latest/UserGuide/access_tags.html\n          // https://docs.aws.amazon.com/IAM/latest/UserGuide/reference_aws-services-that-work-with-iam.html#networking_svcs\n          // conditions: {\n          //   // TODO: Set this to a string unique to each stack\n          //   StringEquals: { 'aws:ResourceTag/microapp-managed': 'true' },\n          // },\n        }),\n      ],\n      ...(removalPolicy ? { removalPolicy } : {}),\n    };\n    if (\n      process.env.NODE_ENV === 'test' &&\n      existsSync(path.join(__dirname, '..', '..', 'microapps-edge-to-origin', 'dist', 'index.js'))\n    ) {\n      // Emit the config file from the construct options\n      writeFileSync(\n        path.join(__dirname, '..', '..', 'microapps-edge-to-origin', 'dist', 'config.yml'),\n        edgeToOriginConfigYaml,\n      );\n      // copyFileSync(\n      //   path.join(__dirname, '..', '..', '..', 'configs', 'microapps-edge-to-origin', 'config.yml'),\n      //   path.join(__dirname, '..', '..', 'microapps-edge-to-origin', 'dist', 'config.yml'),\n      // );\n      // This is for tests run under jest\n      // This is also for anytime when the edge function has already been bundled\n      this._edgeToOriginFunction = new cf.experimental.EdgeFunction(this, 'edge-to-apigwy-func', {\n        code: lambda.Code.fromAsset(\n          path.join(__dirname, '..', '..', 'microapps-edge-to-origin', 'dist'),\n        ),\n        handler: 'index.handler',\n        ...edgeToOriginFuncProps,\n      });\n    } else if (existsSync(path.join(__dirname, 'microapps-edge-to-origin', 'index.js'))) {\n      // Emit the config file from the construct options\n      writeFileSync(\n        path.join(__dirname, 'microapps-edge-to-origin', 'config.yml'),\n        edgeToOriginConfigYaml,\n      );\n\n      // This is for built apps packaged with the CDK construct\n      this._edgeToOriginFunction = new cf.experimental.EdgeFunction(this, 'edge-to-apigwy-func', {\n        code: lambda.Code.fromAsset(path.join(__dirname, 'microapps-edge-to-origin')),\n        handler: 'index.handler',\n        ...edgeToOriginFuncProps,\n      });\n    } else {\n      // Emit the config file from the construct options\n      writeFileSync(\n        path.join(__dirname, '..', '..', 'microapps-edge-to-origin', 'config.yml'),\n        edgeToOriginConfigYaml,\n      );\n\n      // This builds the function for distribution with the CDK Construct\n      // and will be used during local builds and PR builds of microapps-core\n      // if the microapps-edge-to-origin function is not already bundled.\n      // This will fail to deploy in any region other than us-east-1\n      // We cannot use NodejsFunction because it will not create in us-east-1\n      this._edgeToOriginFunction = new lambdaNodejs.NodejsFunction(this, 'edge-to-apigwy-func', {\n        entry: path.join(__dirname, '..', '..', 'microapps-edge-to-origin', 'src', 'index.ts'),\n        handler: 'handler',\n        bundling: {\n          minify: true,\n          sourceMap: true,\n          commandHooks: {\n            beforeInstall: () => [],\n            beforeBundling: () => [],\n            afterBundling: (_inputDir: string, outputDir: string) => {\n              return [\n                `${os.platform() === 'win32' ? 'copy' : 'cp'} ${path.join(\n                  __dirname,\n                  '..',\n                  '..',\n                  '..',\n                  'configs',\n                  'microapps-edge-to-origin',\n                  'config.yml',\n                )} ${outputDir}`,\n              ];\n            },\n          },\n        },\n        ...edgeToOriginFuncProps,\n      });\n    }\n\n    this._edgeToOriginLambdas = [\n      {\n        eventType: cf.LambdaEdgeEventType.ORIGIN_REQUEST,\n        functionVersion: this._edgeToOriginFunction.currentVersion,\n        includeBody: true,\n      },\n    ];\n  }\n}\n"]}
|