@capraconsulting/webapp-deploy-lambda 2.3.0 → 2.3.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/lib/index.d.ts +4 -0
- package/lib/index.js +3 -0
- package/lib/source.d.ts +58 -0
- package/lib/source.js +64 -0
- package/lib/webapp-deploy.d.ts +64 -0
- package/lib/webapp-deploy.js +64 -0
- package/package.json +10 -10
package/lib/index.d.ts
ADDED
package/lib/index.js
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
1
|
+
export { Source } from "./source";
|
|
2
|
+
export { WebappDeploy } from "./webapp-deploy";
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLFVBQVUsQ0FBQTtBQUVqQyxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgeyBTb3VyY2UgfSBmcm9tIFwiLi9zb3VyY2VcIlxuZXhwb3J0IHR5cGUgeyBJU291cmNlLCBTb3VyY2VDb25maWcgfSBmcm9tIFwiLi9zb3VyY2VcIlxuZXhwb3J0IHsgV2ViYXBwRGVwbG95IH0gZnJvbSBcIi4vd2ViYXBwLWRlcGxveVwiXG5leHBvcnQgdHlwZSB7IFdlYmFwcERlcGxveVByb3BzIH0gZnJvbSBcIi4vd2ViYXBwLWRlcGxveVwiXG4iXX0=
|
package/lib/source.d.ts
ADDED
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
import * as constructs from "constructs";
|
|
2
|
+
import * as iam from "aws-cdk-lib/aws-iam";
|
|
3
|
+
import * as s3 from "aws-cdk-lib/aws-s3";
|
|
4
|
+
import * as s3Assets from "aws-cdk-lib/aws-s3-assets";
|
|
5
|
+
export interface SourceConfig {
|
|
6
|
+
/**
|
|
7
|
+
* The source bucket to deploy from.
|
|
8
|
+
*/
|
|
9
|
+
readonly bucket: s3.IBucket;
|
|
10
|
+
/**
|
|
11
|
+
* An S3 object key in the source bucket that points to a zip file.
|
|
12
|
+
*/
|
|
13
|
+
readonly zipObjectKey: string;
|
|
14
|
+
}
|
|
15
|
+
/**
|
|
16
|
+
* Bind context for ISources
|
|
17
|
+
*/
|
|
18
|
+
export interface SourceContext {
|
|
19
|
+
/**
|
|
20
|
+
* The role for the handler
|
|
21
|
+
*
|
|
22
|
+
* @default - no policy is modified
|
|
23
|
+
*/
|
|
24
|
+
readonly handlerRole?: iam.IRole;
|
|
25
|
+
}
|
|
26
|
+
/**
|
|
27
|
+
* Represents a source for bucket deployments.
|
|
28
|
+
*/
|
|
29
|
+
export interface ISource {
|
|
30
|
+
/**
|
|
31
|
+
* Binds the source to a bucket deployment.
|
|
32
|
+
*/
|
|
33
|
+
bind(scope: constructs.Construct, context?: SourceContext): SourceConfig;
|
|
34
|
+
}
|
|
35
|
+
/**
|
|
36
|
+
* Specifies bucket deployment source.
|
|
37
|
+
*
|
|
38
|
+
* Usage:
|
|
39
|
+
*
|
|
40
|
+
* Source.bucket(bucket, key)
|
|
41
|
+
* Source.asset('/local/path/to/directory')
|
|
42
|
+
* Source.asset('/local/path/to/a/file.zip')
|
|
43
|
+
*
|
|
44
|
+
*/
|
|
45
|
+
export declare class Source {
|
|
46
|
+
/**
|
|
47
|
+
* Uses a .zip file stored in an S3 bucket as the source for the destination bucket contents.
|
|
48
|
+
* @param bucket The S3 Bucket
|
|
49
|
+
* @param zipObjectKey The S3 object key of the zip file with contents
|
|
50
|
+
*/
|
|
51
|
+
static bucket(bucket: s3.IBucket, zipObjectKey: string): ISource;
|
|
52
|
+
/**
|
|
53
|
+
* Uses a local asset as the deployment source.
|
|
54
|
+
* @param path The path to a local .zip file or a directory
|
|
55
|
+
*/
|
|
56
|
+
static asset(path: string, options?: s3Assets.AssetOptions): ISource;
|
|
57
|
+
private constructor();
|
|
58
|
+
}
|
package/lib/source.js
ADDED
|
@@ -0,0 +1,64 @@
|
|
|
1
|
+
import * as s3Assets from "aws-cdk-lib/aws-s3-assets";
|
|
2
|
+
/**
|
|
3
|
+
* Specifies bucket deployment source.
|
|
4
|
+
*
|
|
5
|
+
* Usage:
|
|
6
|
+
*
|
|
7
|
+
* Source.bucket(bucket, key)
|
|
8
|
+
* Source.asset('/local/path/to/directory')
|
|
9
|
+
* Source.asset('/local/path/to/a/file.zip')
|
|
10
|
+
*
|
|
11
|
+
*/
|
|
12
|
+
export class Source {
|
|
13
|
+
/**
|
|
14
|
+
* Uses a .zip file stored in an S3 bucket as the source for the destination bucket contents.
|
|
15
|
+
* @param bucket The S3 Bucket
|
|
16
|
+
* @param zipObjectKey The S3 object key of the zip file with contents
|
|
17
|
+
*/
|
|
18
|
+
static bucket(bucket, zipObjectKey) {
|
|
19
|
+
return {
|
|
20
|
+
bind: (_, context) => {
|
|
21
|
+
if (!context) {
|
|
22
|
+
throw new Error("To use a Source.bucket(), context must be provided");
|
|
23
|
+
}
|
|
24
|
+
if (context.handlerRole) {
|
|
25
|
+
bucket.grantRead(context.handlerRole);
|
|
26
|
+
}
|
|
27
|
+
return { bucket, zipObjectKey };
|
|
28
|
+
},
|
|
29
|
+
};
|
|
30
|
+
}
|
|
31
|
+
/**
|
|
32
|
+
* Uses a local asset as the deployment source.
|
|
33
|
+
* @param path The path to a local .zip file or a directory
|
|
34
|
+
*/
|
|
35
|
+
static asset(path, options) {
|
|
36
|
+
return {
|
|
37
|
+
bind(scope, context) {
|
|
38
|
+
if (!context) {
|
|
39
|
+
throw new Error("To use a Source.asset(), context must be provided");
|
|
40
|
+
}
|
|
41
|
+
let id = 1;
|
|
42
|
+
while (scope.node.tryFindChild(`Asset${id}`)) {
|
|
43
|
+
id++;
|
|
44
|
+
}
|
|
45
|
+
const asset = new s3Assets.Asset(scope, `Asset${id}`, {
|
|
46
|
+
path,
|
|
47
|
+
...options,
|
|
48
|
+
});
|
|
49
|
+
if (!asset.isZipArchive) {
|
|
50
|
+
throw new Error("Asset path must be either a .zip file or a directory");
|
|
51
|
+
}
|
|
52
|
+
if (context.handlerRole) {
|
|
53
|
+
asset.grantRead(context.handlerRole);
|
|
54
|
+
}
|
|
55
|
+
return {
|
|
56
|
+
bucket: asset.bucket,
|
|
57
|
+
zipObjectKey: asset.s3ObjectKey,
|
|
58
|
+
};
|
|
59
|
+
},
|
|
60
|
+
};
|
|
61
|
+
}
|
|
62
|
+
constructor() { }
|
|
63
|
+
}
|
|
64
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic291cmNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vc3JjL3NvdXJjZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFHQSxPQUFPLEtBQUssUUFBUSxNQUFNLDJCQUEyQixDQUFBO0FBc0NyRDs7Ozs7Ozs7O0dBU0c7QUFDSCxNQUFNLE9BQU8sTUFBTTtJQUNqQjs7OztPQUlHO0lBQ0ksTUFBTSxDQUFDLE1BQU0sQ0FBQyxNQUFrQixFQUFFLFlBQW9CO1FBQzNELE9BQU87WUFDTCxJQUFJLEVBQUUsQ0FBQyxDQUF1QixFQUFFLE9BQXVCLEVBQUUsRUFBRTtnQkFDekQsSUFBSSxDQUFDLE9BQU8sRUFBRSxDQUFDO29CQUNiLE1BQU0sSUFBSSxLQUFLLENBQUMsb0RBQW9ELENBQUMsQ0FBQTtnQkFDdkUsQ0FBQztnQkFFRCxJQUFJLE9BQU8sQ0FBQyxXQUFXLEVBQUUsQ0FBQztvQkFDeEIsTUFBTSxDQUFDLFNBQVMsQ0FBQyxPQUFPLENBQUMsV0FBVyxDQUFDLENBQUE7Z0JBQ3ZDLENBQUM7Z0JBRUQsT0FBTyxFQUFFLE1BQU0sRUFBRSxZQUFZLEVBQUUsQ0FBQTtZQUNqQyxDQUFDO1NBQ0YsQ0FBQTtJQUNILENBQUM7SUFFRDs7O09BR0c7SUFDSSxNQUFNLENBQUMsS0FBSyxDQUFDLElBQVksRUFBRSxPQUErQjtRQUMvRCxPQUFPO1lBQ0wsSUFBSSxDQUFDLEtBQTJCLEVBQUUsT0FBdUI7Z0JBQ3ZELElBQUksQ0FBQyxPQUFPLEVBQUUsQ0FBQztvQkFDYixNQUFNLElBQUksS0FBSyxDQUFDLG1EQUFtRCxDQUFDLENBQUE7Z0JBQ3RFLENBQUM7Z0JBRUQsSUFBSSxFQUFFLEdBQUcsQ0FBQyxDQUFBO2dCQUNWLE9BQU8sS0FBSyxDQUFDLElBQUksQ0FBQyxZQUFZLENBQUMsUUFBUSxFQUFFLEVBQUUsQ0FBQyxFQUFFLENBQUM7b0JBQzdDLEVBQUUsRUFBRSxDQUFBO2dCQUNOLENBQUM7Z0JBQ0QsTUFBTSxLQUFLLEdBQUcsSUFBSSxRQUFRLENBQUMsS0FBSyxDQUFDLEtBQUssRUFBRSxRQUFRLEVBQUUsRUFBRSxFQUFFO29CQUNwRCxJQUFJO29CQUNKLEdBQUcsT0FBTztpQkFDWCxDQUFDLENBQUE7Z0JBQ0YsSUFBSSxDQUFDLEtBQUssQ0FBQyxZQUFZLEVBQUUsQ0FBQztvQkFDeEIsTUFBTSxJQUFJLEtBQUssQ0FDYixzREFBc0QsQ0FDdkQsQ0FBQTtnQkFDSCxDQUFDO2dCQUVELElBQUksT0FBTyxDQUFDLFdBQVcsRUFBRSxDQUFDO29CQUN4QixLQUFLLENBQUMsU0FBUyxDQUFDLE9BQU8sQ0FBQyxXQUFXLENBQUMsQ0FBQTtnQkFDdEMsQ0FBQztnQkFFRCxPQUFPO29CQUNMLE1BQU0sRUFBRSxLQUFLLENBQUMsTUFBTTtvQkFDcEIsWUFBWSxFQUFFLEtBQUssQ0FBQyxXQUFXO2lCQUNoQyxDQUFBO1lBQ0gsQ0FBQztTQUNGLENBQUE7SUFDSCxDQUFDO0lBRUQsZ0JBQXVCLENBQUM7Q0FDekIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBjb25zdHJ1Y3RzIGZyb20gXCJjb25zdHJ1Y3RzXCJcbmltcG9ydCAqIGFzIGlhbSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWlhbVwiXG5pbXBvcnQgKiBhcyBzMyBmcm9tIFwiYXdzLWNkay1saWIvYXdzLXMzXCJcbmltcG9ydCAqIGFzIHMzQXNzZXRzIGZyb20gXCJhd3MtY2RrLWxpYi9hd3MtczMtYXNzZXRzXCJcblxuLy8gVGhpcyBpcyBtb3N0bHkgYmFzZWQgb24gYXdzLXMzLWRlcGxveW1lbnQgZnJvbSBhd3MtY2RrLlxuXG5leHBvcnQgaW50ZXJmYWNlIFNvdXJjZUNvbmZpZyB7XG4gIC8qKlxuICAgKiBUaGUgc291cmNlIGJ1Y2tldCB0byBkZXBsb3kgZnJvbS5cbiAgICovXG4gIHJlYWRvbmx5IGJ1Y2tldDogczMuSUJ1Y2tldFxuXG4gIC8qKlxuICAgKiBBbiBTMyBvYmplY3Qga2V5IGluIHRoZSBzb3VyY2UgYnVja2V0IHRoYXQgcG9pbnRzIHRvIGEgemlwIGZpbGUuXG4gICAqL1xuICByZWFkb25seSB6aXBPYmplY3RLZXk6IHN0cmluZ1xufVxuXG4vKipcbiAqIEJpbmQgY29udGV4dCBmb3IgSVNvdXJjZXNcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBTb3VyY2VDb250ZXh0IHtcbiAgLyoqXG4gICAqIFRoZSByb2xlIGZvciB0aGUgaGFuZGxlclxuICAgKlxuICAgKiBAZGVmYXVsdCAtIG5vIHBvbGljeSBpcyBtb2RpZmllZFxuICAgKi9cbiAgcmVhZG9ubHkgaGFuZGxlclJvbGU/OiBpYW0uSVJvbGVcbn1cblxuLyoqXG4gKiBSZXByZXNlbnRzIGEgc291cmNlIGZvciBidWNrZXQgZGVwbG95bWVudHMuXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgSVNvdXJjZSB7XG4gIC8qKlxuICAgKiBCaW5kcyB0aGUgc291cmNlIHRvIGEgYnVja2V0IGRlcGxveW1lbnQuXG4gICAqL1xuICBiaW5kKHNjb3BlOiBjb25zdHJ1Y3RzLkNvbnN0cnVjdCwgY29udGV4dD86IFNvdXJjZUNvbnRleHQpOiBTb3VyY2VDb25maWdcbn1cblxuLyoqXG4gKiBTcGVjaWZpZXMgYnVja2V0IGRlcGxveW1lbnQgc291cmNlLlxuICpcbiAqIFVzYWdlOlxuICpcbiAqICAgICBTb3VyY2UuYnVja2V0KGJ1Y2tldCwga2V5KVxuICogICAgIFNvdXJjZS5hc3NldCgnL2xvY2FsL3BhdGgvdG8vZGlyZWN0b3J5JylcbiAqICAgICBTb3VyY2UuYXNzZXQoJy9sb2NhbC9wYXRoL3RvL2EvZmlsZS56aXAnKVxuICpcbiAqL1xuZXhwb3J0IGNsYXNzIFNvdXJjZSB7XG4gIC8qKlxuICAgKiBVc2VzIGEgLnppcCBmaWxlIHN0b3JlZCBpbiBhbiBTMyBidWNrZXQgYXMgdGhlIHNvdXJjZSBmb3IgdGhlIGRlc3RpbmF0aW9uIGJ1Y2tldCBjb250ZW50cy5cbiAgICogQHBhcmFtIGJ1Y2tldCBUaGUgUzMgQnVja2V0XG4gICAqIEBwYXJhbSB6aXBPYmplY3RLZXkgVGhlIFMzIG9iamVjdCBrZXkgb2YgdGhlIHppcCBmaWxlIHdpdGggY29udGVudHNcbiAgICovXG4gIHB1YmxpYyBzdGF0aWMgYnVja2V0KGJ1Y2tldDogczMuSUJ1Y2tldCwgemlwT2JqZWN0S2V5OiBzdHJpbmcpOiBJU291cmNlIHtcbiAgICByZXR1cm4ge1xuICAgICAgYmluZDogKF86IGNvbnN0cnVjdHMuQ29uc3RydWN0LCBjb250ZXh0PzogU291cmNlQ29udGV4dCkgPT4ge1xuICAgICAgICBpZiAoIWNvbnRleHQpIHtcbiAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoXCJUbyB1c2UgYSBTb3VyY2UuYnVja2V0KCksIGNvbnRleHQgbXVzdCBiZSBwcm92aWRlZFwiKVxuICAgICAgICB9XG5cbiAgICAgICAgaWYgKGNvbnRleHQuaGFuZGxlclJvbGUpIHtcbiAgICAgICAgICBidWNrZXQuZ3JhbnRSZWFkKGNvbnRleHQuaGFuZGxlclJvbGUpXG4gICAgICAgIH1cblxuICAgICAgICByZXR1cm4geyBidWNrZXQsIHppcE9iamVjdEtleSB9XG4gICAgICB9LFxuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBVc2VzIGEgbG9jYWwgYXNzZXQgYXMgdGhlIGRlcGxveW1lbnQgc291cmNlLlxuICAgKiBAcGFyYW0gcGF0aCBUaGUgcGF0aCB0byBhIGxvY2FsIC56aXAgZmlsZSBvciBhIGRpcmVjdG9yeVxuICAgKi9cbiAgcHVibGljIHN0YXRpYyBhc3NldChwYXRoOiBzdHJpbmcsIG9wdGlvbnM/OiBzM0Fzc2V0cy5Bc3NldE9wdGlvbnMpOiBJU291cmNlIHtcbiAgICByZXR1cm4ge1xuICAgICAgYmluZChzY29wZTogY29uc3RydWN0cy5Db25zdHJ1Y3QsIGNvbnRleHQ/OiBTb3VyY2VDb250ZXh0KTogU291cmNlQ29uZmlnIHtcbiAgICAgICAgaWYgKCFjb250ZXh0KSB7XG4gICAgICAgICAgdGhyb3cgbmV3IEVycm9yKFwiVG8gdXNlIGEgU291cmNlLmFzc2V0KCksIGNvbnRleHQgbXVzdCBiZSBwcm92aWRlZFwiKVxuICAgICAgICB9XG5cbiAgICAgICAgbGV0IGlkID0gMVxuICAgICAgICB3aGlsZSAoc2NvcGUubm9kZS50cnlGaW5kQ2hpbGQoYEFzc2V0JHtpZH1gKSkge1xuICAgICAgICAgIGlkKytcbiAgICAgICAgfVxuICAgICAgICBjb25zdCBhc3NldCA9IG5ldyBzM0Fzc2V0cy5Bc3NldChzY29wZSwgYEFzc2V0JHtpZH1gLCB7XG4gICAgICAgICAgcGF0aCxcbiAgICAgICAgICAuLi5vcHRpb25zLFxuICAgICAgICB9KVxuICAgICAgICBpZiAoIWFzc2V0LmlzWmlwQXJjaGl2ZSkge1xuICAgICAgICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgICAgICAgIFwiQXNzZXQgcGF0aCBtdXN0IGJlIGVpdGhlciBhIC56aXAgZmlsZSBvciBhIGRpcmVjdG9yeVwiLFxuICAgICAgICAgIClcbiAgICAgICAgfVxuXG4gICAgICAgIGlmIChjb250ZXh0LmhhbmRsZXJSb2xlKSB7XG4gICAgICAgICAgYXNzZXQuZ3JhbnRSZWFkKGNvbnRleHQuaGFuZGxlclJvbGUpXG4gICAgICAgIH1cblxuICAgICAgICByZXR1cm4ge1xuICAgICAgICAgIGJ1Y2tldDogYXNzZXQuYnVja2V0LFxuICAgICAgICAgIHppcE9iamVjdEtleTogYXNzZXQuczNPYmplY3RLZXksXG4gICAgICAgIH1cbiAgICAgIH0sXG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBjb25zdHJ1Y3RvcigpIHt9XG59XG4iXX0=
|
|
@@ -0,0 +1,64 @@
|
|
|
1
|
+
import * as constructs from "constructs";
|
|
2
|
+
import * as cloudfront from "aws-cdk-lib/aws-cloudfront";
|
|
3
|
+
import * as lambda from "aws-cdk-lib/aws-lambda";
|
|
4
|
+
import * as s3 from "aws-cdk-lib/aws-s3";
|
|
5
|
+
import * as cdk from "aws-cdk-lib";
|
|
6
|
+
import { ISource } from "./source";
|
|
7
|
+
export interface WebappDeployProps {
|
|
8
|
+
/**
|
|
9
|
+
* Optional S3 bucket that can be used for deployment from outside CDK.
|
|
10
|
+
*
|
|
11
|
+
* If specified a policy is added so the deploy function can read from
|
|
12
|
+
* the bucket.
|
|
13
|
+
*
|
|
14
|
+
* @default - none
|
|
15
|
+
*/
|
|
16
|
+
buildsBucket?: s3.IBucket;
|
|
17
|
+
/**
|
|
18
|
+
* CloudFront Distribution to be invalidated after deploy.
|
|
19
|
+
*
|
|
20
|
+
* @default - none
|
|
21
|
+
*/
|
|
22
|
+
distribution?: cloudfront.IDistribution;
|
|
23
|
+
/**
|
|
24
|
+
* Regex for patterns of files to be discarded during deployment.
|
|
25
|
+
*
|
|
26
|
+
* Example: `\.map$` will exclude `js/myapp-1b22c248f.js.map`.
|
|
27
|
+
*
|
|
28
|
+
* @default - none
|
|
29
|
+
*/
|
|
30
|
+
excludePattern?: string;
|
|
31
|
+
/**
|
|
32
|
+
* The time when a deployment is considered old and will be deleted
|
|
33
|
+
* unless it is the newest old deployment.
|
|
34
|
+
*
|
|
35
|
+
* @default - 5 days
|
|
36
|
+
*/
|
|
37
|
+
pruneDeploymentsOlderThan?: cdk.Duration;
|
|
38
|
+
/**
|
|
39
|
+
* Name of the lambda function to be created.
|
|
40
|
+
*
|
|
41
|
+
* @default cdk.PhysicalName.GENERATE_IF_NEEDED
|
|
42
|
+
*/
|
|
43
|
+
functionName?: string;
|
|
44
|
+
/**
|
|
45
|
+
* Name of S3 bucket where the contents of the artifacts will be deployed.
|
|
46
|
+
* The files will be deployed under the key "web", which is then expected
|
|
47
|
+
* to be the origin for the CloudFront distribution
|
|
48
|
+
*/
|
|
49
|
+
webBucket: s3.IBucket;
|
|
50
|
+
/**
|
|
51
|
+
* Specific artifact to be deployed to the bucket during CDK deployment.
|
|
52
|
+
*
|
|
53
|
+
* @default - none
|
|
54
|
+
*/
|
|
55
|
+
source?: ISource;
|
|
56
|
+
}
|
|
57
|
+
/**
|
|
58
|
+
* Resource to deploy a webapp from a build artifact into an existing
|
|
59
|
+
* S3 Bucket and CloudFront Distribution.
|
|
60
|
+
*/
|
|
61
|
+
export declare class WebappDeploy extends constructs.Construct {
|
|
62
|
+
readonly deployFn: lambda.Function;
|
|
63
|
+
constructor(scope: constructs.Construct, id: string, props: WebappDeployProps);
|
|
64
|
+
}
|
|
@@ -0,0 +1,64 @@
|
|
|
1
|
+
import * as constructs from "constructs";
|
|
2
|
+
import * as iam from "aws-cdk-lib/aws-iam";
|
|
3
|
+
import * as lambda from "aws-cdk-lib/aws-lambda";
|
|
4
|
+
import * as cdk from "aws-cdk-lib";
|
|
5
|
+
import { Provider } from "aws-cdk-lib/custom-resources";
|
|
6
|
+
import * as path from "path";
|
|
7
|
+
/**
|
|
8
|
+
* Resource to deploy a webapp from a build artifact into an existing
|
|
9
|
+
* S3 Bucket and CloudFront Distribution.
|
|
10
|
+
*/
|
|
11
|
+
export class WebappDeploy extends constructs.Construct {
|
|
12
|
+
deployFn;
|
|
13
|
+
constructor(scope, id, props) {
|
|
14
|
+
super(scope, id);
|
|
15
|
+
const environment = {
|
|
16
|
+
DEPLOY_LOG_BUCKET_URL: `s3://${props.webBucket.bucketName}/deployments.log`,
|
|
17
|
+
EXPIRE_SECONDS: (props.pruneDeploymentsOlderThan ?? cdk.Duration.days(5))
|
|
18
|
+
.toSeconds()
|
|
19
|
+
.toString(),
|
|
20
|
+
TARGET_BUCKET_URL: `s3://${props.webBucket.bucketName}/web`,
|
|
21
|
+
};
|
|
22
|
+
if (props.distribution != null) {
|
|
23
|
+
environment.CF_DISTRIBUTION_ID = props.distribution.distributionId;
|
|
24
|
+
}
|
|
25
|
+
if (props.excludePattern != null) {
|
|
26
|
+
environment.EXCLUDE_PATTERN = props.excludePattern;
|
|
27
|
+
}
|
|
28
|
+
this.deployFn = new lambda.Function(this, "WebappDeployResource", {
|
|
29
|
+
code: lambda.Code.fromAsset(path.join(__dirname, "../dist")),
|
|
30
|
+
environment,
|
|
31
|
+
functionName: props.functionName ?? cdk.PhysicalName.GENERATE_IF_NEEDED,
|
|
32
|
+
handler: "webapp_deploy.main.handler",
|
|
33
|
+
reservedConcurrentExecutions: 1,
|
|
34
|
+
runtime: lambda.Runtime.PYTHON_3_12,
|
|
35
|
+
timeout: cdk.Duration.minutes(2),
|
|
36
|
+
initialPolicy: [
|
|
37
|
+
new iam.PolicyStatement({
|
|
38
|
+
actions: ["cloudfront:CreateInvalidation"],
|
|
39
|
+
// Cannot be restricted
|
|
40
|
+
resources: ["*"],
|
|
41
|
+
}),
|
|
42
|
+
],
|
|
43
|
+
});
|
|
44
|
+
const provider = new Provider(this, "Provider", {
|
|
45
|
+
onEventHandler: this.deployFn,
|
|
46
|
+
});
|
|
47
|
+
props.webBucket.grantReadWrite(this.deployFn);
|
|
48
|
+
if (props.buildsBucket) {
|
|
49
|
+
props.buildsBucket.grantRead(this.deployFn);
|
|
50
|
+
}
|
|
51
|
+
if (props.source) {
|
|
52
|
+
const source = props.source.bind(this, {
|
|
53
|
+
handlerRole: this.deployFn.role,
|
|
54
|
+
});
|
|
55
|
+
new cdk.CustomResource(this, "CustomResource", {
|
|
56
|
+
serviceToken: provider.serviceToken,
|
|
57
|
+
properties: {
|
|
58
|
+
artifactS3Url: `s3://${source.bucket.bucketName}/${source.zipObjectKey}`,
|
|
59
|
+
},
|
|
60
|
+
});
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"webapp-deploy.js","sourceRoot":"","sources":["../src/webapp-deploy.ts"],"names":[],"mappings":"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","sourcesContent":["import * as constructs from \"constructs\"\nimport * as cloudfront 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 s3 from \"aws-cdk-lib/aws-s3\"\nimport * as cdk from \"aws-cdk-lib\"\nimport { Provider } from \"aws-cdk-lib/custom-resources\"\nimport * as path from \"path\"\nimport { ISource } from \"./source\"\n\nexport interface WebappDeployProps {\n  /**\n   * Optional S3 bucket that can be used for deployment from outside CDK.\n   *\n   * If specified a policy is added so the deploy function can read from\n   * the bucket.\n   *\n   * @default - none\n   */\n  buildsBucket?: s3.IBucket\n  /**\n   * CloudFront Distribution to be invalidated after deploy.\n   *\n   * @default - none\n   */\n  distribution?: cloudfront.IDistribution\n  /**\n   * Regex for patterns of files to be discarded during deployment.\n   *\n   * Example: `\\.map$` will exclude `js/myapp-1b22c248f.js.map`.\n   *\n   * @default - none\n   */\n  excludePattern?: string\n  /**\n   * The time when a deployment is considered old and will be deleted\n   * unless it is the newest old deployment.\n   *\n   * @default - 5 days\n   */\n  pruneDeploymentsOlderThan?: cdk.Duration\n  /**\n   * Name of the lambda function to be created.\n   *\n   * @default cdk.PhysicalName.GENERATE_IF_NEEDED\n   */\n  functionName?: string\n  /**\n   * Name of S3 bucket where the contents of the artifacts will be deployed.\n   * The files will be deployed under the key \"web\", which is then expected\n   * to be the origin for the CloudFront distribution\n   */\n  webBucket: s3.IBucket\n  /**\n   * Specific artifact to be deployed to the bucket during CDK deployment.\n   *\n   * @default - none\n   */\n  source?: ISource\n}\n\n/**\n * Resource to deploy a webapp from a build artifact into an existing\n * S3 Bucket and CloudFront Distribution.\n */\nexport class WebappDeploy extends constructs.Construct {\n  readonly deployFn: lambda.Function\n\n  constructor(\n    scope: constructs.Construct,\n    id: string,\n    props: WebappDeployProps,\n  ) {\n    super(scope, id)\n\n    const environment: Record<string, string> = {\n      DEPLOY_LOG_BUCKET_URL: `s3://${props.webBucket.bucketName}/deployments.log`,\n      EXPIRE_SECONDS: (props.pruneDeploymentsOlderThan ?? cdk.Duration.days(5))\n        .toSeconds()\n        .toString(),\n      TARGET_BUCKET_URL: `s3://${props.webBucket.bucketName}/web`,\n    }\n\n    if (props.distribution != null) {\n      environment.CF_DISTRIBUTION_ID = props.distribution.distributionId\n    }\n\n    if (props.excludePattern != null) {\n      environment.EXCLUDE_PATTERN = props.excludePattern\n    }\n\n    this.deployFn = new lambda.Function(this, \"WebappDeployResource\", {\n      code: lambda.Code.fromAsset(path.join(__dirname, \"../dist\")),\n      environment,\n      functionName: props.functionName ?? cdk.PhysicalName.GENERATE_IF_NEEDED,\n      handler: \"webapp_deploy.main.handler\",\n      reservedConcurrentExecutions: 1,\n      runtime: lambda.Runtime.PYTHON_3_12,\n      timeout: cdk.Duration.minutes(2),\n      initialPolicy: [\n        new iam.PolicyStatement({\n          actions: [\"cloudfront:CreateInvalidation\"],\n          // Cannot be restricted\n          resources: [\"*\"],\n        }),\n      ],\n    })\n\n    const provider = new Provider(this, \"Provider\", {\n      onEventHandler: this.deployFn,\n    })\n\n    props.webBucket.grantReadWrite(this.deployFn)\n\n    if (props.buildsBucket) {\n      props.buildsBucket.grantRead(this.deployFn)\n    }\n\n    if (props.source) {\n      const source = props.source.bind(this, {\n        handlerRole: this.deployFn.role!,\n      })\n      new cdk.CustomResource(this, \"CustomResource\", {\n        serviceToken: provider.serviceToken,\n        properties: {\n          artifactS3Url: `s3://${source.bucket.bucketName}/${source.zipObjectKey}`,\n        },\n      })\n    }\n  }\n}\n"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@capraconsulting/webapp-deploy-lambda",
|
|
3
|
-
"version": "2.3.
|
|
3
|
+
"version": "2.3.1",
|
|
4
4
|
"description": "CDK construct for deploying a webapp release to S3 and CloudFront",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"repository": {
|
|
@@ -40,27 +40,27 @@
|
|
|
40
40
|
},
|
|
41
41
|
"devDependencies": {
|
|
42
42
|
"@aws-cdk/assert": "2.68.0",
|
|
43
|
-
"@commitlint/cli": "19.6.
|
|
43
|
+
"@commitlint/cli": "19.6.1",
|
|
44
44
|
"@commitlint/config-conventional": "19.6.0",
|
|
45
45
|
"@eslint/eslintrc": "3.2.0",
|
|
46
|
-
"@eslint/js": "9.
|
|
46
|
+
"@eslint/js": "9.18.0",
|
|
47
47
|
"@types/jest": "29.5.14",
|
|
48
|
-
"@types/node": "22.10.
|
|
49
|
-
"@typescript-eslint/eslint-plugin": "8.
|
|
50
|
-
"@typescript-eslint/parser": "8.
|
|
51
|
-
"aws-cdk-lib": "2.173.
|
|
48
|
+
"@types/node": "22.10.5",
|
|
49
|
+
"@typescript-eslint/eslint-plugin": "8.19.1",
|
|
50
|
+
"@typescript-eslint/parser": "8.19.1",
|
|
51
|
+
"aws-cdk-lib": "2.173.4",
|
|
52
52
|
"constructs": "10.4.2",
|
|
53
|
-
"eslint": "9.
|
|
53
|
+
"eslint": "9.18.0",
|
|
54
54
|
"eslint-config-prettier": "9.1.0",
|
|
55
55
|
"eslint-plugin-prettier": "5.2.1",
|
|
56
56
|
"husky": "9.1.7",
|
|
57
57
|
"jest": "29.7.0",
|
|
58
58
|
"jest-cdk-snapshot": "2.2.5",
|
|
59
59
|
"prettier": "3.4.2",
|
|
60
|
-
"semantic-release": "24.2.
|
|
60
|
+
"semantic-release": "24.2.1",
|
|
61
61
|
"ts-jest": "29.2.5",
|
|
62
62
|
"tsx": "4.19.2",
|
|
63
|
-
"typescript": "5.7.
|
|
63
|
+
"typescript": "5.7.3"
|
|
64
64
|
},
|
|
65
65
|
"peerDependencies": {
|
|
66
66
|
"aws-cdk-lib": "^2.0.0",
|