@orcabus/platform-cdk-constructs 0.0.4
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 +4692 -0
- package/README.md +1 -0
- package/api-gateway/README.md +26 -0
- package/api-gateway/api-gateway.d.ts +78 -0
- package/api-gateway/api-gateway.js +144 -0
- package/api-gateway/config.d.ts +38 -0
- package/api-gateway/config.js +38 -0
- package/api-gateway/index.d.ts +2 -0
- package/api-gateway/index.js +19 -0
- package/deployment-stack-pipeline/README.md +27 -0
- package/deployment-stack-pipeline/config.d.ts +5 -0
- package/deployment-stack-pipeline/config.js +20 -0
- package/deployment-stack-pipeline/index.d.ts +2 -0
- package/deployment-stack-pipeline/index.js +19 -0
- package/deployment-stack-pipeline/pipeline.d.ts +104 -0
- package/deployment-stack-pipeline/pipeline.js +117 -0
- package/index.d.ts +3 -0
- package/index.js +7 -0
- package/index.ts +3 -0
- package/package.json +47 -0
- package/utils/index.d.ts +1 -0
- package/utils/index.js +3 -0
package/README.md
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
# CDK Constructs Package for OrcaBus
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
# ApiGatewayConstruct
|
|
2
|
+
|
|
3
|
+
Usage example:
|
|
4
|
+
|
|
5
|
+
```ts
|
|
6
|
+
const apiGateway = new ApiGatewayConstruct(this, 'ApiGateway', props.apiGatewayCognitoProps);
|
|
7
|
+
const httpApi = apiGateway.httpApi;
|
|
8
|
+
|
|
9
|
+
const apiIntegration = new HttpLambdaIntegration('ApiIntegration', apiFn);
|
|
10
|
+
|
|
11
|
+
new HttpRoute(this, 'GetHttpRoute', {
|
|
12
|
+
httpApi: httpApi,
|
|
13
|
+
integration: apiIntegration,
|
|
14
|
+
routeKey: HttpRouteKey.with('/{proxy+}', HttpMethod.GET),
|
|
15
|
+
});
|
|
16
|
+
|
|
17
|
+
// To protect the path/method with an admin role within the User Pool Cognito
|
|
18
|
+
// More details on ./.../stateful/authorization-stack
|
|
19
|
+
|
|
20
|
+
new HttpRoute(this, 'PostHttpRoute', {
|
|
21
|
+
httpApi: httpApi,
|
|
22
|
+
integration: apiIntegration,
|
|
23
|
+
authorizer: apiGateway.authStackHttpLambdaAuthorizer,
|
|
24
|
+
routeKey: HttpRouteKey.with('/{proxy+}', HttpMethod.POST),
|
|
25
|
+
});
|
|
26
|
+
```
|
|
@@ -0,0 +1,78 @@
|
|
|
1
|
+
import { Construct } from "constructs";
|
|
2
|
+
import { RemovalPolicy } from "aws-cdk-lib";
|
|
3
|
+
import { HttpLambdaAuthorizer } from "aws-cdk-lib/aws-apigatewayv2-authorizers";
|
|
4
|
+
import { HttpApi } from "aws-cdk-lib/aws-apigatewayv2";
|
|
5
|
+
import { RetentionDays } from "aws-cdk-lib/aws-logs";
|
|
6
|
+
export interface ApiGwLogsConfig {
|
|
7
|
+
/**
|
|
8
|
+
* The number of days log events are kept in CloudWatch Logs.
|
|
9
|
+
*/
|
|
10
|
+
readonly retention: RetentionDays;
|
|
11
|
+
/**
|
|
12
|
+
* The removal policy to apply to the log group.
|
|
13
|
+
*/
|
|
14
|
+
readonly removalPolicy: RemovalPolicy;
|
|
15
|
+
}
|
|
16
|
+
export interface OrcaBusApiGatewayProps {
|
|
17
|
+
/**
|
|
18
|
+
* The name of the API.
|
|
19
|
+
*/
|
|
20
|
+
readonly apiName: string;
|
|
21
|
+
/**
|
|
22
|
+
* The prefix for the custom domain name
|
|
23
|
+
*/
|
|
24
|
+
readonly customDomainNamePrefix: string;
|
|
25
|
+
/**
|
|
26
|
+
*The cognito user pool id parameter name.
|
|
27
|
+
|
|
28
|
+
@default DEFAULT_COGNITO_USER_POOL_ID_PARAMETER_NAME
|
|
29
|
+
*/
|
|
30
|
+
readonly cognitoUserPoolIdParameterName?: string;
|
|
31
|
+
/**
|
|
32
|
+
* The parameter name for the cognito client id in array.
|
|
33
|
+
* In order API Gateway to validate the JWT token, it needs to know the client id which usually
|
|
34
|
+
* stored in SSM Parameter. This will accept multiple parameter name in an array.
|
|
35
|
+
*
|
|
36
|
+
* @default DEFAULT_COGNITO_CLIENT_ID_PARAMETER_NAME_ARRAY
|
|
37
|
+
*/
|
|
38
|
+
readonly cognitoClientIdParameterNameArray?: string[];
|
|
39
|
+
/**
|
|
40
|
+
* The configuration for aws cloudwatch logs
|
|
41
|
+
*/
|
|
42
|
+
readonly apiGwLogsConfig: ApiGwLogsConfig;
|
|
43
|
+
/**
|
|
44
|
+
* Allowed CORS origins.
|
|
45
|
+
*/
|
|
46
|
+
readonly corsAllowOrigins: string[];
|
|
47
|
+
}
|
|
48
|
+
export declare class OrcaBusApiGateway extends Construct {
|
|
49
|
+
/**
|
|
50
|
+
* The AWS region where the API Gateway is deployed.
|
|
51
|
+
*/
|
|
52
|
+
readonly region: string;
|
|
53
|
+
/**
|
|
54
|
+
* The HTTP API
|
|
55
|
+
*/
|
|
56
|
+
readonly httpApi: HttpApi;
|
|
57
|
+
/**
|
|
58
|
+
* Domain name defined in this gateway
|
|
59
|
+
*/
|
|
60
|
+
readonly domainName: string;
|
|
61
|
+
/**
|
|
62
|
+
* The Lambda HTTP Authorizer used to enforce authorization policies
|
|
63
|
+
* defined in the authorization stack.
|
|
64
|
+
*
|
|
65
|
+
* To use this, set it as the `authorizer` property in an `HttpRoute` construct.
|
|
66
|
+
* Example: `authorizer: apiGateway.authStackHttpLambdaAuthorizer`
|
|
67
|
+
*/
|
|
68
|
+
readonly authStackHttpLambdaAuthorizer: HttpLambdaAuthorizer;
|
|
69
|
+
constructor(scope: Construct, id: string, props: OrcaBusApiGatewayProps);
|
|
70
|
+
private setupAccessLogs;
|
|
71
|
+
private getJWTAuthorizer;
|
|
72
|
+
/**
|
|
73
|
+
* Get the HTTP Lambda Authorizer defined in the authorization stack manager
|
|
74
|
+
* @param authStackHttpLambdaAuthorizerParameterName The SSM Parameter Name that stores the ARN of the lambda authorizer
|
|
75
|
+
* @returns
|
|
76
|
+
*/
|
|
77
|
+
private getAuthStackHTTPLambdaAuthorizer;
|
|
78
|
+
}
|
|
@@ -0,0 +1,144 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var _a;
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.OrcaBusApiGateway = void 0;
|
|
5
|
+
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
6
|
+
const constructs_1 = require("constructs");
|
|
7
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
8
|
+
const aws_apigatewayv2_authorizers_1 = require("aws-cdk-lib/aws-apigatewayv2-authorizers");
|
|
9
|
+
const aws_apigatewayv2_1 = require("aws-cdk-lib/aws-apigatewayv2");
|
|
10
|
+
const aws_certificatemanager_1 = require("aws-cdk-lib/aws-certificatemanager");
|
|
11
|
+
const aws_ssm_1 = require("aws-cdk-lib/aws-ssm");
|
|
12
|
+
const aws_logs_1 = require("aws-cdk-lib/aws-logs");
|
|
13
|
+
const aws_route53_1 = require("aws-cdk-lib/aws-route53");
|
|
14
|
+
const aws_iam_1 = require("aws-cdk-lib/aws-iam");
|
|
15
|
+
const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
|
|
16
|
+
const aws_route53_targets_1 = require("aws-cdk-lib/aws-route53-targets");
|
|
17
|
+
const config_1 = require("./config");
|
|
18
|
+
class OrcaBusApiGateway extends constructs_1.Construct {
|
|
19
|
+
constructor(scope, id, props) {
|
|
20
|
+
super(scope, id);
|
|
21
|
+
this.region = aws_cdk_lib_1.Stack.of(this).region;
|
|
22
|
+
// umccr acm arn
|
|
23
|
+
const umccrAcmArn = aws_ssm_1.StringParameter.valueForStringParameter(this, "/umccr/certificate_arn");
|
|
24
|
+
const hostedDomainName = aws_ssm_1.StringParameter.valueForStringParameter(this, "/hosted_zone/umccr/name");
|
|
25
|
+
const hostedZoneId = aws_ssm_1.StringParameter.valueForStringParameter(this, "/hosted_zone/umccr/id");
|
|
26
|
+
this.domainName = `${props.customDomainNamePrefix}.${hostedDomainName}`;
|
|
27
|
+
const apiGWDomainName = new aws_apigatewayv2_1.DomainName(this, "UmccrDomainName", {
|
|
28
|
+
domainName: this.domainName,
|
|
29
|
+
certificate: aws_certificatemanager_1.Certificate.fromCertificateArn(this, "cert", umccrAcmArn),
|
|
30
|
+
});
|
|
31
|
+
this.httpApi = new aws_apigatewayv2_1.HttpApi(this, "HttpApi", {
|
|
32
|
+
apiName: "OrcaBusAPI-" + props.apiName,
|
|
33
|
+
corsPreflight: {
|
|
34
|
+
allowHeaders: [
|
|
35
|
+
"content-type",
|
|
36
|
+
"content-disposition",
|
|
37
|
+
"authorization",
|
|
38
|
+
"x-amz-date",
|
|
39
|
+
"x-api-key",
|
|
40
|
+
"x-amz-security-token",
|
|
41
|
+
"x-amz-user-agent",
|
|
42
|
+
],
|
|
43
|
+
allowMethods: [
|
|
44
|
+
aws_apigatewayv2_1.CorsHttpMethod.GET,
|
|
45
|
+
aws_apigatewayv2_1.CorsHttpMethod.HEAD,
|
|
46
|
+
aws_apigatewayv2_1.CorsHttpMethod.OPTIONS,
|
|
47
|
+
aws_apigatewayv2_1.CorsHttpMethod.POST,
|
|
48
|
+
aws_apigatewayv2_1.CorsHttpMethod.PATCH,
|
|
49
|
+
aws_apigatewayv2_1.CorsHttpMethod.DELETE,
|
|
50
|
+
],
|
|
51
|
+
allowOrigins: props.corsAllowOrigins,
|
|
52
|
+
maxAge: aws_cdk_lib_1.Duration.days(10),
|
|
53
|
+
},
|
|
54
|
+
defaultAuthorizer: this.getJWTAuthorizer(props),
|
|
55
|
+
defaultDomainMapping: {
|
|
56
|
+
domainName: apiGWDomainName,
|
|
57
|
+
},
|
|
58
|
+
});
|
|
59
|
+
this.authStackHttpLambdaAuthorizer =
|
|
60
|
+
this.getAuthStackHTTPLambdaAuthorizer();
|
|
61
|
+
new aws_route53_1.ARecord(this, "CustomDomainARecord", {
|
|
62
|
+
zone: aws_route53_1.HostedZone.fromHostedZoneAttributes(this, "UmccrHostedZone", {
|
|
63
|
+
hostedZoneId,
|
|
64
|
+
zoneName: hostedDomainName,
|
|
65
|
+
}),
|
|
66
|
+
recordName: this.domainName,
|
|
67
|
+
target: aws_route53_1.RecordTarget.fromAlias(new aws_route53_targets_1.ApiGatewayv2DomainProperties(apiGWDomainName.regionalDomainName, apiGWDomainName.regionalHostedZoneId)),
|
|
68
|
+
});
|
|
69
|
+
// LogGroups
|
|
70
|
+
this.setupAccessLogs(props.apiGwLogsConfig);
|
|
71
|
+
// CloudMap
|
|
72
|
+
// this.setupCloudServiceDiscovery()
|
|
73
|
+
}
|
|
74
|
+
// TODO: https://github.com/aws-samples/aws-cdk-service-discovery-example/tree/main
|
|
75
|
+
// private setupCloudServiceDiscovery() {
|
|
76
|
+
// }
|
|
77
|
+
// TODO: Taken from https://github.com/aws/aws-cdk/issues/11100#issuecomment-904627081
|
|
78
|
+
// Monitor for higher level CDK construct instead of leveraging CfnStage
|
|
79
|
+
setupAccessLogs(props) {
|
|
80
|
+
const accessLogs = new aws_logs_1.LogGroup(this, "ApiGwAccessLogs", {
|
|
81
|
+
retention: props.retention,
|
|
82
|
+
removalPolicy: props.removalPolicy,
|
|
83
|
+
});
|
|
84
|
+
const stage = this.httpApi.defaultStage?.node.defaultChild;
|
|
85
|
+
stage.accessLogSettings = {
|
|
86
|
+
destinationArn: accessLogs.logGroupArn,
|
|
87
|
+
format: JSON.stringify({
|
|
88
|
+
requestId: "$context.requestId",
|
|
89
|
+
userAgent: "$context.identity.userAgent",
|
|
90
|
+
sourceIp: "$context.identity.sourceIp",
|
|
91
|
+
requestTime: "$context.requestTime",
|
|
92
|
+
requestTimeEpoch: "$context.requestTimeEpoch",
|
|
93
|
+
httpMethod: "$context.httpMethod",
|
|
94
|
+
path: "$context.path",
|
|
95
|
+
status: "$context.status",
|
|
96
|
+
protocol: "$context.protocol",
|
|
97
|
+
responseLength: "$context.responseLength",
|
|
98
|
+
domainName: "$context.domainName",
|
|
99
|
+
}),
|
|
100
|
+
};
|
|
101
|
+
// Allow writing access logs, managed
|
|
102
|
+
const role = new aws_iam_1.Role(this, "AmazonAPIGatewayPushToCloudWatchLogs", {
|
|
103
|
+
assumedBy: new aws_iam_1.ServicePrincipal("apigateway.amazonaws.com"),
|
|
104
|
+
});
|
|
105
|
+
accessLogs.grantWrite(role);
|
|
106
|
+
}
|
|
107
|
+
getJWTAuthorizer({ cognitoUserPoolIdParameterName = config_1.DEFAULT_COGNITO_USER_POOL_ID_PARAMETER_NAME, cognitoClientIdParameterNameArray = config_1.DEFAULT_COGNITO_CLIENT_ID_PARAMETER_NAME_ARRAY, }) {
|
|
108
|
+
/**
|
|
109
|
+
* FIXME One fine day in future when we have proper Cognito AAI setup.
|
|
110
|
+
* For the moment, we leverage Portal and established Cognito infrastructure.
|
|
111
|
+
* See https://github.com/umccr/orcabus/issues/102
|
|
112
|
+
*/
|
|
113
|
+
const userPoolIdParam = aws_ssm_1.StringParameter.fromStringParameterName(this, "CognitoUserPoolIdParameter", cognitoUserPoolIdParameterName);
|
|
114
|
+
const clientIdParamsArray = cognitoClientIdParameterNameArray.map((name) => aws_cdk_lib_1.aws_ssm.StringParameter.fromStringParameterName(this, `CognitoClientId${name}Parameter`, name));
|
|
115
|
+
const issuer = "https://cognito-idp." +
|
|
116
|
+
this.region +
|
|
117
|
+
".amazonaws.com/" +
|
|
118
|
+
userPoolIdParam.stringValue;
|
|
119
|
+
return new aws_apigatewayv2_authorizers_1.HttpJwtAuthorizer("PortalAuthorizer", issuer, {
|
|
120
|
+
jwtAudience: clientIdParamsArray.map((param) => param.stringValue),
|
|
121
|
+
});
|
|
122
|
+
}
|
|
123
|
+
/**
|
|
124
|
+
* Get the HTTP Lambda Authorizer defined in the authorization stack manager
|
|
125
|
+
* @param authStackHttpLambdaAuthorizerParameterName The SSM Parameter Name that stores the ARN of the lambda authorizer
|
|
126
|
+
* @returns
|
|
127
|
+
*/
|
|
128
|
+
getAuthStackHTTPLambdaAuthorizer() {
|
|
129
|
+
const lambdaArn = aws_ssm_1.StringParameter.valueForStringParameter(this, "/orcabus/authorization-stack/http-lambda-authorization-arn");
|
|
130
|
+
// Get the lambda HTTP authorizer defined in the authorization stack manager
|
|
131
|
+
const lambdaAuthorizer = aws_lambda_1.Function.fromFunctionAttributes(this, "AuthStackHTTPLambdaAuthorizer", {
|
|
132
|
+
functionArn: lambdaArn,
|
|
133
|
+
sameEnvironment: true,
|
|
134
|
+
});
|
|
135
|
+
return new aws_apigatewayv2_authorizers_1.HttpLambdaAuthorizer("AuthStackLambdaHttpAuthorizer", lambdaAuthorizer, {
|
|
136
|
+
authorizerName: "AuthStackHTTPLambdaAuthorizer",
|
|
137
|
+
responseTypes: [aws_apigatewayv2_authorizers_1.HttpLambdaResponseType.SIMPLE],
|
|
138
|
+
});
|
|
139
|
+
}
|
|
140
|
+
}
|
|
141
|
+
exports.OrcaBusApiGateway = OrcaBusApiGateway;
|
|
142
|
+
_a = JSII_RTTI_SYMBOL_1;
|
|
143
|
+
OrcaBusApiGateway[_a] = { fqn: "@orcabus/platform-cdk-constructs.apigateway.OrcaBusApiGateway", version: "0.0.4" };
|
|
144
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"api-gateway.js","sourceRoot":"","sources":["api-gateway.ts"],"names":[],"mappings":"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","sourcesContent":["import { Construct } from \"constructs\";\nimport { aws_ssm, Duration, RemovalPolicy, Stack } from \"aws-cdk-lib\";\nimport {\n  HttpJwtAuthorizer,\n  HttpLambdaAuthorizer,\n  HttpLambdaResponseType,\n} from \"aws-cdk-lib/aws-apigatewayv2-authorizers\";\nimport {\n  CfnStage,\n  CorsHttpMethod,\n  DomainName,\n  HttpApi,\n} from \"aws-cdk-lib/aws-apigatewayv2\";\nimport { Certificate } from \"aws-cdk-lib/aws-certificatemanager\";\nimport { IStringParameter, StringParameter } from \"aws-cdk-lib/aws-ssm\";\nimport { LogGroup, RetentionDays } from \"aws-cdk-lib/aws-logs\";\nimport { ARecord, HostedZone, RecordTarget } from \"aws-cdk-lib/aws-route53\";\nimport { Role, ServicePrincipal } from \"aws-cdk-lib/aws-iam\";\nimport { Function } from \"aws-cdk-lib/aws-lambda\";\nimport { ApiGatewayv2DomainProperties } from \"aws-cdk-lib/aws-route53-targets\";\nimport {\n  DEFAULT_COGNITO_CLIENT_ID_PARAMETER_NAME_ARRAY,\n  DEFAULT_COGNITO_USER_POOL_ID_PARAMETER_NAME,\n} from \"./config\";\n\nexport interface ApiGwLogsConfig {\n  /**\n   * The number of days log events are kept in CloudWatch Logs.\n   */\n  readonly retention: RetentionDays;\n  /**\n   * The removal policy to apply to the log group.\n   */\n  readonly removalPolicy: RemovalPolicy;\n}\n\nexport interface OrcaBusApiGatewayProps {\n  /**\n   * The name of the API.\n   */\n  readonly apiName: string;\n  /**\n   * The prefix for the custom domain name\n   */\n  readonly customDomainNamePrefix: string;\n  /**\n   *The cognito user pool id parameter name.\n\n   @default DEFAULT_COGNITO_USER_POOL_ID_PARAMETER_NAME\n   */\n  readonly cognitoUserPoolIdParameterName?: string;\n  /**\n   * The parameter name for the cognito client id in array.\n   * In order API Gateway to validate the JWT token, it needs to know the client id which usually\n   * stored in SSM Parameter. This will accept multiple parameter name in an array.\n   *\n   * @default DEFAULT_COGNITO_CLIENT_ID_PARAMETER_NAME_ARRAY\n   */\n  readonly cognitoClientIdParameterNameArray?: string[];\n  /**\n   * The configuration for aws cloudwatch logs\n   */\n  readonly apiGwLogsConfig: ApiGwLogsConfig;\n  /**\n   * Allowed CORS origins.\n   */\n  readonly corsAllowOrigins: string[];\n}\n\nexport class OrcaBusApiGateway extends Construct {\n  /**\n   * The AWS region where the API Gateway is deployed.\n   */\n  readonly region: string;\n  /**\n   * The HTTP API\n   */\n  readonly httpApi: HttpApi;\n  /**\n   * Domain name defined in this gateway\n   */\n  readonly domainName: string;\n  /**\n   * The Lambda HTTP Authorizer used to enforce authorization policies\n   * defined in the authorization stack.\n   *\n   * To use this, set it as the `authorizer` property in an `HttpRoute` construct.\n   * Example: `authorizer: apiGateway.authStackHttpLambdaAuthorizer`\n   */\n  readonly authStackHttpLambdaAuthorizer: HttpLambdaAuthorizer;\n\n  constructor(scope: Construct, id: string, props: OrcaBusApiGatewayProps) {\n    super(scope, id);\n\n    this.region = Stack.of(this).region;\n\n    // umccr acm arn\n    const umccrAcmArn = StringParameter.valueForStringParameter(\n      this,\n      \"/umccr/certificate_arn\",\n    );\n    const hostedDomainName = StringParameter.valueForStringParameter(\n      this,\n      \"/hosted_zone/umccr/name\",\n    );\n    const hostedZoneId = StringParameter.valueForStringParameter(\n      this,\n      \"/hosted_zone/umccr/id\",\n    );\n\n    this.domainName = `${props.customDomainNamePrefix}.${hostedDomainName}`;\n    const apiGWDomainName = new DomainName(this, \"UmccrDomainName\", {\n      domainName: this.domainName,\n      certificate: Certificate.fromCertificateArn(this, \"cert\", umccrAcmArn),\n    });\n\n    this.httpApi = new HttpApi(this, \"HttpApi\", {\n      apiName: \"OrcaBusAPI-\" + props.apiName,\n      corsPreflight: {\n        allowHeaders: [\n          \"content-type\",\n          \"content-disposition\",\n          \"authorization\",\n          \"x-amz-date\",\n          \"x-api-key\",\n          \"x-amz-security-token\",\n          \"x-amz-user-agent\",\n        ],\n        allowMethods: [\n          CorsHttpMethod.GET,\n          CorsHttpMethod.HEAD,\n          CorsHttpMethod.OPTIONS,\n          CorsHttpMethod.POST,\n          CorsHttpMethod.PATCH,\n          CorsHttpMethod.DELETE,\n        ],\n        allowOrigins: props.corsAllowOrigins,\n        maxAge: Duration.days(10),\n      },\n      defaultAuthorizer: this.getJWTAuthorizer(props),\n      defaultDomainMapping: {\n        domainName: apiGWDomainName,\n      },\n    });\n\n    this.authStackHttpLambdaAuthorizer =\n      this.getAuthStackHTTPLambdaAuthorizer();\n\n    new ARecord(this, \"CustomDomainARecord\", {\n      zone: HostedZone.fromHostedZoneAttributes(this, \"UmccrHostedZone\", {\n        hostedZoneId,\n        zoneName: hostedDomainName,\n      }),\n      recordName: this.domainName,\n      target: RecordTarget.fromAlias(\n        new ApiGatewayv2DomainProperties(\n          apiGWDomainName.regionalDomainName,\n          apiGWDomainName.regionalHostedZoneId,\n        ),\n      ),\n    });\n\n    // LogGroups\n    this.setupAccessLogs(props.apiGwLogsConfig);\n\n    // CloudMap\n    // this.setupCloudServiceDiscovery()\n  }\n\n  // TODO: https://github.com/aws-samples/aws-cdk-service-discovery-example/tree/main\n  // private setupCloudServiceDiscovery() {\n  // }\n\n  // TODO: Taken from https://github.com/aws/aws-cdk/issues/11100#issuecomment-904627081\n  // Monitor for higher level CDK construct instead of leveraging CfnStage\n  private setupAccessLogs(props: ApiGwLogsConfig) {\n    const accessLogs = new LogGroup(this, \"ApiGwAccessLogs\", {\n      retention: props.retention,\n      removalPolicy: props.removalPolicy,\n    });\n    const stage = this.httpApi.defaultStage?.node.defaultChild as CfnStage;\n    stage.accessLogSettings = {\n      destinationArn: accessLogs.logGroupArn,\n      format: JSON.stringify({\n        requestId: \"$context.requestId\",\n        userAgent: \"$context.identity.userAgent\",\n        sourceIp: \"$context.identity.sourceIp\",\n        requestTime: \"$context.requestTime\",\n        requestTimeEpoch: \"$context.requestTimeEpoch\",\n        httpMethod: \"$context.httpMethod\",\n        path: \"$context.path\",\n        status: \"$context.status\",\n        protocol: \"$context.protocol\",\n        responseLength: \"$context.responseLength\",\n        domainName: \"$context.domainName\",\n      }),\n    };\n\n    // Allow writing access logs, managed\n    const role = new Role(this, \"AmazonAPIGatewayPushToCloudWatchLogs\", {\n      assumedBy: new ServicePrincipal(\"apigateway.amazonaws.com\"),\n    });\n\n    accessLogs.grantWrite(role);\n  }\n\n  private getJWTAuthorizer({\n    cognitoUserPoolIdParameterName = DEFAULT_COGNITO_USER_POOL_ID_PARAMETER_NAME,\n    cognitoClientIdParameterNameArray = DEFAULT_COGNITO_CLIENT_ID_PARAMETER_NAME_ARRAY,\n  }: OrcaBusApiGatewayProps): HttpJwtAuthorizer {\n    /**\n     * FIXME One fine day in future when we have proper Cognito AAI setup.\n     *  For the moment, we leverage Portal and established Cognito infrastructure.\n     *  See https://github.com/umccr/orcabus/issues/102\n     */\n\n    const userPoolIdParam: IStringParameter =\n      StringParameter.fromStringParameterName(\n        this,\n        \"CognitoUserPoolIdParameter\",\n        cognitoUserPoolIdParameterName,\n      );\n\n    const clientIdParamsArray: IStringParameter[] =\n      cognitoClientIdParameterNameArray.map((name) =>\n        aws_ssm.StringParameter.fromStringParameterName(\n          this,\n          `CognitoClientId${name}Parameter`,\n          name,\n        ),\n      );\n\n    const issuer =\n      \"https://cognito-idp.\" +\n      this.region +\n      \".amazonaws.com/\" +\n      userPoolIdParam.stringValue;\n\n    return new HttpJwtAuthorizer(\"PortalAuthorizer\", issuer, {\n      jwtAudience: clientIdParamsArray.map((param) => param.stringValue),\n    });\n  }\n\n  /**\n   * Get the HTTP Lambda Authorizer defined in the authorization stack manager\n   * @param authStackHttpLambdaAuthorizerParameterName The SSM Parameter Name that stores the ARN of the lambda authorizer\n   * @returns\n   */\n  private getAuthStackHTTPLambdaAuthorizer() {\n    const lambdaArn = StringParameter.valueForStringParameter(\n      this,\n      \"/orcabus/authorization-stack/http-lambda-authorization-arn\",\n    );\n\n    // Get the lambda HTTP authorizer defined in the authorization stack manager\n    const lambdaAuthorizer = Function.fromFunctionAttributes(\n      this,\n      \"AuthStackHTTPLambdaAuthorizer\",\n      {\n        functionArn: lambdaArn,\n        sameEnvironment: true,\n      },\n    );\n\n    return new HttpLambdaAuthorizer(\n      \"AuthStackLambdaHttpAuthorizer\",\n      lambdaAuthorizer,\n      {\n        authorizerName: \"AuthStackHTTPLambdaAuthorizer\",\n        responseTypes: [HttpLambdaResponseType.SIMPLE],\n      },\n    );\n  }\n}\n"]}
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
import { RetentionDays } from "aws-cdk-lib/aws-logs";
|
|
2
|
+
import { RemovalPolicy } from "aws-cdk-lib";
|
|
3
|
+
import { StageName } from "../utils";
|
|
4
|
+
export declare const DEFAULT_LOGS_CONFIG: {
|
|
5
|
+
BETA: {
|
|
6
|
+
retention: RetentionDays;
|
|
7
|
+
removalPolicy: RemovalPolicy;
|
|
8
|
+
};
|
|
9
|
+
GAMMA: {
|
|
10
|
+
retention: RetentionDays;
|
|
11
|
+
removalPolicy: RemovalPolicy;
|
|
12
|
+
};
|
|
13
|
+
PROD: {
|
|
14
|
+
retention: RetentionDays;
|
|
15
|
+
removalPolicy: RemovalPolicy;
|
|
16
|
+
};
|
|
17
|
+
};
|
|
18
|
+
export declare const DEFAULT_ALLOW_CORS_ORIGINS: {
|
|
19
|
+
BETA: string[];
|
|
20
|
+
GAMMA: string[];
|
|
21
|
+
PROD: string[];
|
|
22
|
+
};
|
|
23
|
+
export declare const DEFAULT_COGNITO_CLIENT_ID_PARAMETER_NAME_ARRAY: string[];
|
|
24
|
+
export declare const DEFAULT_COGNITO_USER_POOL_ID_PARAMETER_NAME = "/data_portal/client/cog_user_pool_id";
|
|
25
|
+
export declare const getDefaultApiGatewayConfiguration: (stage: StageName) => {
|
|
26
|
+
cognitoClientIdParameterNameArray: string[];
|
|
27
|
+
corsAllowOrigins: string[];
|
|
28
|
+
apiGwLogsConfig: {
|
|
29
|
+
retention: RetentionDays;
|
|
30
|
+
removalPolicy: RemovalPolicy;
|
|
31
|
+
} | {
|
|
32
|
+
retention: RetentionDays;
|
|
33
|
+
removalPolicy: RemovalPolicy;
|
|
34
|
+
} | {
|
|
35
|
+
retention: RetentionDays;
|
|
36
|
+
removalPolicy: RemovalPolicy;
|
|
37
|
+
};
|
|
38
|
+
};
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.getDefaultApiGatewayConfiguration = exports.DEFAULT_COGNITO_USER_POOL_ID_PARAMETER_NAME = exports.DEFAULT_COGNITO_CLIENT_ID_PARAMETER_NAME_ARRAY = exports.DEFAULT_ALLOW_CORS_ORIGINS = exports.DEFAULT_LOGS_CONFIG = void 0;
|
|
4
|
+
const aws_logs_1 = require("aws-cdk-lib/aws-logs");
|
|
5
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
6
|
+
exports.DEFAULT_LOGS_CONFIG = {
|
|
7
|
+
BETA: {
|
|
8
|
+
retention: aws_logs_1.RetentionDays.TWO_WEEKS,
|
|
9
|
+
removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY,
|
|
10
|
+
},
|
|
11
|
+
GAMMA: {
|
|
12
|
+
retention: aws_logs_1.RetentionDays.TWO_WEEKS,
|
|
13
|
+
removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY,
|
|
14
|
+
},
|
|
15
|
+
PROD: {
|
|
16
|
+
retention: aws_logs_1.RetentionDays.TWO_WEEKS,
|
|
17
|
+
removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY,
|
|
18
|
+
},
|
|
19
|
+
};
|
|
20
|
+
exports.DEFAULT_ALLOW_CORS_ORIGINS = {
|
|
21
|
+
BETA: ["https://orcaui.dev.umccr.org"],
|
|
22
|
+
GAMMA: ["https://orcaui.stg.umccr.org"],
|
|
23
|
+
PROD: ["https://orcaui.prod.umccr.org", "https://orcaui.umccr.org"],
|
|
24
|
+
};
|
|
25
|
+
exports.DEFAULT_COGNITO_CLIENT_ID_PARAMETER_NAME_ARRAY = [
|
|
26
|
+
"/data_portal/client/data2/cog_app_client_id_stage", // portal - TokenServiceStack
|
|
27
|
+
"/orcaui/cog_app_client_id_stage", // orcaui - https://github.com/umccr/orca-ui
|
|
28
|
+
];
|
|
29
|
+
exports.DEFAULT_COGNITO_USER_POOL_ID_PARAMETER_NAME = "/data_portal/client/cog_user_pool_id";
|
|
30
|
+
const getDefaultApiGatewayConfiguration = (stage) => {
|
|
31
|
+
return {
|
|
32
|
+
cognitoClientIdParameterNameArray: exports.DEFAULT_COGNITO_CLIENT_ID_PARAMETER_NAME_ARRAY,
|
|
33
|
+
corsAllowOrigins: exports.DEFAULT_ALLOW_CORS_ORIGINS[stage],
|
|
34
|
+
apiGwLogsConfig: exports.DEFAULT_LOGS_CONFIG[stage],
|
|
35
|
+
};
|
|
36
|
+
};
|
|
37
|
+
exports.getDefaultApiGatewayConfiguration = getDefaultApiGatewayConfiguration;
|
|
38
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uZmlnLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiY29uZmlnLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLG1EQUFxRDtBQUNyRCw2Q0FBNEM7QUFHL0IsUUFBQSxtQkFBbUIsR0FBRztJQUNqQyxJQUFJLEVBQUU7UUFDSixTQUFTLEVBQUUsd0JBQWEsQ0FBQyxTQUFTO1FBQ2xDLGFBQWEsRUFBRSwyQkFBYSxDQUFDLE9BQU87S0FDckM7SUFDRCxLQUFLLEVBQUU7UUFDTCxTQUFTLEVBQUUsd0JBQWEsQ0FBQyxTQUFTO1FBQ2xDLGFBQWEsRUFBRSwyQkFBYSxDQUFDLE9BQU87S0FDckM7SUFDRCxJQUFJLEVBQUU7UUFDSixTQUFTLEVBQUUsd0JBQWEsQ0FBQyxTQUFTO1FBQ2xDLGFBQWEsRUFBRSwyQkFBYSxDQUFDLE9BQU87S0FDckM7Q0FDRixDQUFDO0FBRVcsUUFBQSwwQkFBMEIsR0FBRztJQUN4QyxJQUFJLEVBQUUsQ0FBQyw4QkFBOEIsQ0FBQztJQUN0QyxLQUFLLEVBQUUsQ0FBQyw4QkFBOEIsQ0FBQztJQUN2QyxJQUFJLEVBQUUsQ0FBQywrQkFBK0IsRUFBRSwwQkFBMEIsQ0FBQztDQUNwRSxDQUFDO0FBRVcsUUFBQSw4Q0FBOEMsR0FBRztJQUM1RCxtREFBbUQsRUFBRSw2QkFBNkI7SUFDbEYsaUNBQWlDLEVBQUUsNENBQTRDO0NBQ2hGLENBQUM7QUFFVyxRQUFBLDJDQUEyQyxHQUN0RCxzQ0FBc0MsQ0FBQztBQUVsQyxNQUFNLGlDQUFpQyxHQUFHLENBQUMsS0FBZ0IsRUFBRSxFQUFFO0lBQ3BFLE9BQU87UUFDTCxpQ0FBaUMsRUFDL0Isc0RBQThDO1FBQ2hELGdCQUFnQixFQUFFLGtDQUEwQixDQUFDLEtBQUssQ0FBQztRQUNuRCxlQUFlLEVBQUUsMkJBQW1CLENBQUMsS0FBSyxDQUFDO0tBQzVDLENBQUM7QUFDSixDQUFDLENBQUM7QUFQVyxRQUFBLGlDQUFpQyxxQ0FPNUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBSZXRlbnRpb25EYXlzIH0gZnJvbSBcImF3cy1jZGstbGliL2F3cy1sb2dzXCI7XG5pbXBvcnQgeyBSZW1vdmFsUG9saWN5IH0gZnJvbSBcImF3cy1jZGstbGliXCI7XG5pbXBvcnQgeyBTdGFnZU5hbWUgfSBmcm9tIFwiLi4vdXRpbHNcIjtcblxuZXhwb3J0IGNvbnN0IERFRkFVTFRfTE9HU19DT05GSUcgPSB7XG4gIEJFVEE6IHtcbiAgICByZXRlbnRpb246IFJldGVudGlvbkRheXMuVFdPX1dFRUtTLFxuICAgIHJlbW92YWxQb2xpY3k6IFJlbW92YWxQb2xpY3kuREVTVFJPWSxcbiAgfSxcbiAgR0FNTUE6IHtcbiAgICByZXRlbnRpb246IFJldGVudGlvbkRheXMuVFdPX1dFRUtTLFxuICAgIHJlbW92YWxQb2xpY3k6IFJlbW92YWxQb2xpY3kuREVTVFJPWSxcbiAgfSxcbiAgUFJPRDoge1xuICAgIHJldGVudGlvbjogUmV0ZW50aW9uRGF5cy5UV09fV0VFS1MsXG4gICAgcmVtb3ZhbFBvbGljeTogUmVtb3ZhbFBvbGljeS5ERVNUUk9ZLFxuICB9LFxufTtcblxuZXhwb3J0IGNvbnN0IERFRkFVTFRfQUxMT1dfQ09SU19PUklHSU5TID0ge1xuICBCRVRBOiBbXCJodHRwczovL29yY2F1aS5kZXYudW1jY3Iub3JnXCJdLFxuICBHQU1NQTogW1wiaHR0cHM6Ly9vcmNhdWkuc3RnLnVtY2NyLm9yZ1wiXSxcbiAgUFJPRDogW1wiaHR0cHM6Ly9vcmNhdWkucHJvZC51bWNjci5vcmdcIiwgXCJodHRwczovL29yY2F1aS51bWNjci5vcmdcIl0sXG59O1xuXG5leHBvcnQgY29uc3QgREVGQVVMVF9DT0dOSVRPX0NMSUVOVF9JRF9QQVJBTUVURVJfTkFNRV9BUlJBWSA9IFtcbiAgXCIvZGF0YV9wb3J0YWwvY2xpZW50L2RhdGEyL2NvZ19hcHBfY2xpZW50X2lkX3N0YWdlXCIsIC8vIHBvcnRhbCAtIFRva2VuU2VydmljZVN0YWNrXG4gIFwiL29yY2F1aS9jb2dfYXBwX2NsaWVudF9pZF9zdGFnZVwiLCAvLyBvcmNhdWkgLSBodHRwczovL2dpdGh1Yi5jb20vdW1jY3Ivb3JjYS11aVxuXTtcblxuZXhwb3J0IGNvbnN0IERFRkFVTFRfQ09HTklUT19VU0VSX1BPT0xfSURfUEFSQU1FVEVSX05BTUUgPVxuICBcIi9kYXRhX3BvcnRhbC9jbGllbnQvY29nX3VzZXJfcG9vbF9pZFwiO1xuXG5leHBvcnQgY29uc3QgZ2V0RGVmYXVsdEFwaUdhdGV3YXlDb25maWd1cmF0aW9uID0gKHN0YWdlOiBTdGFnZU5hbWUpID0+IHtcbiAgcmV0dXJuIHtcbiAgICBjb2duaXRvQ2xpZW50SWRQYXJhbWV0ZXJOYW1lQXJyYXk6XG4gICAgICBERUZBVUxUX0NPR05JVE9fQ0xJRU5UX0lEX1BBUkFNRVRFUl9OQU1FX0FSUkFZLFxuICAgIGNvcnNBbGxvd09yaWdpbnM6IERFRkFVTFRfQUxMT1dfQ09SU19PUklHSU5TW3N0YWdlXSxcbiAgICBhcGlHd0xvZ3NDb25maWc6IERFRkFVTFRfTE9HU19DT05GSUdbc3RhZ2VdLFxuICB9O1xufTtcbiJdfQ==
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
14
|
+
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
15
|
+
};
|
|
16
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
17
|
+
__exportStar(require("./config"), exports);
|
|
18
|
+
__exportStar(require("./api-gateway"), exports);
|
|
19
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7O0FBQUEsMkNBQXlCO0FBQ3pCLGdEQUE4QiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gXCIuL2NvbmZpZ1wiO1xuZXhwb3J0ICogZnJvbSBcIi4vYXBpLWdhdGV3YXlcIjtcbiJdfQ==
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
# DeploymentStackPipeline
|
|
2
|
+
|
|
3
|
+
This construct will deploy stack defined as props across 3 stages in our Beta, Gamma, and Prod environment.
|
|
4
|
+
|
|
5
|
+
Usage example
|
|
6
|
+
|
|
7
|
+
```ts
|
|
8
|
+
new DeploymentStackPipeline(this, "DeploymentPipeline", {
|
|
9
|
+
githubBranch: "main",
|
|
10
|
+
githubRepo: "platform-cdk-constructs",
|
|
11
|
+
stack: DeploymentStack,
|
|
12
|
+
stackName: "TestStack",
|
|
13
|
+
stackConfig: {
|
|
14
|
+
beta: { bucketName: "my-simple-bucket-dev" },
|
|
15
|
+
gamma: { bucketName: "my-simple-bucket-stg" },
|
|
16
|
+
prod: { bucketName: "my-simple-bucket-prod" },
|
|
17
|
+
},
|
|
18
|
+
pipelineName: "TestDeploymentPipeline",
|
|
19
|
+
cdkSynthCmd: [
|
|
20
|
+
"pnpm install --frozen-lockfile --ignore-scripts",
|
|
21
|
+
"cd dev",
|
|
22
|
+
"pnpm cdk synth",
|
|
23
|
+
],
|
|
24
|
+
cdkOut: "dev/cdk.out",
|
|
25
|
+
});
|
|
26
|
+
|
|
27
|
+
```
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.PROD_ENVIRONMENT = exports.GAMMA_ENVIRONMENT = exports.BETA_ENVIRONMENT = exports.TOOLCHAIN_ENVIRONMENT = void 0;
|
|
4
|
+
exports.TOOLCHAIN_ENVIRONMENT = {
|
|
5
|
+
account: "383856791668",
|
|
6
|
+
region: "ap-southeast-2",
|
|
7
|
+
};
|
|
8
|
+
exports.BETA_ENVIRONMENT = {
|
|
9
|
+
account: "843407916570",
|
|
10
|
+
region: "ap-southeast-2",
|
|
11
|
+
};
|
|
12
|
+
exports.GAMMA_ENVIRONMENT = {
|
|
13
|
+
account: "455634345446",
|
|
14
|
+
region: "ap-southeast-2",
|
|
15
|
+
};
|
|
16
|
+
exports.PROD_ENVIRONMENT = {
|
|
17
|
+
account: "472057503814",
|
|
18
|
+
region: "ap-southeast-2",
|
|
19
|
+
};
|
|
20
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uZmlnLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiY29uZmlnLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUVhLFFBQUEscUJBQXFCLEdBQWdCO0lBQ2hELE9BQU8sRUFBRSxjQUFjO0lBQ3ZCLE1BQU0sRUFBRSxnQkFBZ0I7Q0FDekIsQ0FBQztBQUVXLFFBQUEsZ0JBQWdCLEdBQWdCO0lBQzNDLE9BQU8sRUFBRSxjQUFjO0lBQ3ZCLE1BQU0sRUFBRSxnQkFBZ0I7Q0FDekIsQ0FBQztBQUVXLFFBQUEsaUJBQWlCLEdBQWdCO0lBQzVDLE9BQU8sRUFBRSxjQUFjO0lBQ3ZCLE1BQU0sRUFBRSxnQkFBZ0I7Q0FDekIsQ0FBQztBQUVXLFFBQUEsZ0JBQWdCLEdBQWdCO0lBQzNDLE9BQU8sRUFBRSxjQUFjO0lBQ3ZCLE1BQU0sRUFBRSxnQkFBZ0I7Q0FDekIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEVudmlyb25tZW50IH0gZnJvbSBcImF3cy1jZGstbGliXCI7XG5cbmV4cG9ydCBjb25zdCBUT09MQ0hBSU5fRU5WSVJPTk1FTlQ6IEVudmlyb25tZW50ID0ge1xuICBhY2NvdW50OiBcIjM4Mzg1Njc5MTY2OFwiLFxuICByZWdpb246IFwiYXAtc291dGhlYXN0LTJcIixcbn07XG5cbmV4cG9ydCBjb25zdCBCRVRBX0VOVklST05NRU5UOiBFbnZpcm9ubWVudCA9IHtcbiAgYWNjb3VudDogXCI4NDM0MDc5MTY1NzBcIixcbiAgcmVnaW9uOiBcImFwLXNvdXRoZWFzdC0yXCIsXG59O1xuXG5leHBvcnQgY29uc3QgR0FNTUFfRU5WSVJPTk1FTlQ6IEVudmlyb25tZW50ID0ge1xuICBhY2NvdW50OiBcIjQ1NTYzNDM0NTQ0NlwiLFxuICByZWdpb246IFwiYXAtc291dGhlYXN0LTJcIixcbn07XG5cbmV4cG9ydCBjb25zdCBQUk9EX0VOVklST05NRU5UOiBFbnZpcm9ubWVudCA9IHtcbiAgYWNjb3VudDogXCI0NzIwNTc1MDM4MTRcIixcbiAgcmVnaW9uOiBcImFwLXNvdXRoZWFzdC0yXCIsXG59O1xuIl19
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
14
|
+
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
15
|
+
};
|
|
16
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
17
|
+
__exportStar(require("./config"), exports);
|
|
18
|
+
__exportStar(require("./pipeline"), exports);
|
|
19
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7O0FBQUEsMkNBQXlCO0FBQ3pCLDZDQUEyQiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gXCIuL2NvbmZpZ1wiO1xuZXhwb3J0ICogZnJvbSBcIi4vcGlwZWxpbmVcIjtcbiJdfQ==
|
|
@@ -0,0 +1,104 @@
|
|
|
1
|
+
import { Construct } from "constructs";
|
|
2
|
+
import { Environment } from "aws-cdk-lib";
|
|
3
|
+
import { Pipeline } from "aws-cdk-lib/aws-codepipeline";
|
|
4
|
+
/**
|
|
5
|
+
* The default partial build spec for the synth step in the pipeline.
|
|
6
|
+
*/
|
|
7
|
+
export declare const DEFAULT_SYNTH_STEP_PARTIAL_BUILD_SPEC: {
|
|
8
|
+
phases: {
|
|
9
|
+
install: {
|
|
10
|
+
"runtime-versions": {
|
|
11
|
+
nodejs: string;
|
|
12
|
+
};
|
|
13
|
+
};
|
|
14
|
+
};
|
|
15
|
+
};
|
|
16
|
+
export interface StageEnvProps {
|
|
17
|
+
/**
|
|
18
|
+
* The environment for the beta stage
|
|
19
|
+
*/
|
|
20
|
+
readonly beta: Environment;
|
|
21
|
+
/**
|
|
22
|
+
* The environment for the gamma stage
|
|
23
|
+
*/
|
|
24
|
+
readonly gamma: Environment;
|
|
25
|
+
/**
|
|
26
|
+
* The environment for the prod stage
|
|
27
|
+
*/
|
|
28
|
+
readonly prod: Environment;
|
|
29
|
+
}
|
|
30
|
+
export interface StackConfigProps {
|
|
31
|
+
/**
|
|
32
|
+
* The configuration for the beta (dev) stage
|
|
33
|
+
*/
|
|
34
|
+
readonly beta: Record<string, any>;
|
|
35
|
+
/**
|
|
36
|
+
* The configuration for the gamma (stg) stage
|
|
37
|
+
*/
|
|
38
|
+
readonly gamma: Record<string, any>;
|
|
39
|
+
/**
|
|
40
|
+
* The configuration for the prod stage
|
|
41
|
+
*/
|
|
42
|
+
readonly prod: Record<string, any>;
|
|
43
|
+
}
|
|
44
|
+
export interface DeploymentStackPipelineProps {
|
|
45
|
+
/**
|
|
46
|
+
* The github branch name it will listen to.
|
|
47
|
+
*/
|
|
48
|
+
readonly githubBranch: string;
|
|
49
|
+
/**
|
|
50
|
+
* The repository name that exist in the 'OrcaBus' github organisation. e.g. `a-micro-service-repo`
|
|
51
|
+
*/
|
|
52
|
+
readonly githubRepo: string;
|
|
53
|
+
/**
|
|
54
|
+
* The stack to which the pipeline will be deploying to its respective account
|
|
55
|
+
*/
|
|
56
|
+
readonly stack: any;
|
|
57
|
+
/**
|
|
58
|
+
* The stack name (in cloudformation) for the stack defined in `stack`. The stack name will prepend with the stage
|
|
59
|
+
* name e.g. `OrcaBusBeta-<stackName>`, `OrcaBusGamma-<stackName>`, `OrcaBusProd-<stackName>`
|
|
60
|
+
*/
|
|
61
|
+
readonly stackName: string;
|
|
62
|
+
/**
|
|
63
|
+
* The stack configuration/constants that will be passed to the stack props.
|
|
64
|
+
*/
|
|
65
|
+
readonly stackConfig: StackConfigProps;
|
|
66
|
+
/**
|
|
67
|
+
* The pipeline name in the bastion account.
|
|
68
|
+
*/
|
|
69
|
+
readonly pipelineName: string;
|
|
70
|
+
/**
|
|
71
|
+
* The file paths to trigger the pipeline. e.g. ["stateless/**"]
|
|
72
|
+
*
|
|
73
|
+
* https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/aws-properties-codepipeline-pipeline-gitfilepathfiltercriteria.html
|
|
74
|
+
*
|
|
75
|
+
*/
|
|
76
|
+
readonly filePaths?: string[];
|
|
77
|
+
/**
|
|
78
|
+
* The command to run to synth the cdk stack which also installing the cdk dependencies. e.g. ["yarn install --immutable", "yarn cdk synth"]
|
|
79
|
+
*/
|
|
80
|
+
readonly cdkSynthCmd: string[];
|
|
81
|
+
/**
|
|
82
|
+
* The location where the cdk output will be stored.
|
|
83
|
+
*
|
|
84
|
+
* @default cdk.out
|
|
85
|
+
*/
|
|
86
|
+
readonly cdkOut?: string;
|
|
87
|
+
/**
|
|
88
|
+
* Additional configuration for the CodeBuild step during the CDK synth phase. It will passed as the `partialBuildSpec` to the `CodeBuildStep`.
|
|
89
|
+
*
|
|
90
|
+
* @default DEFAULT_SYNTH_STEP_PARTIAL_BUILD_SPEC
|
|
91
|
+
*/
|
|
92
|
+
readonly synthBuildSpec?: Record<string, any>;
|
|
93
|
+
/**
|
|
94
|
+
* The stage environment for the deployment stack
|
|
95
|
+
*/
|
|
96
|
+
readonly stageEnv?: StageEnvProps;
|
|
97
|
+
}
|
|
98
|
+
export declare class DeploymentStackPipeline extends Construct {
|
|
99
|
+
/**
|
|
100
|
+
* The code pipeline construct that is created.
|
|
101
|
+
*/
|
|
102
|
+
readonly pipeline: Pipeline;
|
|
103
|
+
constructor(scope: Construct, id: string, props: DeploymentStackPipelineProps);
|
|
104
|
+
}
|