@restatedev/restate-cdk 0.7.0 → 0.8.0

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.
@@ -0,0 +1,4 @@
1
+ export declare enum TracingMode {
2
+ DISABLED = "DISABLED",
3
+ AWS_XRAY = "AWS_XRAY"
4
+ }
@@ -0,0 +1,9 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.TracingMode = void 0;
4
+ var TracingMode;
5
+ (function (TracingMode) {
6
+ TracingMode["DISABLED"] = "DISABLED";
7
+ TracingMode["AWS_XRAY"] = "AWS_XRAY";
8
+ })(TracingMode || (exports.TracingMode = TracingMode = {}));
9
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGVwbG95bWVudHMtY29tbW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vbGliL3Jlc3RhdGUtY29uc3RydWN0cy9kZXBsb3ltZW50cy1jb21tb24udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsSUFBWSxXQUdYO0FBSEQsV0FBWSxXQUFXO0lBQ3JCLG9DQUFxQixDQUFBO0lBQ3JCLG9DQUFxQixDQUFBO0FBQ3ZCLENBQUMsRUFIVyxXQUFXLDJCQUFYLFdBQVcsUUFHdEIiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgZW51bSBUcmFjaW5nTW9kZSB7XG4gIERJU0FCTEVEID0gXCJESVNBQkxFRFwiLFxuICBBV1NfWFJBWSA9IFwiQVdTX1hSQVlcIixcbn1cbiJdfQ==
@@ -0,0 +1,77 @@
1
+ import { Construct } from "constructs";
2
+ import * as cdk from "aws-cdk-lib";
3
+ import * as logs from "aws-cdk-lib/aws-logs";
4
+ import * as ec2 from "aws-cdk-lib/aws-ec2";
5
+ import * as iam from "aws-cdk-lib/aws-iam";
6
+ import * as efs from "aws-cdk-lib/aws-efs";
7
+ import * as elb2 from "aws-cdk-lib/aws-elasticloadbalancingv2";
8
+ import * as r53 from "aws-cdk-lib/aws-route53";
9
+ import { IRestateEnvironment } from "./restate-environment";
10
+ import { TracingMode } from "./deployments-common";
11
+ export interface RestateFargateProps {
12
+ /** The VPC in which to launch the Restate host. */
13
+ vpc?: ec2.IVpc;
14
+ /** Log group for Restate service logs. */
15
+ logGroup?: logs.LogGroup;
16
+ /** Tracing mode for Restate services. Defaults to {@link TracingMode.DISABLED}. */
17
+ tracing?: TracingMode;
18
+ /** Prefix for resources created by this construct that require unique names. */
19
+ prefix?: string;
20
+ /** ECS cluster name. */
21
+ clusterName?: string;
22
+ /** Restate Docker image name. Defaults to `latest`. */
23
+ restateImage?: string;
24
+ /** Restate Docker image tag. Defaults to `latest`. */
25
+ restateTag?: string;
26
+ /** Amazon Distro for Open Telemetry Docker image tag. Defaults to `latest`. */
27
+ adotTag?: string;
28
+ /**
29
+ * Environment for Restate container. Use it to configure logging and other process-level settings.
30
+ */
31
+ environment?: Record<string, string>;
32
+ /**
33
+ * Restate container extra arguments.
34
+ */
35
+ command?: string[];
36
+ /**
37
+ * The full name for the public endpoint.
38
+ */
39
+ dnsName: string;
40
+ /**
41
+ * DNS zone in which to create the public endpoint.
42
+ */
43
+ hostedZone: r53.IHostedZone;
44
+ /**
45
+ * Removal policy for long-lived resources (storage, logs). Default: `cdk.RemovalPolicy.DESTROY`.
46
+ */
47
+ removalPolicy?: cdk.RemovalPolicy;
48
+ /**
49
+ * Load balancer configuration.
50
+ */
51
+ loadBalancer?: {
52
+ /** @see BaseLoadBalancerProps.internetFacing */
53
+ internetFacing?: boolean;
54
+ /**
55
+ * If you set this to false, you can customize the access to the pair of ALB listeners via
56
+ * {@link FargateRestateDeployment.ingressListener} and {@link FargateRestateDeployment.adminListener}.
57
+ *
58
+ * @see BaseApplicationListenerProps.open */
59
+ open?: boolean;
60
+ };
61
+ }
62
+ /**
63
+ * Creates a Restate service deployment running as a Fargate task and backed by EFS.
64
+ *
65
+ * Please note that this construct is still experimental! Use with caution.
66
+ */
67
+ export declare class FargateRestateDeployment extends Construct implements IRestateEnvironment {
68
+ readonly invokerRole: iam.IRole;
69
+ readonly vpc: ec2.IVpc;
70
+ readonly ingressUrl: string;
71
+ readonly adminUrl: string;
72
+ readonly securityGroup: ec2.SecurityGroup;
73
+ readonly dataStore: efs.FileSystem;
74
+ readonly ingressListener: elb2.ApplicationListener;
75
+ readonly adminListener: elb2.ApplicationListener;
76
+ constructor(scope: Construct, id: string, props: RestateFargateProps);
77
+ }
@@ -0,0 +1,239 @@
1
+ "use strict";
2
+ /*
3
+ * Copyright (c) 2023 - Restate Software, Inc., Restate GmbH
4
+ *
5
+ * This file is part of the Restate SDK for Node.js/TypeScript,
6
+ * which is released under the MIT license.
7
+ *
8
+ * You can find a copy of the license in file LICENSE in the root
9
+ * directory of this repository or package, or at
10
+ * https://github.com/restatedev/sdk-typescript/blob/main/LICENSE
11
+ */
12
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
13
+ if (k2 === undefined) k2 = k;
14
+ var desc = Object.getOwnPropertyDescriptor(m, k);
15
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
16
+ desc = { enumerable: true, get: function() { return m[k]; } };
17
+ }
18
+ Object.defineProperty(o, k2, desc);
19
+ }) : (function(o, m, k, k2) {
20
+ if (k2 === undefined) k2 = k;
21
+ o[k2] = m[k];
22
+ }));
23
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
24
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
25
+ }) : function(o, v) {
26
+ o["default"] = v;
27
+ });
28
+ var __importStar = (this && this.__importStar) || function (mod) {
29
+ if (mod && mod.__esModule) return mod;
30
+ var result = {};
31
+ if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
32
+ __setModuleDefault(result, mod);
33
+ return result;
34
+ };
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.FargateRestateDeployment = void 0;
37
+ const constructs_1 = require("constructs");
38
+ const cdk = __importStar(require("aws-cdk-lib"));
39
+ const acm = __importStar(require("aws-cdk-lib/aws-certificatemanager"));
40
+ const logs = __importStar(require("aws-cdk-lib/aws-logs"));
41
+ const ec2 = __importStar(require("aws-cdk-lib/aws-ec2"));
42
+ const iam = __importStar(require("aws-cdk-lib/aws-iam"));
43
+ const ecs = __importStar(require("aws-cdk-lib/aws-ecs"));
44
+ const efs = __importStar(require("aws-cdk-lib/aws-efs"));
45
+ const elb2 = __importStar(require("aws-cdk-lib/aws-elasticloadbalancingv2"));
46
+ const r53 = __importStar(require("aws-cdk-lib/aws-route53"));
47
+ const targets = __importStar(require("aws-cdk-lib/aws-route53-targets"));
48
+ const PUBLIC_INGRESS_PORT = 443;
49
+ const PUBLIC_ADMIN_PORT = 9070;
50
+ const RESTATE_INGRESS_PORT = 8080;
51
+ const RESTATE_ADMIN_PORT = 9070;
52
+ const RESTATE_IMAGE_DEFAULT = "docker.io/restatedev/restate";
53
+ const RESTATE_DOCKER_DEFAULT_TAG = "latest";
54
+ const ADOT_DOCKER_DEFAULT_TAG = "latest";
55
+ /**
56
+ * Creates a Restate service deployment running as a Fargate task and backed by EFS.
57
+ *
58
+ * Please note that this construct is still experimental! Use with caution.
59
+ */
60
+ class FargateRestateDeployment extends constructs_1.Construct {
61
+ constructor(scope, id, props) {
62
+ super(scope, id);
63
+ this.vpc = props.vpc ?? ec2.Vpc.fromLookup(this, "Vpc", { isDefault: true });
64
+ const restateImage = props.restateImage ?? RESTATE_IMAGE_DEFAULT;
65
+ const restateTag = props.restateTag ?? RESTATE_DOCKER_DEFAULT_TAG;
66
+ const adotTag = props.adotTag ?? ADOT_DOCKER_DEFAULT_TAG; // TODO: add X-Ray support like we have for EC2
67
+ const fs = new efs.FileSystem(this, "DataStore", {
68
+ vpc: this.vpc,
69
+ lifecyclePolicy: efs.LifecyclePolicy.AFTER_30_DAYS,
70
+ performanceMode: efs.PerformanceMode.GENERAL_PURPOSE,
71
+ throughputMode: efs.ThroughputMode.BURSTING,
72
+ removalPolicy: props.removalPolicy ?? cdk.RemovalPolicy.DESTROY,
73
+ });
74
+ fs.addToResourcePolicy(new iam.PolicyStatement({
75
+ sid: "AllowEfsMount",
76
+ actions: ["elasticfilesystem:ClientMount"],
77
+ // Restricting to the ECS execution role does not work; probably doesn't matter - EFS access is secured by a security group
78
+ principals: [new iam.AnyPrincipal()],
79
+ conditions: {
80
+ Bool: {
81
+ "elasticfilesystem:AccessedViaMountTarget": "true",
82
+ },
83
+ },
84
+ }));
85
+ this.dataStore = fs;
86
+ const cluster = new ecs.Cluster(this, "Cluster", {
87
+ vpc: this.vpc,
88
+ clusterName: props.clusterName,
89
+ });
90
+ const restateTask = new ecs.FargateTaskDefinition(this, "RestateTask", {
91
+ cpu: 4096,
92
+ memoryLimitMiB: 8192,
93
+ runtimePlatform: {
94
+ cpuArchitecture: ecs.CpuArchitecture.ARM64,
95
+ operatingSystemFamily: ecs.OperatingSystemFamily.LINUX,
96
+ },
97
+ volumes: [
98
+ {
99
+ name: "restateStore",
100
+ efsVolumeConfiguration: {
101
+ fileSystemId: fs.fileSystemId,
102
+ authorizationConfig: {},
103
+ },
104
+ },
105
+ ],
106
+ });
107
+ // TODO: Start an ADOT container and hook it up to Restate and AWS X-Ray or another OTel sink
108
+ // if (props.tracing === TracingMode.AWS_XRAY) {
109
+ // restateTask.taskRole.addManagedPolicy(iam.ManagedPolicy.fromAwsManagedPolicyName("AWSXrayWriteOnlyAccess"));
110
+ // }
111
+ new iam.Policy(this, "TaskPolicy", {
112
+ statements: [
113
+ new iam.PolicyStatement({
114
+ sid: "AllowAssumeAnyRole",
115
+ actions: ["sts:AssumeRole"],
116
+ resources: ["*"], // we don't know upfront what invoker roles we may be asked to assume at runtime
117
+ }),
118
+ ],
119
+ }).attachToRole(restateTask.taskRole);
120
+ const invokerRole = new iam.Role(this, "InvokerRole", {
121
+ assumedBy: new iam.ArnPrincipal(restateTask.taskRole.roleArn),
122
+ description: "Assumed by Restate deployment to invoke Lambda-based services",
123
+ });
124
+ invokerRole.grantAssumeRole(restateTask.taskRole);
125
+ this.invokerRole = invokerRole;
126
+ const logGroup = props.logGroup ??
127
+ new logs.LogGroup(this, "Logs", {
128
+ logGroupName: `/restate/${id}`,
129
+ retention: logs.RetentionDays.ONE_MONTH,
130
+ removalPolicy: props.removalPolicy ?? cdk.RemovalPolicy.DESTROY,
131
+ });
132
+ const restate = restateTask.addContainer("Restate", {
133
+ containerName: "restate-runtime",
134
+ image: ecs.ContainerImage.fromRegistry(`${restateImage}:${restateTag}`),
135
+ portMappings: [{ containerPort: RESTATE_INGRESS_PORT }, { containerPort: RESTATE_ADMIN_PORT }],
136
+ logging: ecs.LogDriver.awsLogs({
137
+ logGroup,
138
+ streamPrefix: "restate",
139
+ }),
140
+ environment: {
141
+ RESTATE_OBSERVABILITY__LOG__FORMAT: "Json",
142
+ // RUST_LOG: "warn,restate=info",
143
+ },
144
+ command: props.command,
145
+ startTimeout: cdk.Duration.seconds(20),
146
+ stopTimeout: cdk.Duration.seconds(20),
147
+ });
148
+ restate.addMountPoints({
149
+ containerPath: "/target",
150
+ readOnly: false,
151
+ sourceVolume: "restateStore",
152
+ });
153
+ const restateSecurityGroup = new ec2.SecurityGroup(this, "SecurityGroup", {
154
+ vpc: this.vpc,
155
+ allowAllOutbound: true,
156
+ });
157
+ this.securityGroup = restateSecurityGroup;
158
+ const restateFargateService = new ecs.FargateService(this, "Service", {
159
+ cluster,
160
+ taskDefinition: restateTask,
161
+ assignPublicIp: true,
162
+ circuitBreaker: {
163
+ enable: true,
164
+ rollback: true,
165
+ },
166
+ minHealthyPercent: 0, // allow scale down to zero during deployments (required for at-most-1 max setting)
167
+ maxHealthyPercent: 100, // don't start more than one copy
168
+ securityGroups: [restateSecurityGroup],
169
+ });
170
+ fs.connections.allowDefaultPortFrom(restateSecurityGroup);
171
+ fs.connections.allowDefaultPortTo(restateSecurityGroup);
172
+ fs.grantRootAccess(restateFargateService.taskDefinition.taskRole.grantPrincipal);
173
+ const alb = new elb2.ApplicationLoadBalancer(this, "Alb", {
174
+ vpc: this.vpc,
175
+ internetFacing: props.loadBalancer?.internetFacing,
176
+ });
177
+ const publicApiCertificate = new acm.Certificate(this, "Certificate", {
178
+ domainName: props.dnsName,
179
+ validation: acm.CertificateValidation.fromDns(props.hostedZone),
180
+ });
181
+ const ingressListener = alb.addListener("IngressListener", {
182
+ port: PUBLIC_INGRESS_PORT,
183
+ protocol: elb2.ApplicationProtocol.HTTPS,
184
+ certificates: [publicApiCertificate],
185
+ open: props.loadBalancer?.open,
186
+ });
187
+ ingressListener.addTargets("FargateIngressTarget", {
188
+ targets: [
189
+ restateFargateService.loadBalancerTarget({
190
+ containerName: restate.containerName,
191
+ containerPort: RESTATE_INGRESS_PORT,
192
+ }),
193
+ ],
194
+ protocol: elb2.ApplicationProtocol.HTTP,
195
+ healthCheck: {
196
+ path: "/grpc.health.v1.Health/Check",
197
+ interval: cdk.Duration.seconds(5),
198
+ healthyThresholdCount: 3,
199
+ unhealthyThresholdCount: 3,
200
+ timeout: cdk.Duration.seconds(2),
201
+ },
202
+ deregistrationDelay: cdk.Duration.seconds(5),
203
+ });
204
+ this.ingressListener = ingressListener;
205
+ const adminListener = alb.addListener("AdminListener", {
206
+ port: PUBLIC_ADMIN_PORT,
207
+ protocol: elb2.ApplicationProtocol.HTTPS,
208
+ certificates: [publicApiCertificate],
209
+ });
210
+ adminListener.addTargets("FargateAdminTarget", {
211
+ targets: [
212
+ restateFargateService.loadBalancerTarget({
213
+ containerName: restate.containerName,
214
+ containerPort: RESTATE_ADMIN_PORT,
215
+ }),
216
+ ],
217
+ protocol: elb2.ApplicationProtocol.HTTP,
218
+ healthCheck: {
219
+ path: "/health",
220
+ interval: cdk.Duration.seconds(5),
221
+ healthyThresholdCount: 3,
222
+ unhealthyThresholdCount: 3,
223
+ timeout: cdk.Duration.seconds(2),
224
+ },
225
+ deregistrationDelay: cdk.Duration.seconds(5),
226
+ });
227
+ this.adminListener = adminListener;
228
+ new r53.ARecord(this, "AlbAlias", {
229
+ zone: props.hostedZone,
230
+ recordName: props.dnsName.split(".")[0],
231
+ target: r53.RecordTarget.fromAlias(new targets.LoadBalancerTarget(alb)),
232
+ // other ARecord configuration...
233
+ });
234
+ this.ingressUrl = `https://${props.dnsName}${PUBLIC_INGRESS_PORT == 443 ? "" : `:${PUBLIC_INGRESS_PORT}`}`;
235
+ this.adminUrl = `https://${props.dnsName}:${PUBLIC_ADMIN_PORT}`;
236
+ }
237
+ }
238
+ exports.FargateRestateDeployment = FargateRestateDeployment;
239
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fargate-restate-deployment.js","sourceRoot":"","sources":["../lib/restate-constructs/fargate-restate-deployment.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2023 - Restate Software, Inc., Restate GmbH\n *\n * This file is part of the Restate SDK for Node.js/TypeScript,\n * which is released under the MIT license.\n *\n * You can find a copy of the license in file LICENSE in the root\n * directory of this repository or package, or at\n * https://github.com/restatedev/sdk-typescript/blob/main/LICENSE\n */\n\nimport { Construct } from \"constructs\";\nimport * as cdk from \"aws-cdk-lib\";\nimport * as acm from \"aws-cdk-lib/aws-certificatemanager\";\nimport * as logs from \"aws-cdk-lib/aws-logs\";\nimport * as ec2 from \"aws-cdk-lib/aws-ec2\";\nimport * as iam from \"aws-cdk-lib/aws-iam\";\nimport * as ecs from \"aws-cdk-lib/aws-ecs\";\nimport * as efs from \"aws-cdk-lib/aws-efs\";\nimport * as elb2 from \"aws-cdk-lib/aws-elasticloadbalancingv2\";\nimport * as r53 from \"aws-cdk-lib/aws-route53\";\nimport * as targets from \"aws-cdk-lib/aws-route53-targets\";\nimport { IRestateEnvironment } from \"./restate-environment\";\nimport { TracingMode } from \"./deployments-common\";\n\nconst PUBLIC_INGRESS_PORT = 443;\nconst PUBLIC_ADMIN_PORT = 9070;\nconst RESTATE_INGRESS_PORT = 8080;\nconst RESTATE_ADMIN_PORT = 9070;\nconst RESTATE_IMAGE_DEFAULT = \"docker.io/restatedev/restate\";\nconst RESTATE_DOCKER_DEFAULT_TAG = \"latest\";\nconst ADOT_DOCKER_DEFAULT_TAG = \"latest\";\n\nexport interface RestateFargateProps {\n  /** The VPC in which to launch the Restate host. */\n  vpc?: ec2.IVpc;\n\n  /** Log group for Restate service logs. */\n  logGroup?: logs.LogGroup;\n\n  /** Tracing mode for Restate services. Defaults to {@link TracingMode.DISABLED}. */\n  tracing?: TracingMode;\n\n  /** Prefix for resources created by this construct that require unique names. */\n  prefix?: string;\n\n  /** ECS cluster name. */\n  clusterName?: string;\n\n  /** Restate Docker image name. Defaults to `latest`. */\n  restateImage?: string;\n\n  /** Restate Docker image tag. Defaults to `latest`. */\n  restateTag?: string;\n\n  /** Amazon Distro for Open Telemetry Docker image tag. Defaults to `latest`. */\n  adotTag?: string;\n\n  /**\n   * Environment for Restate container. Use it to configure logging and other process-level settings.\n   */\n  environment?: Record<string, string>;\n\n  /**\n   * Restate container extra arguments.\n   */\n  command?: string[];\n\n  /**\n   * The full name for the public endpoint.\n   */\n  dnsName: string;\n\n  /**\n   * DNS zone in which to create the public endpoint.\n   */\n  hostedZone: r53.IHostedZone;\n\n  /**\n   * Removal policy for long-lived resources (storage, logs). Default: `cdk.RemovalPolicy.DESTROY`.\n   */\n  removalPolicy?: cdk.RemovalPolicy;\n\n  /**\n   * Load balancer configuration.\n   */\n  loadBalancer?: {\n    /** @see BaseLoadBalancerProps.internetFacing */\n    internetFacing?: boolean;\n\n    /**\n     * If you set this to false, you can customize the access to the pair of ALB listeners via\n     * {@link FargateRestateDeployment.ingressListener} and {@link FargateRestateDeployment.adminListener}.\n     *\n     * @see BaseApplicationListenerProps.open */\n    open?: boolean;\n  };\n}\n\n/**\n * Creates a Restate service deployment running as a Fargate task and backed by EFS.\n *\n * Please note that this construct is still experimental! Use with caution.\n */\nexport class FargateRestateDeployment extends Construct implements IRestateEnvironment {\n  readonly invokerRole: iam.IRole;\n  readonly vpc: ec2.IVpc;\n\n  readonly ingressUrl: string;\n  readonly adminUrl: string;\n  readonly securityGroup: ec2.SecurityGroup;\n  readonly dataStore: efs.FileSystem;\n  readonly ingressListener: elb2.ApplicationListener;\n  readonly adminListener: elb2.ApplicationListener;\n\n  constructor(scope: Construct, id: string, props: RestateFargateProps) {\n    super(scope, id);\n\n    this.vpc = props.vpc ?? ec2.Vpc.fromLookup(this, \"Vpc\", { isDefault: true });\n\n    const restateImage = props.restateImage ?? RESTATE_IMAGE_DEFAULT;\n    const restateTag = props.restateTag ?? RESTATE_DOCKER_DEFAULT_TAG;\n    const adotTag = props.adotTag ?? ADOT_DOCKER_DEFAULT_TAG; // TODO: add X-Ray support like we have for EC2\n\n    const fs = new efs.FileSystem(this, \"DataStore\", {\n      vpc: this.vpc,\n      lifecyclePolicy: efs.LifecyclePolicy.AFTER_30_DAYS,\n      performanceMode: efs.PerformanceMode.GENERAL_PURPOSE,\n      throughputMode: efs.ThroughputMode.BURSTING,\n      removalPolicy: props.removalPolicy ?? cdk.RemovalPolicy.DESTROY,\n    });\n    fs.addToResourcePolicy(\n      new iam.PolicyStatement({\n        sid: \"AllowEfsMount\",\n        actions: [\"elasticfilesystem:ClientMount\"],\n        // Restricting to the ECS execution role does not work; probably doesn't matter - EFS access is secured by a security group\n        principals: [new iam.AnyPrincipal()],\n        conditions: {\n          Bool: {\n            \"elasticfilesystem:AccessedViaMountTarget\": \"true\",\n          },\n        },\n      }),\n    );\n    this.dataStore = fs;\n\n    const cluster = new ecs.Cluster(this, \"Cluster\", {\n      vpc: this.vpc,\n      clusterName: props.clusterName,\n    });\n\n    const restateTask = new ecs.FargateTaskDefinition(this, \"RestateTask\", {\n      cpu: 4096,\n      memoryLimitMiB: 8192,\n      runtimePlatform: {\n        cpuArchitecture: ecs.CpuArchitecture.ARM64,\n        operatingSystemFamily: ecs.OperatingSystemFamily.LINUX,\n      },\n      volumes: [\n        {\n          name: \"restateStore\",\n          efsVolumeConfiguration: {\n            fileSystemId: fs.fileSystemId,\n            authorizationConfig: {},\n          },\n        },\n      ],\n    });\n\n    // TODO: Start an ADOT container and hook it up to Restate and AWS X-Ray or another OTel sink\n    // if (props.tracing === TracingMode.AWS_XRAY) {\n    //   restateTask.taskRole.addManagedPolicy(iam.ManagedPolicy.fromAwsManagedPolicyName(\"AWSXrayWriteOnlyAccess\"));\n    // }\n\n    new iam.Policy(this, \"TaskPolicy\", {\n      statements: [\n        new iam.PolicyStatement({\n          sid: \"AllowAssumeAnyRole\",\n          actions: [\"sts:AssumeRole\"],\n          resources: [\"*\"], // we don't know upfront what invoker roles we may be asked to assume at runtime\n        }),\n      ],\n    }).attachToRole(restateTask.taskRole);\n\n    const invokerRole = new iam.Role(this, \"InvokerRole\", {\n      assumedBy: new iam.ArnPrincipal(restateTask.taskRole.roleArn),\n      description: \"Assumed by Restate deployment to invoke Lambda-based services\",\n    });\n    invokerRole.grantAssumeRole(restateTask.taskRole);\n    this.invokerRole = invokerRole;\n\n    const logGroup =\n      props.logGroup ??\n      new logs.LogGroup(this, \"Logs\", {\n        logGroupName: `/restate/${id}`,\n        retention: logs.RetentionDays.ONE_MONTH,\n        removalPolicy: props.removalPolicy ?? cdk.RemovalPolicy.DESTROY,\n      });\n\n    const restate = restateTask.addContainer(\"Restate\", {\n      containerName: \"restate-runtime\",\n      image: ecs.ContainerImage.fromRegistry(`${restateImage}:${restateTag}`),\n      portMappings: [{ containerPort: RESTATE_INGRESS_PORT }, { containerPort: RESTATE_ADMIN_PORT }],\n      logging: ecs.LogDriver.awsLogs({\n        logGroup,\n        streamPrefix: \"restate\",\n      }),\n      environment: {\n        RESTATE_OBSERVABILITY__LOG__FORMAT: \"Json\",\n        // RUST_LOG: \"warn,restate=info\",\n      },\n      command: props.command,\n      startTimeout: cdk.Duration.seconds(20),\n      stopTimeout: cdk.Duration.seconds(20),\n    });\n    restate.addMountPoints({\n      containerPath: \"/target\",\n      readOnly: false,\n      sourceVolume: \"restateStore\",\n    });\n\n    const restateSecurityGroup = new ec2.SecurityGroup(this, \"SecurityGroup\", {\n      vpc: this.vpc,\n      allowAllOutbound: true,\n    });\n    this.securityGroup = restateSecurityGroup;\n\n    const restateFargateService = new ecs.FargateService(this, \"Service\", {\n      cluster,\n      taskDefinition: restateTask,\n      assignPublicIp: true,\n      circuitBreaker: {\n        enable: true,\n        rollback: true,\n      },\n      minHealthyPercent: 0, // allow scale down to zero during deployments (required for at-most-1 max setting)\n      maxHealthyPercent: 100, // don't start more than one copy\n      securityGroups: [restateSecurityGroup],\n    });\n\n    fs.connections.allowDefaultPortFrom(restateSecurityGroup);\n    fs.connections.allowDefaultPortTo(restateSecurityGroup);\n    fs.grantRootAccess(restateFargateService.taskDefinition.taskRole.grantPrincipal);\n\n    const alb = new elb2.ApplicationLoadBalancer(this, \"Alb\", {\n      vpc: this.vpc,\n      internetFacing: props.loadBalancer?.internetFacing,\n    });\n\n    const publicApiCertificate = new acm.Certificate(this, \"Certificate\", {\n      domainName: props.dnsName,\n      validation: acm.CertificateValidation.fromDns(props.hostedZone),\n    });\n\n    const ingressListener = alb.addListener(\"IngressListener\", {\n      port: PUBLIC_INGRESS_PORT,\n      protocol: elb2.ApplicationProtocol.HTTPS,\n      certificates: [publicApiCertificate],\n      open: props.loadBalancer?.open,\n    });\n    ingressListener.addTargets(\"FargateIngressTarget\", {\n      targets: [\n        restateFargateService.loadBalancerTarget({\n          containerName: restate.containerName,\n          containerPort: RESTATE_INGRESS_PORT,\n        }),\n      ],\n      protocol: elb2.ApplicationProtocol.HTTP,\n      healthCheck: {\n        path: \"/grpc.health.v1.Health/Check\",\n        interval: cdk.Duration.seconds(5),\n        healthyThresholdCount: 3,\n        unhealthyThresholdCount: 3,\n        timeout: cdk.Duration.seconds(2),\n      },\n      deregistrationDelay: cdk.Duration.seconds(5),\n    });\n    this.ingressListener = ingressListener;\n\n    const adminListener = alb.addListener(\"AdminListener\", {\n      port: PUBLIC_ADMIN_PORT,\n      protocol: elb2.ApplicationProtocol.HTTPS,\n      certificates: [publicApiCertificate],\n    });\n    adminListener.addTargets(\"FargateAdminTarget\", {\n      targets: [\n        restateFargateService.loadBalancerTarget({\n          containerName: restate.containerName,\n          containerPort: RESTATE_ADMIN_PORT,\n        }),\n      ],\n      protocol: elb2.ApplicationProtocol.HTTP,\n      healthCheck: {\n        path: \"/health\",\n        interval: cdk.Duration.seconds(5),\n        healthyThresholdCount: 3,\n        unhealthyThresholdCount: 3,\n        timeout: cdk.Duration.seconds(2),\n      },\n      deregistrationDelay: cdk.Duration.seconds(5),\n    });\n    this.adminListener = adminListener;\n\n    new r53.ARecord(this, \"AlbAlias\", {\n      zone: props.hostedZone,\n      recordName: props.dnsName.split(\".\")[0],\n      target: r53.RecordTarget.fromAlias(new targets.LoadBalancerTarget(alb)),\n      // other ARecord configuration...\n    });\n\n    this.ingressUrl = `https://${props.dnsName}${PUBLIC_INGRESS_PORT == 443 ? \"\" : `:${PUBLIC_INGRESS_PORT}`}`;\n    this.adminUrl = `https://${props.dnsName}:${PUBLIC_ADMIN_PORT}`;\n  }\n}\n"]}
package/dist/index.d.ts CHANGED
@@ -1,5 +1,5 @@
1
- export * from "./lambda-service-registry";
2
- export * from "./registration-provider";
3
- export * from "./restate-cloud-environment";
1
+ export * from "./service-deployer";
4
2
  export * from "./restate-environment";
