@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.
Files changed (41) hide show
  1. package/.jsii +637 -0
  2. package/API.md +5671 -0
  3. package/LICENSE +202 -0
  4. package/README.md +86 -0
  5. package/lib/ApiConstructFile.d.ts +5 -0
  6. package/lib/ApiConstructFile.js +175 -0
  7. package/lib/ApiFunctionFile.d.ts +5 -0
  8. package/lib/ApiFunctionFile.js +30 -0
  9. package/lib/Application.d.ts +21 -0
  10. package/lib/Application.js +72 -0
  11. package/lib/ApplicationStackFile.d.ts +5 -0
  12. package/lib/ApplicationStackFile.js +26 -0
  13. package/lib/BaseApplicationStackFile.d.ts +5 -0
  14. package/lib/BaseApplicationStackFile.js +172 -0
  15. package/lib/BatmanProject.d.ts +25 -0
  16. package/lib/BatmanProject.js +101 -0
  17. package/lib/GitHubRolesFile.d.ts +9 -0
  18. package/lib/GitHubRolesFile.js +26 -0
  19. package/lib/GitHubRolesStackFile.d.ts +8 -0
  20. package/lib/GitHubRolesStackFile.js +165 -0
  21. package/lib/LocalDevAppFile.d.ts +10 -0
  22. package/lib/LocalDevAppFile.js +34 -0
  23. package/lib/MainFile.d.ts +24 -0
  24. package/lib/MainFile.js +64 -0
  25. package/lib/MainTestFile.d.ts +5 -0
  26. package/lib/MainTestFile.js +33 -0
  27. package/lib/PrCleanupWorkflow.d.ts +9 -0
  28. package/lib/PrCleanupWorkflow.js +100 -0
  29. package/lib/PrDeployWorkflow.d.ts +10 -0
  30. package/lib/PrDeployWorkflow.js +106 -0
  31. package/lib/ProductionDeployWorkflow.d.ts +11 -0
  32. package/lib/ProductionDeployWorkflow.js +68 -0
  33. package/lib/StagingDeployWorkflow.d.ts +11 -0
  34. package/lib/StagingDeployWorkflow.js +68 -0
  35. package/lib/StaticWebsiteConstructFile.d.ts +5 -0
  36. package/lib/StaticWebsiteConstructFile.js +198 -0
  37. package/lib/ViteReactProject.d.ts +45 -0
  38. package/lib/ViteReactProject.js +654 -0
  39. package/lib/index.d.ts +5 -0
  40. package/lib/index.js +20 -0
  41. 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
+ }