@macedon-technologies/batman 1.0.16
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/.jsii +637 -0
- package/API.md +5671 -0
- package/LICENSE +202 -0
- package/README.md +86 -0
- package/lib/ApiConstructFile.d.ts +5 -0
- package/lib/ApiConstructFile.js +175 -0
- package/lib/ApiFunctionFile.d.ts +5 -0
- package/lib/ApiFunctionFile.js +30 -0
- package/lib/Application.d.ts +21 -0
- package/lib/Application.js +72 -0
- package/lib/ApplicationStackFile.d.ts +5 -0
- package/lib/ApplicationStackFile.js +26 -0
- package/lib/BaseApplicationStackFile.d.ts +5 -0
- package/lib/BaseApplicationStackFile.js +172 -0
- package/lib/BatmanProject.d.ts +25 -0
- package/lib/BatmanProject.js +101 -0
- package/lib/GitHubRolesFile.d.ts +9 -0
- package/lib/GitHubRolesFile.js +26 -0
- package/lib/GitHubRolesStackFile.d.ts +8 -0
- package/lib/GitHubRolesStackFile.js +165 -0
- package/lib/LocalDevAppFile.d.ts +10 -0
- package/lib/LocalDevAppFile.js +34 -0
- package/lib/MainFile.d.ts +24 -0
- package/lib/MainFile.js +64 -0
- package/lib/MainTestFile.d.ts +5 -0
- package/lib/MainTestFile.js +33 -0
- package/lib/PrCleanupWorkflow.d.ts +9 -0
- package/lib/PrCleanupWorkflow.js +100 -0
- package/lib/PrDeployWorkflow.d.ts +10 -0
- package/lib/PrDeployWorkflow.js +106 -0
- package/lib/ProductionDeployWorkflow.d.ts +11 -0
- package/lib/ProductionDeployWorkflow.js +68 -0
- package/lib/StagingDeployWorkflow.d.ts +11 -0
- package/lib/StagingDeployWorkflow.js +68 -0
- package/lib/StaticWebsiteConstructFile.d.ts +5 -0
- package/lib/StaticWebsiteConstructFile.js +198 -0
- package/lib/ViteReactProject.d.ts +45 -0
- package/lib/ViteReactProject.js +654 -0
- package/lib/index.d.ts +5 -0
- package/lib/index.js +20 -0
- package/package.json +125 -0
|
@@ -0,0 +1,198 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.StaticWebsiteConstructFile = void 0;
|
|
4
|
+
const projen_1 = require("projen");
|
|
5
|
+
class StaticWebsiteConstructFile extends projen_1.SampleFile {
|
|
6
|
+
constructor(project) {
|
|
7
|
+
super(project, 'src/constructs/StaticWebsite.ts', {
|
|
8
|
+
contents: `import * as path from 'path';
|
|
9
|
+
import { DockerImage, Duration, RemovalPolicy } from 'aws-cdk-lib';
|
|
10
|
+
import { RestApi } from 'aws-cdk-lib/aws-apigateway';
|
|
11
|
+
import { Certificate, CertificateValidation } from 'aws-cdk-lib/aws-certificatemanager';
|
|
12
|
+
import {
|
|
13
|
+
AllowedMethods,
|
|
14
|
+
CacheHeaderBehavior,
|
|
15
|
+
CachePolicy,
|
|
16
|
+
Distribution,
|
|
17
|
+
ViewerProtocolPolicy,
|
|
18
|
+
} from 'aws-cdk-lib/aws-cloudfront';
|
|
19
|
+
import { RestApiOrigin, S3BucketOrigin } from 'aws-cdk-lib/aws-cloudfront-origins';
|
|
20
|
+
import { PolicyStatement, ServicePrincipal } from 'aws-cdk-lib/aws-iam';
|
|
21
|
+
import { ARecord, IHostedZone, RecordTarget } from 'aws-cdk-lib/aws-route53';
|
|
22
|
+
import { CloudFrontTarget } from 'aws-cdk-lib/aws-route53-targets';
|
|
23
|
+
import { BlockPublicAccess, Bucket, BucketAccessControl } from 'aws-cdk-lib/aws-s3';
|
|
24
|
+
import { BucketDeployment, Source } from 'aws-cdk-lib/aws-s3-deployment';
|
|
25
|
+
import { Construct } from 'constructs';
|
|
26
|
+
|
|
27
|
+
export interface StaticWebsiteProps {
|
|
28
|
+
readonly bucketName?: string;
|
|
29
|
+
readonly hostedZone: IHostedZone;
|
|
30
|
+
readonly api: RestApi;
|
|
31
|
+
readonly hostname?: string;
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
export class StaticWebsite extends Construct {
|
|
35
|
+
public bucket: Bucket;
|
|
36
|
+
public distribution: Distribution;
|
|
37
|
+
public certificate: Certificate;
|
|
38
|
+
|
|
39
|
+
private apiCachePolicy: CachePolicy;
|
|
40
|
+
private aRecord: ARecord;
|
|
41
|
+
private bucketDeployment: BucketDeployment;
|
|
42
|
+
private hostname: string;
|
|
43
|
+
|
|
44
|
+
constructor(scope: Construct, id: string, private props: StaticWebsiteProps) {
|
|
45
|
+
super(scope, id);
|
|
46
|
+
|
|
47
|
+
const domainName = props.hostedZone.zoneName;
|
|
48
|
+
const hostname = props.hostname || 'mvp';
|
|
49
|
+
this.hostname = \`\${hostname}.\${domainName}\`;
|
|
50
|
+
|
|
51
|
+
this.createCertificate();
|
|
52
|
+
if (!this.certificate) {
|
|
53
|
+
throw new Error('Failed to create certificate');
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
this.createBucket();
|
|
57
|
+
if (!this.bucket) {
|
|
58
|
+
throw new Error('Failed to create bucket');
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
this.createApiCachePolicy();
|
|
62
|
+
if (!this.apiCachePolicy) {
|
|
63
|
+
throw new Error('Failed to create API cache policy');
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
this.createDistribution();
|
|
67
|
+
if (!this.distribution) {
|
|
68
|
+
throw new Error('Failed to create distribution');
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
this.createDnsRecord();
|
|
72
|
+
if (!this.aRecord) {
|
|
73
|
+
throw new Error('Failed to create DNS A record');
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
this.addBucketPolicy();
|
|
77
|
+
this.createBucketDeployment();
|
|
78
|
+
if (!this.bucketDeployment) {
|
|
79
|
+
throw new Error('Failed to create bucket deployment');
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
protected createCertificate() {
|
|
84
|
+
this.certificate = new Certificate(this, 'Certificate', {
|
|
85
|
+
domainName: this.hostname,
|
|
86
|
+
validation: CertificateValidation.fromDns(this.props.hostedZone),
|
|
87
|
+
});
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
protected createBucket() {
|
|
91
|
+
this.bucket = new Bucket(this, 'Bucket', {
|
|
92
|
+
blockPublicAccess: BlockPublicAccess.BLOCK_ALL,
|
|
93
|
+
accessControl: BucketAccessControl.PRIVATE,
|
|
94
|
+
removalPolicy: RemovalPolicy.DESTROY,
|
|
95
|
+
autoDeleteObjects: true,
|
|
96
|
+
});
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
protected createApiCachePolicy() {
|
|
100
|
+
this.apiCachePolicy = new CachePolicy(this, 'ApiCachePolicy', {
|
|
101
|
+
comment: 'Cache policy for API Gateway with dynamic content',
|
|
102
|
+
defaultTtl: Duration.seconds(0),
|
|
103
|
+
maxTtl: Duration.seconds(1),
|
|
104
|
+
minTtl: Duration.seconds(0),
|
|
105
|
+
enableAcceptEncodingBrotli: true,
|
|
106
|
+
enableAcceptEncodingGzip: true,
|
|
107
|
+
headerBehavior: CacheHeaderBehavior.allowList('Authorization'),
|
|
108
|
+
});
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
protected createDistribution() {
|
|
112
|
+
this.distribution = new Distribution(this, 'Distribution', {
|
|
113
|
+
domainNames: [this.hostname],
|
|
114
|
+
certificate: this.certificate,
|
|
115
|
+
defaultBehavior: {
|
|
116
|
+
origin: S3BucketOrigin.withOriginAccessControl(this.bucket),
|
|
117
|
+
viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,
|
|
118
|
+
},
|
|
119
|
+
additionalBehaviors: {
|
|
120
|
+
'/api/*': {
|
|
121
|
+
origin: new RestApiOrigin(this.props.api),
|
|
122
|
+
viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,
|
|
123
|
+
allowedMethods: AllowedMethods.ALLOW_ALL,
|
|
124
|
+
cachePolicy: this.apiCachePolicy,
|
|
125
|
+
|
|
126
|
+
},
|
|
127
|
+
},
|
|
128
|
+
defaultRootObject: 'index.html',
|
|
129
|
+
errorResponses: [
|
|
130
|
+
{
|
|
131
|
+
httpStatus: 404,
|
|
132
|
+
responsePagePath: '/index.html',
|
|
133
|
+
},
|
|
134
|
+
{
|
|
135
|
+
httpStatus: 403,
|
|
136
|
+
responsePagePath: '/index.html',
|
|
137
|
+
},
|
|
138
|
+
],
|
|
139
|
+
});
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
protected createDnsRecord() {
|
|
143
|
+
this.aRecord = new ARecord(this, 'ARecord', {
|
|
144
|
+
zone: this.props.hostedZone,
|
|
145
|
+
recordName: this.hostname,
|
|
146
|
+
target: RecordTarget.fromAlias(new CloudFrontTarget(this.distribution)),
|
|
147
|
+
});
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
protected addBucketPolicy() {
|
|
151
|
+
this.bucket.addToResourcePolicy(
|
|
152
|
+
new PolicyStatement({
|
|
153
|
+
sid: 'AllowCloudFrontServicePrincipal',
|
|
154
|
+
actions: ['s3:GetObject'],
|
|
155
|
+
resources: [\`\${this.bucket.bucketArn}/*\`],
|
|
156
|
+
principals: [new ServicePrincipal('cloudfront.amazonaws.com')],
|
|
157
|
+
conditions: {
|
|
158
|
+
StringEquals: {
|
|
159
|
+
'AWS:SourceArn': \`arn:aws:cloudfront::\${this.bucket.stack.account}:distribution/\${this.distribution.distributionId}\`,
|
|
160
|
+
},
|
|
161
|
+
},
|
|
162
|
+
}),
|
|
163
|
+
);
|
|
164
|
+
}
|
|
165
|
+
|
|
166
|
+
protected createBucketDeployment() {
|
|
167
|
+
this.bucketDeployment = new BucketDeployment(this, 'BucketDeployment', {
|
|
168
|
+
sources: [Source.asset(path.join(__dirname, '../..'), {
|
|
169
|
+
bundling: {
|
|
170
|
+
image: DockerImage.fromRegistry('node:20-alpine'),
|
|
171
|
+
environment: {
|
|
172
|
+
npm_config_cache: '/tmp/.npm',
|
|
173
|
+
},
|
|
174
|
+
command: [
|
|
175
|
+
'sh',
|
|
176
|
+
'-c',
|
|
177
|
+
[
|
|
178
|
+
'cd /asset-input/web',
|
|
179
|
+
'yarn install --frozen-lockfile --cache-folder /tmp/.yarn',
|
|
180
|
+
'yarn build',
|
|
181
|
+
\`sed -i 's|"apiBaseUrl": "[^"]*"|"apiBaseUrl": "https://\${this.hostname}/api"|g' dist/config.json\`,
|
|
182
|
+
'cp -r dist/* /asset-output/',
|
|
183
|
+
].join(' && '),
|
|
184
|
+
],
|
|
185
|
+
},
|
|
186
|
+
})],
|
|
187
|
+
destinationBucket: this.bucket,
|
|
188
|
+
distribution: this.distribution,
|
|
189
|
+
distributionPaths: ['/*'],
|
|
190
|
+
});
|
|
191
|
+
}
|
|
192
|
+
}
|
|
193
|
+
`,
|
|
194
|
+
});
|
|
195
|
+
}
|
|
196
|
+
}
|
|
197
|
+
exports.StaticWebsiteConstructFile = StaticWebsiteConstructFile;
|
|
198
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"StaticWebsiteConstructFile.js","sourceRoot":"","sources":["../src/StaticWebsiteConstructFile.ts"],"names":[],"mappings":";;;AAAA,mCAAoC;AAGpC,MAAa,0BAA2B,SAAQ,mBAAU;IACxD,YAAY,OAA4B;QACtC,KAAK,CAAC,OAAO,EAAE,iCAAiC,EAAE;YAChD,QAAQ,EAAE;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;CAyLf;SACI,CAAC,CAAC;IACL,CAAC;CACF;AA/LD,gEA+LC","sourcesContent":["import { SampleFile } from 'projen';\nimport { AwsCdkTypeScriptApp } from 'projen/lib/awscdk';\n\nexport class StaticWebsiteConstructFile extends SampleFile {\n  constructor(project: AwsCdkTypeScriptApp) {\n    super(project, 'src/constructs/StaticWebsite.ts', {\n      contents: `import * as path from 'path';\nimport { DockerImage, Duration, RemovalPolicy } from 'aws-cdk-lib';\nimport { RestApi } from 'aws-cdk-lib/aws-apigateway';\nimport { Certificate, CertificateValidation } from 'aws-cdk-lib/aws-certificatemanager';\nimport {\n  AllowedMethods,\n  CacheHeaderBehavior,\n  CachePolicy,\n  Distribution,\n  ViewerProtocolPolicy,\n} from 'aws-cdk-lib/aws-cloudfront';\nimport { RestApiOrigin, S3BucketOrigin } from 'aws-cdk-lib/aws-cloudfront-origins';\nimport { PolicyStatement, ServicePrincipal } from 'aws-cdk-lib/aws-iam';\nimport { ARecord, IHostedZone, RecordTarget } from 'aws-cdk-lib/aws-route53';\nimport { CloudFrontTarget } from 'aws-cdk-lib/aws-route53-targets';\nimport { BlockPublicAccess, Bucket, BucketAccessControl } from 'aws-cdk-lib/aws-s3';\nimport { BucketDeployment, Source } from 'aws-cdk-lib/aws-s3-deployment';\nimport { Construct } from 'constructs';\n\nexport interface StaticWebsiteProps {\n  readonly bucketName?: string;\n  readonly hostedZone: IHostedZone;\n  readonly api: RestApi;\n  readonly hostname?: string;\n}\n\nexport class StaticWebsite extends Construct {\n  public bucket: Bucket;\n  public distribution: Distribution;\n  public certificate: Certificate;\n\n  private apiCachePolicy: CachePolicy;\n  private aRecord: ARecord;\n  private bucketDeployment: BucketDeployment;\n  private hostname: string;\n\n  constructor(scope: Construct, id: string, private props: StaticWebsiteProps) {\n    super(scope, id);\n\n    const domainName = props.hostedZone.zoneName;\n    const hostname = props.hostname || 'mvp';\n    this.hostname = \\`\\${hostname}.\\${domainName}\\`;\n\n    this.createCertificate();\n    if (!this.certificate) {\n      throw new Error('Failed to create certificate');\n    }\n\n    this.createBucket();\n    if (!this.bucket) {\n      throw new Error('Failed to create bucket');\n    }\n\n    this.createApiCachePolicy();\n    if (!this.apiCachePolicy) {\n      throw new Error('Failed to create API cache policy');\n    }\n\n    this.createDistribution();\n    if (!this.distribution) {\n      throw new Error('Failed to create distribution');\n    }\n\n    this.createDnsRecord();\n    if (!this.aRecord) {\n      throw new Error('Failed to create DNS A record');\n    }\n\n    this.addBucketPolicy();\n    this.createBucketDeployment();\n    if (!this.bucketDeployment) {\n      throw new Error('Failed to create bucket deployment');\n    }\n  }\n\n  protected createCertificate() {\n    this.certificate = new Certificate(this, 'Certificate', {\n      domainName: this.hostname,\n      validation: CertificateValidation.fromDns(this.props.hostedZone),\n    });\n  }\n\n  protected createBucket() {\n    this.bucket = new Bucket(this, 'Bucket', {\n      blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n      accessControl: BucketAccessControl.PRIVATE,\n      removalPolicy: RemovalPolicy.DESTROY,\n      autoDeleteObjects: true,\n    });\n  }\n\n  protected createApiCachePolicy() {\n    this.apiCachePolicy = new CachePolicy(this, 'ApiCachePolicy', {\n      comment: 'Cache policy for API Gateway with dynamic content',\n      defaultTtl: Duration.seconds(0),\n      maxTtl: Duration.seconds(1),\n      minTtl: Duration.seconds(0),\n      enableAcceptEncodingBrotli: true,\n      enableAcceptEncodingGzip: true,\n      headerBehavior: CacheHeaderBehavior.allowList('Authorization'),\n    });\n  }\n\n  protected createDistribution() {\n    this.distribution = new Distribution(this, 'Distribution', {\n      domainNames: [this.hostname],\n      certificate: this.certificate,\n      defaultBehavior: {\n        origin: S3BucketOrigin.withOriginAccessControl(this.bucket),\n        viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n      },\n      additionalBehaviors: {\n        '/api/*': {\n          origin: new RestApiOrigin(this.props.api),\n          viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n          allowedMethods: AllowedMethods.ALLOW_ALL,\n          cachePolicy: this.apiCachePolicy,\n\n        },\n      },\n      defaultRootObject: 'index.html',\n      errorResponses: [\n        {\n          httpStatus: 404,\n          responsePagePath: '/index.html',\n        },\n        {\n          httpStatus: 403,\n          responsePagePath: '/index.html',\n        },\n      ],\n    });\n  }\n\n  protected createDnsRecord() {\n    this.aRecord = new ARecord(this, 'ARecord', {\n      zone: this.props.hostedZone,\n      recordName: this.hostname,\n      target: RecordTarget.fromAlias(new CloudFrontTarget(this.distribution)),\n    });\n  }\n\n  protected addBucketPolicy() {\n    this.bucket.addToResourcePolicy(\n      new PolicyStatement({\n        sid: 'AllowCloudFrontServicePrincipal',\n        actions: ['s3:GetObject'],\n        resources: [\\`\\${this.bucket.bucketArn}/*\\`],\n        principals: [new ServicePrincipal('cloudfront.amazonaws.com')],\n        conditions: {\n          StringEquals: {\n            'AWS:SourceArn': \\`arn:aws:cloudfront::\\${this.bucket.stack.account}:distribution/\\${this.distribution.distributionId}\\`,\n          },\n        },\n      }),\n    );\n  }\n\n  protected createBucketDeployment() {\n    this.bucketDeployment = new BucketDeployment(this, 'BucketDeployment', {\n      sources: [Source.asset(path.join(__dirname, '../..'), {\n        bundling: {\n          image: DockerImage.fromRegistry('node:20-alpine'),\n          environment: {\n            npm_config_cache: '/tmp/.npm',\n          },\n          command: [\n            'sh',\n            '-c',\n            [\n              'cd /asset-input/web',\n              'yarn install --frozen-lockfile --cache-folder /tmp/.yarn',\n              'yarn build',\n              \\`sed -i 's|\"apiBaseUrl\": \"[^\"]*\"|\"apiBaseUrl\": \"https://\\${this.hostname}/api\"|g' dist/config.json\\`,\n              'cp -r dist/* /asset-output/',\n            ].join(' && '),\n          ],\n        },\n      })],\n      destinationBucket: this.bucket,\n      distribution: this.distribution,\n      distributionPaths: ['/*'],\n    });\n  }\n}\n`,\n    });\n  }\n}\n"]}
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
import { TypeScriptAppProject, TypeScriptProjectOptions } from 'projen/lib/typescript';
|
|
2
|
+
export interface ViteReactProjectOptions extends TypeScriptProjectOptions {
|
|
3
|
+
/**
|
|
4
|
+
* Whether to generate sample code
|
|
5
|
+
* @default true
|
|
6
|
+
*/
|
|
7
|
+
readonly sampleCode?: boolean;
|
|
8
|
+
/**
|
|
9
|
+
* Source directory
|
|
10
|
+
* @default "src"
|
|
11
|
+
*/
|
|
12
|
+
readonly srcdir?: string;
|
|
13
|
+
/**
|
|
14
|
+
* Public directory for static assets
|
|
15
|
+
* @default "public"
|
|
16
|
+
*/
|
|
17
|
+
readonly publicDir?: string;
|
|
18
|
+
/**
|
|
19
|
+
* Title for the HTML page
|
|
20
|
+
* @default project name
|
|
21
|
+
*/
|
|
22
|
+
readonly title?: string;
|
|
23
|
+
}
|
|
24
|
+
/**
|
|
25
|
+
* Vite + React TypeScript project
|
|
26
|
+
*/
|
|
27
|
+
export declare class ViteReactProject extends TypeScriptAppProject {
|
|
28
|
+
readonly srcdir: string;
|
|
29
|
+
readonly publicDir: string;
|
|
30
|
+
readonly devCommand: string;
|
|
31
|
+
private readonly configFields;
|
|
32
|
+
constructor(options: ViteReactProjectOptions);
|
|
33
|
+
/**
|
|
34
|
+
* Add a custom field to the Config type
|
|
35
|
+
* @param fieldName The name of the field
|
|
36
|
+
* @param fieldType The TypeScript type of the field (e.g., 'string', 'number', 'boolean', 'string[]')
|
|
37
|
+
* @returns this project instance for chaining
|
|
38
|
+
*/
|
|
39
|
+
addConfigField(fieldName: string, fieldType: string): this;
|
|
40
|
+
/**
|
|
41
|
+
* Get the config fields map (for internal use by ConfigFile)
|
|
42
|
+
* @internal
|
|
43
|
+
*/
|
|
44
|
+
getConfigFields(): Map<string, string>;
|
|
45
|
+
}
|