3
+ export * from "./deployments-common";
5
4
  export * from "./single-node-restate-deployment";
5
+ export * from "./fargate-restate-deployment";
package/dist/index.js CHANGED
@@ -24,9 +24,9 @@ var __exportStar = (this && this.__exportStar) || function(m, exports) {
24
24
  for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
25
25
  };
26
26
  Object.defineProperty(exports, "__esModule", { value: true });
27
- __exportStar(require("./lambda-service-registry"), exports);
28
- __exportStar(require("./registration-provider"), exports);
29
- __exportStar(require("./restate-cloud-environment"), exports);
27
+ __exportStar(require("./service-deployer"), exports);
30
28
  __exportStar(require("./restate-environment"), exports);
29
+ __exportStar(require("./deployments-common"), exports);
31
30
  __exportStar(require("./single-node-restate-deployment"), exports);
32
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9saWIvcmVzdGF0ZS1jb25zdHJ1Y3RzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQTs7Ozs7Ozs7O0dBU0c7Ozs7Ozs7Ozs7Ozs7Ozs7QUFFSCw0REFBMEM7QUFDMUMsMERBQXdDO0FBQ3hDLDhEQUE0QztBQUM1Qyx3REFBc0M7QUFDdEMsbUVBQWlEIiwic291cmNlc0NvbnRlbnQiOlsiLypcbiAqIENvcHlyaWdodCAoYykgMjAyMyAtIFJlc3RhdGUgU29mdHdhcmUsIEluYy4sIFJlc3RhdGUgR21iSFxuICpcbiAqIFRoaXMgZmlsZSBpcyBwYXJ0IG9mIHRoZSBSZXN0YXRlIFNESyBmb3IgTm9kZS5qcy9UeXBlU2NyaXB0LFxuICogd2hpY2ggaXMgcmVsZWFzZWQgdW5kZXIgdGhlIE1JVCBsaWNlbnNlLlxuICpcbiAqIFlvdSBjYW4gZmluZCBhIGNvcHkgb2YgdGhlIGxpY2Vuc2UgaW4gZmlsZSBMSUNFTlNFIGluIHRoZSByb290XG4gKiBkaXJlY3Rvcnkgb2YgdGhpcyByZXBvc2l0b3J5IG9yIHBhY2thZ2UsIG9yIGF0XG4gKiBodHRwczovL2dpdGh1Yi5jb20vcmVzdGF0ZWRldi9zZGstdHlwZXNjcmlwdC9ibG9iL21haW4vTElDRU5TRVxuICovXG5cbmV4cG9ydCAqIGZyb20gXCIuL2xhbWJkYS1zZXJ2aWNlLXJlZ2lzdHJ5XCI7XG5leHBvcnQgKiBmcm9tIFwiLi9yZWdpc3RyYXRpb24tcHJvdmlkZXJcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3Jlc3RhdGUtY2xvdWQtZW52aXJvbm1lbnRcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3Jlc3RhdGUtZW52aXJvbm1lbnRcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3NpbmdsZS1ub2RlLXJlc3RhdGUtZGVwbG95bWVudFwiOyJdfQ==
31
+ __exportStar(require("./fargate-restate-deployment"), exports);
32
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9saWIvcmVzdGF0ZS1jb25zdHJ1Y3RzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQTs7Ozs7Ozs7O0dBU0c7Ozs7Ozs7Ozs7Ozs7Ozs7QUFFSCxxREFBbUM7QUFDbkMsd0RBQXNDO0FBQ3RDLHVEQUFxQztBQUNyQyxtRUFBaUQ7QUFDakQsK0RBQTZDIiwic291cmNlc0NvbnRlbnQiOlsiLypcbiAqIENvcHlyaWdodCAoYykgMjAyMyAtIFJlc3RhdGUgU29mdHdhcmUsIEluYy4sIFJlc3RhdGUgR21iSFxuICpcbiAqIFRoaXMgZmlsZSBpcyBwYXJ0IG9mIHRoZSBSZXN0YXRlIFNESyBmb3IgTm9kZS5qcy9UeXBlU2NyaXB0LFxuICogd2hpY2ggaXMgcmVsZWFzZWQgdW5kZXIgdGhlIE1JVCBsaWNlbnNlLlxuICpcbiAqIFlvdSBjYW4gZmluZCBhIGNvcHkgb2YgdGhlIGxpY2Vuc2UgaW4gZmlsZSBMSUNFTlNFIGluIHRoZSByb290XG4gKiBkaXJlY3Rvcnkgb2YgdGhpcyByZXBvc2l0b3J5IG9yIHBhY2thZ2UsIG9yIGF0XG4gKiBodHRwczovL2dpdGh1Yi5jb20vcmVzdGF0ZWRldi9zZGstdHlwZXNjcmlwdC9ibG9iL21haW4vTElDRU5TRVxuICovXG5cbmV4cG9ydCAqIGZyb20gXCIuL3NlcnZpY2UtZGVwbG95ZXJcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3Jlc3RhdGUtZW52aXJvbm1lbnRcIjtcbmV4cG9ydCAqIGZyb20gXCIuL2RlcGxveW1lbnRzLWNvbW1vblwiO1xuZXhwb3J0ICogZnJvbSBcIi4vc2luZ2xlLW5vZGUtcmVzdGF0ZS1kZXBsb3ltZW50XCI7XG5leHBvcnQgKiBmcm9tIFwiLi9mYXJnYXRlLXJlc3RhdGUtZGVwbG95bWVudFwiO1xuIl19
@@ -8,6 +8,9 @@ export interface RegistrationProperties {
8
8
  invokeRoleArn?: string;
9
9
  removalPolicy?: cdk.RemovalPolicy;
10
10
  authTokenSecretArn?: string;
11
+ configurationVersion?: string;
12
+ private?: "true" | "false";
13
+ insecure?: "true" | "false";
11
14
  }
12
15
  /**
13
16
  * Custom Resource event handler for Restate service registration. This handler backs the custom resources created by
@@ -39,12 +39,13 @@ Object.defineProperty(exports, "__esModule", { value: true });
39
39
  exports.handler = void 0;
40
40
  const node_fetch_1 = __importDefault(require("node-fetch"));
41
41
  const client_secrets_manager_1 = require("@aws-sdk/client-secrets-manager");
42
- const https = __importStar(require("https"));
43
42
  const crypto_1 = require("crypto");
43
+ const https = __importStar(require("https"));
44
44
  const MAX_HEALTH_CHECK_ATTEMPTS = 5; // This is intentionally quite long to allow some time for first-run EC2 and Docker boot up
45
45
  const MAX_REGISTRATION_ATTEMPTS = 3;
46
- const INSECURE = true;
46
+ // const INSECURE = true;
47
47
  const DEPLOYMENTS_PATH = "deployments";
48
+ const SERVICES_PATH = "services";
48
49
  const DEPLOYMENTS_PATH_LEGACY = "endpoints"; // temporarily fall back for legacy clusters
49
50
  /**
50
51
  * Custom Resource event handler for Restate service registration. This handler backs the custom resources created by
@@ -80,11 +81,11 @@ const handler = async function (event) {
80
81
  const props = event.ResourceProperties;
81
82
  const authHeader = await createAuthHeader(props);
82
83
  let attempt;
83
- const controller = new AbortController();
84
84
  const healthCheckUrl = `${props.adminUrl}/health`;
85
85
  console.log(`Performing health check against: ${healthCheckUrl}`);
86
86
  attempt = 1;
87
87
  while (true) {
88
+ const controller = new AbortController();
88
89
  const healthCheckTimeout = setTimeout(() => controller.abort("timeout"), 5000);
89
90
  let healthResponse = undefined;
90
91
  let errorMessage = undefined;
@@ -92,7 +93,7 @@ const handler = async function (event) {
92
93
  healthResponse = await (0, node_fetch_1.default)(healthCheckUrl, {
93
94
  signal: controller.signal,
94
95
  headers: authHeader,
95
- agent: INSECURE ? new https.Agent({ rejectUnauthorized: false }) : undefined,
96
+ agent: props.insecure ? new https.Agent({ rejectUnauthorized: false }) : undefined,
96
97
  }).finally(() => clearTimeout(healthCheckTimeout));
97
98
  console.log(`Got health check response back: ${healthResponse.status}`);
98
99
  if (healthResponse.status >= 200 && healthResponse.status < 300) {
@@ -123,6 +124,8 @@ const handler = async function (event) {
123
124
  attempt = 1;
124
125
  while (true) {
125
126
  try {
127
+ console.log(`Making request #${attempt}...`);
128
+ const controller = new AbortController();
126
129
  const registerCallTimeout = setTimeout(() => controller.abort("timeout"), 10000);
127
130
  const registerDeploymentResponse = await (0, node_fetch_1.default)(deploymentsUrl, {
128
131
  signal: controller.signal,
@@ -132,24 +135,50 @@ const handler = async function (event) {
132
135
  "Content-Type": "application/json",
133
136
  ...authHeader,
134
137
  },
135
- agent: INSECURE ? new https.Agent({ rejectUnauthorized: false }) : undefined,
138
+ agent: props.insecure ? new https.Agent({ rejectUnauthorized: false }) : undefined,
136
139
  }).finally(() => clearTimeout(registerCallTimeout));
137
- console.log(`Got registration response back: ${registerDeploymentResponse.status}`);
138
140
  if (registerDeploymentResponse.status == 404 && attempt == 1) {
139
141
  deploymentsUrl = `${props.adminUrl}/${DEPLOYMENTS_PATH_LEGACY}`;
140
142
  console.log(`Got 404, falling back to <0.7.0 legacy endpoint registration at: ${deploymentsUrl}`);
141
143
  }
142
144
  if (registerDeploymentResponse.status >= 200 && registerDeploymentResponse.status < 300) {
143
145
  const response = (await registerDeploymentResponse.json());
146
+ // TODO: there may be more than one! support optional exact/partial matching
144
147
  if (response?.services?.[0]?.name !== props.servicePath) {
145
148
  failureReason =
146
149
  "Restate service registration failed: service name indicated by service response" +
147
150
  ` ("${response?.services?.[0]?.name})) does not match the expected value ("${props.servicePath}")!`;
148
- console.error(failureReason);
149
151
  break; // don't throw immediately - let retry loop decide whether to abort
150
152
  }
151
- console.log("Success!");
152
- return;
153
+ console.log("Successful registration!");
154
+ const isPublic = (props.private ?? "false") === "false";
155
+ console.log(`Marking service ${props.servicePath} as ${isPublic ? "public" : "private"}...`);
156
+ const controller = new AbortController();
157
+ const privateCallTimeout = setTimeout(() => controller.abort("timeout"), 5000);
158
+ const patchResponse = await (0, node_fetch_1.default)(`${props.adminUrl}/${SERVICES_PATH}/${props.servicePath}`, {
159
+ signal: controller.signal,
160
+ method: "PATCH",
161
+ headers: {
162
+ "Content-Type": "application/json",
163
+ ...authHeader,
164
+ },
165
+ body: JSON.stringify({ public: isPublic }),
166
+ agent: props.insecure ? new https.Agent({ rejectUnauthorized: false }) : undefined,
167
+ }).finally(() => clearTimeout(privateCallTimeout));
168
+ console.log(`Got patch response back: ${patchResponse.status}`);
169
+ if (patchResponse.status != 200) {
170
+ failureReason = `Marking service as ${props.private ? "private" : "public"} failed: ${patchResponse.statusText} (${patchResponse.status})`;
171
+ break; // don't throw immediately - let retry loop decide whether to abort s
172
+ }
173
+ console.log(`Successfully marked service as ${isPublic ? "public" : "private"}.`);
174
+ return; // Overall success!
175
+ }
176
+ else {
177
+ console.log({
178
+ message: `Got error response from Restate.`,
179
+ code: registerDeploymentResponse.status,
180
+ body: await registerDeploymentResponse.text(),
181
+ });
153
182
  }
154
183
  }
155
184
  catch (e) {
@@ -157,7 +186,7 @@ const handler = async function (event) {
157
186
  failureReason = `Restate service registration failed: ${e?.message}`;
158
187
  }
159
188
  if (attempt >= MAX_REGISTRATION_ATTEMPTS) {
160
- console.error(`Service registration failed after ${attempt} attempts.`);
189
+ failureReason = `Service registration failed after ${attempt} attempts.`;
161
190
  break;
162
191
  }
163
192
  attempt += 1;
@@ -165,6 +194,7 @@ const handler = async function (event) {
165
194
  console.log(`Retrying registration after ${waitTimeMillis} ms...`);
166
195
  await sleep(waitTimeMillis);
167
196
  }
197
+ console.error(failureReason);
168
198
  throw new Error(failureReason ?? "Restate service registration failed. Please see logs for details.");
169
199
  };
170
200
  exports.handler = handler;
@@ -183,6 +213,6 @@ async function createAuthHeader(props) {
183
213
  };
184
214
  }
185
215
  async function sleep(millis) {
186
- await new Promise((resolve) => setTimeout(resolve, millis));
216
+ return new Promise((resolve) => setTimeout(resolve, millis));
187
217
  }
188
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../lib/restate-constructs/register-service-handler/index.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2023 - Restate Software, Inc., Restate GmbH\n *\n * This file is part of the Restate SDK for Node.js/TypeScript,\n * which is released under the MIT license.\n *\n * You can find a copy of the license in file LICENSE in the root\n * directory of this repository or package, or at\n * https://github.com/restatedev/sdk-typescript/blob/main/LICENSE\n */\n\nimport { Handler } from \"aws-lambda/handler\";\nimport { CloudFormationCustomResourceEvent } from \"aws-lambda/trigger/cloudformation-custom-resource\";\nimport fetch from \"node-fetch\";\nimport * as cdk from \"aws-cdk-lib\";\nimport { GetSecretValueCommand, SecretsManagerClient } from \"@aws-sdk/client-secrets-manager\";\nimport * as https from \"https\";\nimport { randomInt } from \"crypto\";\n\nexport interface RegistrationProperties {\n  servicePath?: string;\n  adminUrl?: string;\n  serviceLambdaArn?: string;\n  invokeRoleArn?: string;\n  removalPolicy?: cdk.RemovalPolicy;\n  authTokenSecretArn?: string;\n}\n\ntype RegisterDeploymentResponse = {\n  id?: string;\n  services?: { name?: string; revision?: number }[];\n};\n\nconst MAX_HEALTH_CHECK_ATTEMPTS = 5; // This is intentionally quite long to allow some time for first-run EC2 and Docker boot up\nconst MAX_REGISTRATION_ATTEMPTS = 3;\n\nconst INSECURE = true;\n\nconst DEPLOYMENTS_PATH = \"deployments\";\nconst DEPLOYMENTS_PATH_LEGACY = \"endpoints\"; // temporarily fall back for legacy clusters\n\n/**\n * Custom Resource event handler for Restate service registration. This handler backs the custom resources created by\n * {@link LambdaServiceRegistry} to facilitate Lambda service handler discovery.\n */\nexport const handler: Handler<CloudFormationCustomResourceEvent, void> = async function(event) {\n  console.log({ event });\n\n  if (event.RequestType === \"Delete\") {\n    // Since we retain older Lambda handler versions on update, we also leave the registered service alone. There may\n    // be unfinished invocations that require it; in the future we would want to inform Restate that we want to\n    // de-register the service, and wait for Restate to let us know that it is safe to delete the deployed Function\n    // version from Lambda.\n\n    // const props = event.ResourceProperties as RegistrationProperties;\n    // if (props.removalPolicy === cdk.RemovalPolicy.DESTROY) {\n    //   console.log(`De-registering service ${props.serviceLambdaArn}`);\n    //   const controller = new AbortController();\n    //   const id = btoa(props.serviceLambdaArn!); // TODO: we should be treating service ids as opaque\n    //   const deleteCallTimeout = setTimeout(() => controller.abort(\"timeout\"), 5_000);\n    //   const deleteResponse = await fetch(`${props.adminUrl}/${DEPLOYMENTS_PATH}/${id}?force=true`, {\n    //     signal: controller.signal,\n    //     method: \"DELETE\",\n    //     agent: INSECURE ? new https.Agent({ rejectUnauthorized: false }) : undefined,\n    //   }).finally(() => clearTimeout(deleteCallTimeout));\n    //\n    //   console.log(`Got delete response back: ${deleteResponse.status}`);\n    //   if (deleteResponse.status != 202) {\n    //     throw new Error(`Removing service deployment failed: ${deleteResponse.statusText} (${deleteResponse.status})`);\n    //   }\n    // }\n\n    console.warn(\"De-registering services is not supported currently. Previous version will remain registered.\");\n    return;\n  }\n\n  const props = event.ResourceProperties as RegistrationProperties;\n  const authHeader = await createAuthHeader(props);\n\n  let attempt;\n  const controller = new AbortController();\n\n  const healthCheckUrl = `${props.adminUrl}/health`;\n\n  console.log(`Performing health check against: ${healthCheckUrl}`);\n  attempt = 1;\n  while (true) {\n    const healthCheckTimeout = setTimeout(() => controller.abort(\"timeout\"), 5_000);\n    let healthResponse = undefined;\n    let errorMessage = undefined;\n    try {\n      healthResponse = await fetch(healthCheckUrl, {\n        signal: controller.signal,\n        headers: authHeader,\n        agent: INSECURE ? new https.Agent({ rejectUnauthorized: false }) : undefined,\n      }).finally(() => clearTimeout(healthCheckTimeout));\n\n      console.log(`Got health check response back: ${healthResponse.status}`);\n      if (healthResponse.status >= 200 && healthResponse.status < 300) {\n        break;\n      }\n      console.error(\n        `Restate health check failed: ${healthResponse.statusText} (${healthResponse.status}; attempt ${attempt})`,\n      );\n    } catch (e) {\n      errorMessage = (e as Error)?.message;\n      console.error(`Restate health check failed: \"${errorMessage}\" (attempt ${attempt})`);\n    }\n\n    if (attempt >= MAX_HEALTH_CHECK_ATTEMPTS) {\n      console.error(`Admin service health check failing after ${attempt} attempts.`);\n      throw new Error(errorMessage ?? `${healthResponse?.statusText} (${healthResponse?.status})`);\n    }\n    attempt += 1;\n\n    const waitTimeMillis = randomInt(2_000) + 2 ** attempt * 1_000; // 3s -> 6s -> 10s -> 18s -> 34s\n    console.log(`Retrying after ${waitTimeMillis} ms...`);\n    await sleep(waitTimeMillis);\n  }\n\n  let deploymentsUrl = `${props.adminUrl}/${DEPLOYMENTS_PATH}`;\n  const registrationRequest = JSON.stringify({\n    arn: props.serviceLambdaArn,\n    assume_role_arn: props.invokeRoleArn,\n  });\n\n  let failureReason;\n  console.log(`Triggering registration at ${deploymentsUrl}: ${registrationRequest}`);\n  attempt = 1;\n  while (true) {\n    try {\n      const registerCallTimeout = setTimeout(() => controller.abort(\"timeout\"), 10_000);\n      const registerDeploymentResponse = await fetch(deploymentsUrl, {\n        signal: controller.signal,\n        method: \"POST\",\n        body: registrationRequest,\n        headers: {\n          \"Content-Type\": \"application/json\",\n          ...authHeader,\n        },\n        agent: INSECURE ? new https.Agent({ rejectUnauthorized: false }) : undefined,\n      }).finally(() => clearTimeout(registerCallTimeout));\n\n      console.log(`Got registration response back: ${registerDeploymentResponse.status}`);\n\n      if (registerDeploymentResponse.status == 404 && attempt == 1) {\n        deploymentsUrl = `${props.adminUrl}/${DEPLOYMENTS_PATH_LEGACY}`;\n        console.log(`Got 404, falling back to <0.7.0 legacy endpoint registration at: ${deploymentsUrl}`);\n      }\n\n      if (registerDeploymentResponse.status >= 200 && registerDeploymentResponse.status < 300) {\n        const response = (await registerDeploymentResponse.json()) as RegisterDeploymentResponse;\n\n        if (response?.services?.[0]?.name !== props.servicePath) {\n          failureReason =\n            \"Restate service registration failed: service name indicated by service response\" +\n            ` (\"${response?.services?.[0]?.name})) does not match the expected value (\"${props.servicePath}\")!`;\n          console.error(failureReason);\n          break; // don't throw immediately - let retry loop decide whether to abort\n        }\n\n        console.log(\"Success!\");\n        return;\n      }\n    } catch (e) {\n      console.error(`Service registration call failed: ${(e as Error)?.message} (attempt ${attempt})`);\n      failureReason = `Restate service registration failed: ${(e as Error)?.message}`;\n    }\n\n    if (attempt >= MAX_REGISTRATION_ATTEMPTS) {\n      console.error(`Service registration failed after ${attempt} attempts.`);\n      break;\n    }\n    attempt += 1;\n    const waitTimeMillis = randomInt(2_000) + 2 ** attempt * 1_000; // 3s -> 6s -> 10s\n    console.log(`Retrying registration after ${waitTimeMillis} ms...`);\n    await sleep(waitTimeMillis);\n  }\n\n  throw new Error(failureReason ?? \"Restate service registration failed. Please see logs for details.\");\n};\n\nasync function createAuthHeader(props: RegistrationProperties): Promise<Record<string, string>> {\n  if (!props.authTokenSecretArn) {\n    return {};\n  }\n\n  console.log(`Using bearer authentication token from secret ${props.authTokenSecretArn}`);\n  const ssm = new SecretsManagerClient();\n  const response = await ssm.send(\n    new GetSecretValueCommand({\n      SecretId: props.authTokenSecretArn,\n    }),\n  );\n\n  console.log(`Successfully retrieved secret \"${response.Name}\" version ${response.VersionId}`);\n  return {\n    Authorization: `Bearer ${response.SecretString}`,\n  };\n}\n\nasync function sleep(millis: number) {\n  await new Promise((resolve) => setTimeout(resolve, millis));\n}\n"]}
218
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../lib/restate-constructs/register-service-handler/index.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2023 - Restate Software, Inc., Restate GmbH\n *\n * This file is part of the Restate SDK for Node.js/TypeScript,\n * which is released under the MIT license.\n *\n * You can find a copy of the license in file LICENSE in the root\n * directory of this repository or package, or at\n * https://github.com/restatedev/sdk-typescript/blob/main/LICENSE\n */\n\nimport { Handler } from \"aws-lambda/handler\";\nimport { CloudFormationCustomResourceEvent } from \"aws-lambda/trigger/cloudformation-custom-resource\";\nimport fetch from \"node-fetch\";\nimport * as cdk from \"aws-cdk-lib\";\nimport { GetSecretValueCommand, SecretsManagerClient } from \"@aws-sdk/client-secrets-manager\";\nimport { randomInt } from \"crypto\";\nimport * as https from \"https\";\n\nexport interface RegistrationProperties {\n  servicePath?: string;\n  adminUrl?: string;\n  serviceLambdaArn?: string;\n  invokeRoleArn?: string;\n  removalPolicy?: cdk.RemovalPolicy;\n  authTokenSecretArn?: string;\n  /* Not used by the handler, purely used to trick CloudFormation to perform an update when it otherwise would not. */\n  configurationVersion?: string;\n  /* Whether to mark the service as private, and make it unavailable to be called via Restate ingress. */\n  private?: \"true\" | \"false\";\n  /* Whether to trust any certificate from the admin endpoint. */\n  insecure?: \"true\" | \"false\";\n}\n\ntype RegisterDeploymentResponse = {\n  id?: string;\n  services?: { name?: string; revision?: number }[];\n};\n\nconst MAX_HEALTH_CHECK_ATTEMPTS = 5; // This is intentionally quite long to allow some time for first-run EC2 and Docker boot up\nconst MAX_REGISTRATION_ATTEMPTS = 3;\n\n// const INSECURE = true;\n\nconst DEPLOYMENTS_PATH = \"deployments\";\nconst SERVICES_PATH = \"services\";\nconst DEPLOYMENTS_PATH_LEGACY = \"endpoints\"; // temporarily fall back for legacy clusters\n\n/**\n * Custom Resource event handler for Restate service registration. This handler backs the custom resources created by\n * {@link LambdaServiceRegistry} to facilitate Lambda service handler discovery.\n */\nexport const handler: Handler<CloudFormationCustomResourceEvent, void> = async function (event) {\n  console.log({ event });\n\n  if (event.RequestType === \"Delete\") {\n    // Since we retain older Lambda handler versions on update, we also leave the registered service alone. There may\n    // be unfinished invocations that require it; in the future we would want to inform Restate that we want to\n    // de-register the service, and wait for Restate to let us know that it is safe to delete the deployed Function\n    // version from Lambda.\n\n    // const props = event.ResourceProperties as RegistrationProperties;\n    // if (props.removalPolicy === cdk.RemovalPolicy.DESTROY) {\n    //   console.log(`De-registering service ${props.serviceLambdaArn}`);\n    //   const controller = new AbortController();\n    //   const id = btoa(props.serviceLambdaArn!); // TODO: we should be treating service ids as opaque\n    //   const deleteCallTimeout = setTimeout(() => controller.abort(\"timeout\"), 5_000);\n    //   const deleteResponse = await fetch(`${props.adminUrl}/${DEPLOYMENTS_PATH}/${id}?force=true`, {\n    //     signal: controller.signal,\n    //     method: \"DELETE\",\n    //     agent: INSECURE ? new https.Agent({ rejectUnauthorized: false }) : undefined,\n    //   }).finally(() => clearTimeout(deleteCallTimeout));\n    //\n    //   console.log(`Got delete response back: ${deleteResponse.status}`);\n    //   if (deleteResponse.status != 202) {\n    //     throw new Error(`Removing service deployment failed: ${deleteResponse.statusText} (${deleteResponse.status})`);\n    //   }\n    // }\n\n    console.warn(\"De-registering services is not supported currently. Previous version will remain registered.\");\n    return;\n  }\n\n  const props = event.ResourceProperties as RegistrationProperties;\n  const authHeader = await createAuthHeader(props);\n\n  let attempt;\n\n  const healthCheckUrl = `${props.adminUrl}/health`;\n\n  console.log(`Performing health check against: ${healthCheckUrl}`);\n  attempt = 1;\n  while (true) {\n    const controller = new AbortController();\n    const healthCheckTimeout = setTimeout(() => controller.abort(\"timeout\"), 5_000);\n    let healthResponse = undefined;\n    let errorMessage = undefined;\n    try {\n      healthResponse = await fetch(healthCheckUrl, {\n        signal: controller.signal,\n        headers: authHeader,\n        agent: props.insecure ? new https.Agent({ rejectUnauthorized: false }) : undefined,\n      }).finally(() => clearTimeout(healthCheckTimeout));\n\n      console.log(`Got health check response back: ${healthResponse.status}`);\n      if (healthResponse.status >= 200 && healthResponse.status < 300) {\n        break;\n      }\n      console.error(\n        `Restate health check failed: ${healthResponse.statusText} (${healthResponse.status}; attempt ${attempt})`,\n      );\n    } catch (e) {\n      errorMessage = (e as Error)?.message;\n      console.error(`Restate health check failed: \"${errorMessage}\" (attempt ${attempt})`);\n    }\n\n    if (attempt >= MAX_HEALTH_CHECK_ATTEMPTS) {\n      console.error(`Admin service health check failing after ${attempt} attempts.`);\n      throw new Error(errorMessage ?? `${healthResponse?.statusText} (${healthResponse?.status})`);\n    }\n    attempt += 1;\n\n    const waitTimeMillis = randomInt(2_000) + 2 ** attempt * 1_000; // 3s -> 6s -> 10s -> 18s -> 34s\n    console.log(`Retrying after ${waitTimeMillis} ms...`);\n    await sleep(waitTimeMillis);\n  }\n\n  let deploymentsUrl = `${props.adminUrl}/${DEPLOYMENTS_PATH}`;\n  const registrationRequest = JSON.stringify({\n    arn: props.serviceLambdaArn,\n    assume_role_arn: props.invokeRoleArn,\n  });\n\n  let failureReason;\n  console.log(`Triggering registration at ${deploymentsUrl}: ${registrationRequest}`);\n  attempt = 1;\n  while (true) {\n    try {\n      console.log(`Making request #${attempt}...`);\n      const controller = new AbortController();\n      const registerCallTimeout = setTimeout(() => controller.abort(\"timeout\"), 10_000);\n      const registerDeploymentResponse = await fetch(deploymentsUrl, {\n        signal: controller.signal,\n        method: \"POST\",\n        body: registrationRequest,\n        headers: {\n          \"Content-Type\": \"application/json\",\n          ...authHeader,\n        },\n        agent: props.insecure ? new https.Agent({ rejectUnauthorized: false }) : undefined,\n      }).finally(() => clearTimeout(registerCallTimeout));\n\n      if (registerDeploymentResponse.status == 404 && attempt == 1) {\n        deploymentsUrl = `${props.adminUrl}/${DEPLOYMENTS_PATH_LEGACY}`;\n        console.log(`Got 404, falling back to <0.7.0 legacy endpoint registration at: ${deploymentsUrl}`);\n      }\n\n      if (registerDeploymentResponse.status >= 200 && registerDeploymentResponse.status < 300) {\n        const response = (await registerDeploymentResponse.json()) as RegisterDeploymentResponse;\n\n        // TODO: there may be more than one! support optional exact/partial matching\n        if (response?.services?.[0]?.name !== props.servicePath) {\n          failureReason =\n            \"Restate service registration failed: service name indicated by service response\" +\n            ` (\"${response?.services?.[0]?.name})) does not match the expected value (\"${props.servicePath}\")!`;\n          break; // don't throw immediately - let retry loop decide whether to abort\n        }\n\n        console.log(\"Successful registration!\");\n\n        const isPublic = (props.private ?? \"false\") === \"false\";\n        console.log(`Marking service ${props.servicePath} as ${isPublic ? \"public\" : \"private\"}...`);\n        const controller = new AbortController();\n        const privateCallTimeout = setTimeout(() => controller.abort(\"timeout\"), 5_000);\n        const patchResponse = await fetch(`${props.adminUrl}/${SERVICES_PATH}/${props.servicePath}`, {\n          signal: controller.signal,\n          method: \"PATCH\",\n          headers: {\n            \"Content-Type\": \"application/json\",\n            ...authHeader,\n          },\n          body: JSON.stringify({ public: isPublic }),\n          agent: props.insecure ? new https.Agent({ rejectUnauthorized: false }) : undefined,\n        }).finally(() => clearTimeout(privateCallTimeout));\n\n        console.log(`Got patch response back: ${patchResponse.status}`);\n        if (patchResponse.status != 200) {\n          failureReason = `Marking service as ${props.private ? \"private\" : \"public\"} failed: ${patchResponse.statusText} (${patchResponse.status})`;\n          break; // don't throw immediately - let retry loop decide whether to abort s\n        }\n\n        console.log(`Successfully marked service as ${isPublic ? \"public\" : \"private\"}.`);\n\n        return; // Overall success!\n      } else {\n        console.log({\n          message: `Got error response from Restate.`,\n          code: registerDeploymentResponse.status,\n          body: await registerDeploymentResponse.text(),\n        });\n      }\n    } catch (e) {\n      console.error(`Service registration call failed: ${(e as Error)?.message} (attempt ${attempt})`);\n      failureReason = `Restate service registration failed: ${(e as Error)?.message}`;\n    }\n\n    if (attempt >= MAX_REGISTRATION_ATTEMPTS) {\n      failureReason = `Service registration failed after ${attempt} attempts.`;\n      break;\n    }\n    attempt += 1;\n    const waitTimeMillis = randomInt(2_000) + 2 ** attempt * 1_000; // 3s -> 6s -> 10s\n    console.log(`Retrying registration after ${waitTimeMillis} ms...`);\n    await sleep(waitTimeMillis);\n  }\n\n  console.error(failureReason);\n  throw new Error(failureReason ?? \"Restate service registration failed. Please see logs for details.\");\n};\n\nasync function createAuthHeader(props: RegistrationProperties): Promise<Record<string, string>> {\n  if (!props.authTokenSecretArn) {\n    return {};\n  }\n\n  console.log(`Using bearer authentication token from secret ${props.authTokenSecretArn}`);\n  const ssm = new SecretsManagerClient();\n  const response = await ssm.send(\n    new GetSecretValueCommand({\n      SecretId: props.authTokenSecretArn,\n    }),\n  );\n\n  console.log(`Successfully retrieved secret \"${response.Name}\" version ${response.VersionId}`);\n  return {\n    Authorization: `Bearer ${response.SecretString}`,\n  };\n}\n\nasync function sleep(millis: number) {\n  return new Promise((resolve) => setTimeout(resolve, millis));\n}\n"]}