@restatedev/restate-cdk 0.3.0 → 0.4.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/index.d.ts +3 -0
- package/dist/index.js +4 -1
- package/dist/lambda-service-registry.d.ts +3 -1
- package/dist/lambda-service-registry.js +9 -4
- package/dist/register-service-handler/index.d.ts +4 -3
- package/dist/register-service-handler/index.js +119 -48
- package/dist/registration-provider.d.ts +12 -0
- package/dist/registration-provider.js +77 -0
- package/dist/restate-cloud-endpoint.d.ts +27 -0
- package/dist/restate-cloud-endpoint.js +70 -0
- package/dist/restate-instance.d.ts +13 -0
- package/dist/restate-instance.js +3 -0
- package/dist/single-node-restate-instance.d.ts +7 -24
- package/dist/single-node-restate-instance.js +61 -85
- package/package.json +5 -3
package/dist/index.d.ts
CHANGED
package/dist/index.js
CHANGED
|
@@ -25,5 +25,8 @@ var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
|
25
25
|
};
|
|
26
26
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
27
27
|
__exportStar(require("./lambda-service-registry"), exports);
|
|
28
|
+
__exportStar(require("./registration-provider"), exports);
|
|
29
|
+
__exportStar(require("./restate-cloud-endpoint"), exports);
|
|
30
|
+
__exportStar(require("./restate-instance"), exports);
|
|
28
31
|
__exportStar(require("./single-node-restate-instance"), exports);
|
|
29
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
32
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9saWIvcmVzdGF0ZS1jb25zdHJ1Y3RzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQTs7Ozs7Ozs7O0dBU0c7Ozs7Ozs7Ozs7Ozs7Ozs7QUFFSCw0REFBMEM7QUFDMUMsMERBQXdDO0FBQ3hDLDJEQUF5QztBQUN6QyxxREFBbUM7QUFDbkMsaUVBQStDIiwic291cmNlc0NvbnRlbnQiOlsiLypcbiAqIENvcHlyaWdodCAoYykgMjAyMyAtIFJlc3RhdGUgU29mdHdhcmUsIEluYy4sIFJlc3RhdGUgR21iSFxuICpcbiAqIFRoaXMgZmlsZSBpcyBwYXJ0IG9mIHRoZSBSZXN0YXRlIFNESyBmb3IgTm9kZS5qcy9UeXBlU2NyaXB0LFxuICogd2hpY2ggaXMgcmVsZWFzZWQgdW5kZXIgdGhlIE1JVCBsaWNlbnNlLlxuICpcbiAqIFlvdSBjYW4gZmluZCBhIGNvcHkgb2YgdGhlIGxpY2Vuc2UgaW4gZmlsZSBMSUNFTlNFIGluIHRoZSByb290XG4gKiBkaXJlY3Rvcnkgb2YgdGhpcyByZXBvc2l0b3J5IG9yIHBhY2thZ2UsIG9yIGF0XG4gKiBodHRwczovL2dpdGh1Yi5jb20vcmVzdGF0ZWRldi9zZGstdHlwZXNjcmlwdC9ibG9iL21haW4vTElDRU5TRVxuICovXG5cbmV4cG9ydCAqIGZyb20gXCIuL2xhbWJkYS1zZXJ2aWNlLXJlZ2lzdHJ5XCI7XG5leHBvcnQgKiBmcm9tIFwiLi9yZWdpc3RyYXRpb24tcHJvdmlkZXJcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3Jlc3RhdGUtY2xvdWQtZW5kcG9pbnRcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3Jlc3RhdGUtaW5zdGFuY2VcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3NpbmdsZS1ub2RlLXJlc3RhdGUtaW5zdGFuY2VcIjsiXX0=
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import * as lambda from "aws-cdk-lib/aws-lambda";
|
|
2
2
|
import { Construct } from "constructs";
|
|
3
|
+
import { RestateInstance } from "./restate-instance";
|
|
3
4
|
/**
|
|
4
5
|
* A Restate RPC service path. Example: `greeter`.
|
|
5
6
|
*/
|
|
@@ -7,6 +8,7 @@ type RestatePath = string;
|
|
|
7
8
|
export interface RestateInstanceRef {
|
|
8
9
|
readonly metaEndpoint: string;
|
|
9
10
|
readonly invokerRoleArn: string;
|
|
11
|
+
readonly authTokenSecretArn?: string;
|
|
10
12
|
}
|
|
11
13
|
/**
|
|
12
14
|
* A collection of Lambda Restate RPC Service handlers.
|
|
@@ -19,7 +21,7 @@ export type LambdaServiceRegistryProps = {
|
|
|
19
21
|
/**
|
|
20
22
|
* Custom resource provider token required for service discovery.
|
|
21
23
|
*/
|
|
22
|
-
|
|
24
|
+
restate: RestateInstance;
|
|
23
25
|
};
|
|
24
26
|
/**
|
|
25
27
|
* Represents a collection of Lambda-based Restate RPC services. This component is used to register
|
|
@@ -45,10 +45,14 @@ const constructs_1 = require("constructs");
|
|
|
45
45
|
class LambdaServiceRegistry extends constructs_1.Construct {
|
|
46
46
|
constructor(scope, id, props) {
|
|
47
47
|
super(scope, id);
|
|
48
|
+
if (Object.values(props.serviceHandlers).length == 0) {
|
|
49
|
+
throw new Error("Please specify at least one service handler.");
|
|
50
|
+
}
|
|
48
51
|
this.serviceHandlers = props.serviceHandlers;
|
|
49
|
-
this.registrationProviderToken = props.registrationProviderToken;
|
|
52
|
+
this.registrationProviderToken = props.restate.registrationProviderToken.value;
|
|
50
53
|
}
|
|
51
54
|
register(restate) {
|
|
55
|
+
const invokerRole = iam.Role.fromRoleArn(this, "InvokerRole", restate.invokerRoleArn);
|
|
52
56
|
const allowInvokeFunction = new iam.Policy(this, "AllowInvokeFunction", {
|
|
53
57
|
statements: [
|
|
54
58
|
new iam.PolicyStatement({
|
|
@@ -59,7 +63,6 @@ class LambdaServiceRegistry extends constructs_1.Construct {
|
|
|
59
63
|
}),
|
|
60
64
|
],
|
|
61
65
|
});
|
|
62
|
-
const invokerRole = iam.Role.fromRoleArn(this, "InvokerRole", restate.invokerRoleArn);
|
|
63
66
|
invokerRole.attachInlinePolicy(allowInvokeFunction);
|
|
64
67
|
for (const [path, handler] of Object.entries(this.serviceHandlers)) {
|
|
65
68
|
this.registerHandler(restate, { path, handler }, allowInvokeFunction);
|
|
@@ -86,10 +89,12 @@ class RestateServiceRegistrar extends constructs_1.Construct {
|
|
|
86
89
|
properties: {
|
|
87
90
|
servicePath: props.service.path,
|
|
88
91
|
metaEndpoint: props.restate.metaEndpoint,
|
|
92
|
+
authTokenSecretArn: props.restate.authTokenSecretArn,
|
|
89
93
|
serviceLambdaArn: props.service.handler.currentVersion.functionArn,
|
|
90
|
-
|
|
94
|
+
invokeRoleArn: props.restate.invokerRoleArn,
|
|
95
|
+
removalPolicy: cdk.RemovalPolicy.RETAIN,
|
|
91
96
|
},
|
|
92
97
|
});
|
|
93
98
|
}
|
|
94
99
|
}
|
|
95
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
100
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lambda-service-registry.js","sourceRoot":"","sources":["../lib/restate-constructs/lambda-service-registry.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 * as cdk from \"aws-cdk-lib\";\nimport * as iam from \"aws-cdk-lib/aws-iam\";\nimport * as lambda from \"aws-cdk-lib/aws-lambda\";\nimport { Construct } from \"constructs\";\nimport { RegistrationProperties } from \"./register-service-handler\";\n\nimport { RestateInstance } from \"./restate-instance\";\n\n/**\n * A Restate RPC service path. Example: `greeter`.\n */\ntype RestatePath = string;\n\nexport interface RestateInstanceRef {\n  readonly metaEndpoint: string;\n  readonly invokerRoleArn: string;\n  readonly authTokenSecretArn?: string;\n}\n\n/**\n * A collection of Lambda Restate RPC Service handlers.\n */\nexport type LambdaServiceRegistryProps = {\n  /**\n   * Mappings from service path to Lambda handler.\n   */\n  serviceHandlers: Record<RestatePath, lambda.Function>;\n\n  /**\n   * Custom resource provider token required for service discovery.\n   */\n  restate: RestateInstance;\n}\n\n/**\n * Represents a collection of Lambda-based Restate RPC services. This component is used to register\n * them with a single Restate instance. This creates a custom resource which will trigger service\n * discovery on any handler changes deployed through CDK/CloudFormation.\n */\nexport class LambdaServiceRegistry extends Construct {\n  private readonly serviceHandlers: Record<RestatePath, lambda.Function>;\n  private readonly registrationProviderToken: string;\n\n  constructor(scope: Construct, id: string, props: LambdaServiceRegistryProps) {\n    super(scope, id);\n\n    if (Object.values(props.serviceHandlers).length == 0) {\n      throw new Error(\"Please specify at least one service handler.\");\n    }\n\n    this.serviceHandlers = props.serviceHandlers;\n    this.registrationProviderToken = props.restate.registrationProviderToken.value;\n  }\n\n  public register(restate: RestateInstanceRef) {\n    const invokerRole = iam.Role.fromRoleArn(this, \"InvokerRole\", restate.invokerRoleArn);\n\n    const allowInvokeFunction = new iam.Policy(this, \"AllowInvokeFunction\", {\n      statements: [\n        new iam.PolicyStatement({\n          sid: \"AllowInvokeAnyFunctionVersion\",\n          actions: [\"lambda:InvokeFunction\"],\n          resources: Object.values(this.serviceHandlers)\n            .map(handler => handler.functionArn + \":*\"),\n        }),\n      ],\n    });\n\n    invokerRole.attachInlinePolicy(allowInvokeFunction);\n\n    for (const [path, handler] of Object.entries(this.serviceHandlers)) {\n      this.registerHandler(restate, { path, handler }, allowInvokeFunction);\n    }\n  }\n\n  private registerHandler(restate: RestateInstanceRef, service: {\n    path: RestatePath,\n    handler: lambda.Function\n  }, allowInvokeFunction: iam.Policy) {\n    const registrar = new RestateServiceRegistrar(this, service.handler.node.id + \"Discovery\", {\n      restate,\n      service,\n      serviceToken: this.registrationProviderToken,\n    });\n\n    // CloudFormation doesn't know that Restate depends on this role to call services; we must ensure that Lambda\n    // permission changes are applied before we can trigger discovery (represented by the registrar).\n    registrar.node.addDependency(allowInvokeFunction);\n  }\n}\n\nclass RestateServiceRegistrar extends Construct {\n  constructor(scope: Construct, id: string,\n              props: {\n                restate: RestateInstanceRef,\n                service: {\n                  path: RestatePath,\n                  handler: lambda.Function\n                },\n                serviceToken: string,\n              },\n  ) {\n    super(scope, id);\n\n    new cdk.CustomResource(this, props.service.handler.node.id + \"Discovery\", {\n      serviceToken: props.serviceToken,\n      resourceType: \"Custom::RestateServiceRegistrar\",\n      properties: {\n        servicePath: props.service.path,\n        metaEndpoint: props.restate.metaEndpoint,\n        authTokenSecretArn: props.restate.authTokenSecretArn,\n        serviceLambdaArn: props.service.handler.currentVersion.functionArn,\n        invokeRoleArn: props.restate.invokerRoleArn,\n        removalPolicy: cdk.RemovalPolicy.RETAIN,\n      } satisfies RegistrationProperties,\n    });\n  }\n}"]}
|
|
@@ -1,4 +1,3 @@
|
|
|
1
|
-
import { CloudFormationCustomResourceResponse } from "aws-lambda";
|
|
2
1
|
import { Handler } from "aws-lambda/handler";
|
|
3
2
|
import { CloudFormationCustomResourceEvent } from "aws-lambda/trigger/cloudformation-custom-resource";
|
|
4
3
|
import * as cdk from "aws-cdk-lib";
|
|
@@ -7,10 +6,12 @@ export interface RegistrationProperties {
|
|
|
7
6
|
metaEndpoint?: string;
|
|
8
7
|
serviceEndpoint?: string;
|
|
9
8
|
serviceLambdaArn?: string;
|
|
9
|
+
invokeRoleArn?: string;
|
|
10
10
|
removalPolicy?: cdk.RemovalPolicy;
|
|
11
|
+
authTokenSecretArn?: string;
|
|
11
12
|
}
|
|
12
13
|
/**
|
|
13
14
|
* Custom Resource event handler for Restate service registration. This handler backs the custom resources created by
|
|
14
|
-
* {@link
|
|
15
|
+
* {@link LambdaServiceRegistry} to facilitate Lambda service handler discovery.
|
|
15
16
|
*/
|
|
16
|
-
export declare const handler: Handler<CloudFormationCustomResourceEvent,
|
|
17
|
+
export declare const handler: Handler<CloudFormationCustomResourceEvent, void>;
|
|
@@ -9,15 +9,43 @@
|
|
|
9
9
|
* directory of this repository or package, or at
|
|
10
10
|
* https://github.com/restatedev/sdk-typescript/blob/main/LICENSE
|
|
11
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
|
+
};
|
|
12
35
|
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
13
36
|
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
14
37
|
};
|
|
15
38
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
16
39
|
exports.handler = void 0;
|
|
17
40
|
const node_fetch_1 = __importDefault(require("node-fetch"));
|
|
41
|
+
const client_secrets_manager_1 = require("@aws-sdk/client-secrets-manager");
|
|
42
|
+
const https = __importStar(require("https"));
|
|
43
|
+
const MAX_HEALTH_CHECK_ATTEMPTS = 3;
|
|
44
|
+
const MAX_REGISTRATION_ATTEMPTS = 3;
|
|
45
|
+
const INSECURE = true;
|
|
18
46
|
/**
|
|
19
47
|
* Custom Resource event handler for Restate service registration. This handler backs the custom resources created by
|
|
20
|
-
* {@link
|
|
48
|
+
* {@link LambdaServiceRegistry} to facilitate Lambda service handler discovery.
|
|
21
49
|
*/
|
|
22
50
|
const handler = async function (event) {
|
|
23
51
|
console.log({ event });
|
|
@@ -28,82 +56,125 @@ const handler = async function (event) {
|
|
|
28
56
|
// version from Lambda.
|
|
29
57
|
// const props = event.ResourceProperties as RegistrationProperties;
|
|
30
58
|
// if (props.removalPolicy === cdk.RemovalPolicy.DESTROY) {
|
|
59
|
+
// console.log(`De-registering service ${props.serviceLambdaArn}`);
|
|
31
60
|
// const controller = new AbortController();
|
|
32
61
|
// const id = btoa(props.serviceLambdaArn!); // TODO: we should be treating service ids as opaque
|
|
33
|
-
// const
|
|
34
|
-
//
|
|
35
|
-
//
|
|
36
|
-
//
|
|
37
|
-
// })
|
|
38
|
-
//
|
|
62
|
+
// const deleteCallTimeout = setTimeout(() => controller.abort("timeout"), 5_000);
|
|
63
|
+
// const deleteResponse = await fetch(`${props.metaEndpoint}/endpoints/${id}?force=true`, {
|
|
64
|
+
// signal: controller.signal,
|
|
65
|
+
// method: "DELETE",
|
|
66
|
+
// agent: INSECURE ? new https.Agent({ rejectUnauthorized: false }) : undefined,
|
|
67
|
+
// }).finally(() => clearTimeout(deleteCallTimeout));
|
|
68
|
+
//
|
|
39
69
|
// console.log(`Got delete response back: ${deleteResponse.status}`);
|
|
70
|
+
// if (deleteResponse.status != 202) {
|
|
71
|
+
// throw new Error(`Deleting service endpoint failed: ${deleteResponse.statusText} (${deleteResponse.status})`);
|
|
72
|
+
// }
|
|
40
73
|
// }
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
};
|
|
74
|
+
console.warn("De-registering services is not supported currently. Previous version will remain registered.");
|
|
75
|
+
return;
|
|
44
76
|
}
|
|
45
77
|
const props = event.ResourceProperties;
|
|
78
|
+
const authHeader = await createAuthHeader(props);
|
|
79
|
+
let attempt;
|
|
46
80
|
const controller = new AbortController();
|
|
47
|
-
const healthCheckTimeout = setTimeout(() => controller.abort("timeout"), 5000);
|
|
48
81
|
const healthCheckUrl = `${props.metaEndpoint}/health`;
|
|
49
82
|
console.log(`Performing health check against: ${healthCheckUrl}`);
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
83
|
+
attempt = 1;
|
|
84
|
+
while (true) {
|
|
85
|
+
const healthCheckTimeout = setTimeout(() => controller.abort("timeout"), 5000);
|
|
86
|
+
let healthResponse = undefined;
|
|
87
|
+
let errorMessage = undefined;
|
|
88
|
+
try {
|
|
89
|
+
healthResponse = await (0, node_fetch_1.default)(healthCheckUrl, {
|
|
90
|
+
signal: controller.signal,
|
|
91
|
+
headers: authHeader,
|
|
92
|
+
agent: INSECURE ? new https.Agent({ rejectUnauthorized: false }) : undefined,
|
|
93
|
+
}).finally(() => clearTimeout(healthCheckTimeout));
|
|
94
|
+
console.log(`Got health check response back: ${healthResponse.status}`);
|
|
95
|
+
if (healthResponse.status >= 200 && healthResponse.status < 300) {
|
|
96
|
+
break;
|
|
97
|
+
}
|
|
98
|
+
console.error(`Restate health check failed: ${healthResponse.statusText} (${healthResponse.status}; attempt ${attempt})`);
|
|
99
|
+
}
|
|
100
|
+
catch (e) {
|
|
101
|
+
errorMessage = e?.message;
|
|
102
|
+
console.error(`Restate health check failed: "${errorMessage}" (attempt ${attempt})`);
|
|
103
|
+
}
|
|
104
|
+
if (attempt > MAX_HEALTH_CHECK_ATTEMPTS) {
|
|
105
|
+
console.error(`Meta health check still failing after ${attempt} attempts.`);
|
|
106
|
+
throw new Error(errorMessage ?? `${healthResponse?.statusText} (${healthResponse?.status})`);
|
|
107
|
+
}
|
|
108
|
+
attempt += 1;
|
|
109
|
+
const waitTimeMillis = 2 ** attempt * 1000;
|
|
110
|
+
console.log(`Retrying after ${waitTimeMillis} ms...`);
|
|
111
|
+
await sleep(waitTimeMillis);
|
|
61
112
|
}
|
|
62
|
-
|
|
63
|
-
const
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
113
|
+
const endpointsUrl = `${props.metaEndpoint}/endpoints`;
|
|
114
|
+
const registrationRequest = JSON.stringify({
|
|
115
|
+
arn: props.serviceLambdaArn,
|
|
116
|
+
assume_role_arn: props.invokeRoleArn,
|
|
117
|
+
});
|
|
118
|
+
let failureReason;
|
|
119
|
+
console.log(`Triggering registration at ${endpointsUrl}: ${registrationRequest}`);
|
|
120
|
+
attempt = 1;
|
|
67
121
|
while (true) {
|
|
68
122
|
try {
|
|
69
|
-
const
|
|
123
|
+
const registerCallTimeout = setTimeout(() => controller.abort("timeout"), 10000);
|
|
124
|
+
const discoveryResponse = await (0, node_fetch_1.default)(endpointsUrl, {
|
|
70
125
|
signal: controller.signal,
|
|
71
126
|
method: "POST",
|
|
72
127
|
body: registrationRequest,
|
|
73
128
|
headers: {
|
|
74
129
|
"Content-Type": "application/json",
|
|
130
|
+
...authHeader,
|
|
75
131
|
},
|
|
76
|
-
|
|
77
|
-
|
|
132
|
+
agent: INSECURE ? new https.Agent({ rejectUnauthorized: false }) : undefined,
|
|
133
|
+
}).finally(() => clearTimeout(registerCallTimeout));
|
|
78
134
|
console.log(`Got registration response back: ${discoveryResponse.status}`);
|
|
79
135
|
if (discoveryResponse.status >= 200 && discoveryResponse.status < 300) {
|
|
80
|
-
const response = await discoveryResponse.json();
|
|
136
|
+
const response = (await discoveryResponse.json());
|
|
81
137
|
if (response?.services?.[0]?.name !== props.servicePath) {
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
138
|
+
failureReason =
|
|
139
|
+
"Restate service registration failed: service name indicated by service response" +
|
|
140
|
+
` ("${response?.services?.[0]?.name})) does not match the expected value ("${props.servicePath}")!`;
|
|
141
|
+
console.error(failureReason);
|
|
142
|
+
break; // don't throw immediately - let retry loop decide whether to abort
|
|
87
143
|
}
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
Status: "SUCCESS",
|
|
91
|
-
};
|
|
144
|
+
console.log("Success!");
|
|
145
|
+
return;
|
|
92
146
|
}
|
|
93
147
|
}
|
|
94
148
|
catch (e) {
|
|
95
|
-
console.
|
|
149
|
+
console.error(`Service registration call failed: ${e?.message} (attempt ${attempt})`);
|
|
96
150
|
}
|
|
97
|
-
attempt
|
|
98
|
-
if (attempt >= 3) {
|
|
151
|
+
if (attempt > MAX_REGISTRATION_ATTEMPTS) {
|
|
99
152
|
console.error(`Service registration failed after ${attempt} attempts.`);
|
|
100
153
|
break;
|
|
101
154
|
}
|
|
155
|
+
attempt += 1;
|
|
156
|
+
const waitTimeMillis = 2000 + 2 ** attempt * 1000; // 3s -> 6s -> 10s
|
|
157
|
+
console.log(`Retrying registration after ${waitTimeMillis} ms...`);
|
|
158
|
+
await sleep(waitTimeMillis);
|
|
102
159
|
}
|
|
103
|
-
|
|
104
|
-
Reason: `Restate service registration failed: ${healthResponse.statusText} (${healthResponse.status})`,
|
|
105
|
-
Status: "FAILED",
|
|
106
|
-
};
|
|
160
|
+
throw new Error(failureReason ?? "Restate service registration failed. Please see logs for details.");
|
|
107
161
|
};
|
|
108
162
|
exports.handler = handler;
|
|
109
|
-
//# 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 { CloudFormationCustomResourceResponse } from \"aws-lambda\";\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\";\n\nexport interface RegistrationProperties {\n  servicePath?: string;\n  metaEndpoint?: string;\n  serviceEndpoint?: string;\n  serviceLambdaArn?: string;\n  removalPolicy?: cdk.RemovalPolicy;\n}\n\ntype EndpointResponse = {\n  id?: string,\n  services?: { name?: string, revision?: number }[]\n}\n\n/**\n * Custom Resource event handler for Restate service registration. This handler backs the custom resources created by\n * {@link RestateLambdaServiceCollection} to facilitate Lambda service handler discovery.\n */\nexport const handler: Handler<CloudFormationCustomResourceEvent, Partial<CloudFormationCustomResourceResponse>> =\n  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      //   const controller = new AbortController();\n      //   const id = btoa(props.serviceLambdaArn!); // TODO: we should be treating service ids as opaque\n      //   const deleteResponse = await fetch(`${props.metaEndpoint}/endpoints/${id}?force=true`,\n      //     {\n      //       signal: controller.signal,\n      //       method: \"DELETE\",\n      //     })\n      //     .finally(() => clearTimeout(registerCallTimeout));\n      //   console.log(`Got delete response back: ${deleteResponse.status}`);\n      // }\n\n      return {\n        Status: \"SUCCESS\",\n      } satisfies Partial<CloudFormationCustomResourceResponse>;\n    }\n\n    const props = event.ResourceProperties as RegistrationProperties;\n\n    const controller = new AbortController();\n    const healthCheckTimeout = setTimeout(() => controller.abort(\"timeout\"), 5_000);\n    const healthCheckUrl = `${props.metaEndpoint}/health`;\n    console.log(`Performing health check against: ${healthCheckUrl}`);\n    const healthResponse = await fetch(healthCheckUrl,\n      {\n        signal: controller.signal,\n      })\n      .finally(() => clearTimeout(healthCheckTimeout));\n\n    console.log(`Got health check response back: ${healthResponse.status}`);\n    if (!(healthResponse.status >= 200 && healthResponse.status < 300)) {\n      console.error(`Restate health check failed: ${healthResponse.statusText} (${healthResponse.status})`);\n      return {\n        Reason: `Restate health check failed: ${healthResponse.statusText} (${healthResponse.status})`,\n        Status: \"FAILED\",\n      };\n    }\n\n    let attempt = 1;\n    const registerCallTimeout = setTimeout(() => controller.abort(\"timeout\"), 10_000);\n    const discoveryEndpointUrl = `${props.metaEndpoint}/endpoints`;\n    const registrationRequest = JSON.stringify({ arn: props.serviceLambdaArn });\n    console.log(`Triggering registration at ${discoveryEndpointUrl}: ${registrationRequest} (attempt ${attempt})`);\n    while (true) {\n      try {\n        const discoveryResponse = await fetch(discoveryEndpointUrl,\n          {\n            signal: controller.signal,\n            method: \"POST\",\n            body: registrationRequest,\n            headers: {\n              \"Content-Type\": \"application/json\",\n            },\n          })\n          .finally(() => clearTimeout(registerCallTimeout));\n\n        console.log(`Got registration response back: ${discoveryResponse.status}`);\n\n        if (discoveryResponse.status >= 200 && discoveryResponse.status < 300) {\n          const response = await discoveryResponse.json() as EndpointResponse;\n\n          if (response?.services?.[0]?.name !== props.servicePath) {\n            console.error(`Service registration failed: ${discoveryResponse.statusText} (${discoveryResponse.status})`);\n            return {\n              Reason: `Restate service registration failed: name returned by service (\"${response?.services?.[0]?.name})) does not match expected (\"${props.servicePath}\")`,\n              Status: \"FAILED\",\n            };\n          }\n\n          return {\n            Data: response,\n            Status: \"SUCCESS\",\n          } satisfies Partial<CloudFormationCustomResourceResponse>;\n        }\n      } catch (e) {\n        console.log(`Service registration call failed: ${(e as Error)?.message} (attempt ${attempt})`);\n      }\n\n      attempt += 1;\n      if (attempt >= 3) {\n        console.error(`Service registration failed after ${attempt} attempts.`);\n        break;\n      }\n    }\n\n    return {\n      Reason: `Restate service registration failed: ${healthResponse.statusText} (${healthResponse.status})`,\n      Status: \"FAILED\",\n    };\n  };"]}
|
|
163
|
+
async function createAuthHeader(props) {
|
|
164
|
+
if (!props.authTokenSecretArn) {
|
|
165
|
+
return {};
|
|
166
|
+
}
|
|
167
|
+
console.log(`Using bearer authentication token from secret ${props.authTokenSecretArn}`);
|
|
168
|
+
const ssm = new client_secrets_manager_1.SecretsManagerClient();
|
|
169
|
+
const response = await ssm.send(new client_secrets_manager_1.GetSecretValueCommand({
|
|
170
|
+
SecretId: props.authTokenSecretArn,
|
|
171
|
+
}));
|
|
172
|
+
console.log(`Successfully retrieved secret "${response.Name}" version ${response.VersionId}`);
|
|
173
|
+
return {
|
|
174
|
+
Authorization: `Bearer ${response.SecretString}`,
|
|
175
|
+
};
|
|
176
|
+
}
|
|
177
|
+
async function sleep(millis) {
|
|
178
|
+
await new Promise((resolve) => setTimeout(resolve, millis));
|
|
179
|
+
}
|
|
180
|
+
//# 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\";\n\nexport interface RegistrationProperties {\n  servicePath?: string;\n  metaEndpoint?: string;\n  serviceEndpoint?: string;\n  serviceLambdaArn?: string;\n  invokeRoleArn?: string;\n  removalPolicy?: cdk.RemovalPolicy;\n  authTokenSecretArn?: string;\n}\n\ntype EndpointResponse = {\n  id?: string;\n  services?: { name?: string; revision?: number }[];\n};\n\nconst MAX_HEALTH_CHECK_ATTEMPTS = 3;\nconst MAX_REGISTRATION_ATTEMPTS = 3;\n\nconst INSECURE = true;\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.metaEndpoint}/endpoints/${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(`Deleting service endpoint 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.metaEndpoint}/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(`Meta health check still failing after ${attempt} attempts.`);\n      throw new Error(errorMessage ?? `${healthResponse?.statusText} (${healthResponse?.status})`);\n    }\n    attempt += 1;\n\n    const waitTimeMillis = 2 ** attempt * 1_000;\n    console.log(`Retrying after ${waitTimeMillis} ms...`);\n    await sleep(waitTimeMillis);\n  }\n\n  const endpointsUrl = `${props.metaEndpoint}/endpoints`;\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 ${endpointsUrl}: ${registrationRequest}`);\n  attempt = 1;\n  while (true) {\n    try {\n      const registerCallTimeout = setTimeout(() => controller.abort(\"timeout\"), 10_000);\n      const discoveryResponse = await fetch(endpointsUrl, {\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: ${discoveryResponse.status}`);\n\n      if (discoveryResponse.status >= 200 && discoveryResponse.status < 300) {\n        const response = (await discoveryResponse.json()) as EndpointResponse;\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    }\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 = 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"]}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import { Construct } from "constructs";
|
|
2
|
+
import * as ssm from "aws-cdk-lib/aws-secretsmanager";
|
|
3
|
+
import * as cdk from "aws-cdk-lib";
|
|
4
|
+
import * as ec2 from "aws-cdk-lib/aws-ec2";
|
|
5
|
+
export declare class RegistrationProvider extends Construct {
|
|
6
|
+
readonly serviceToken: string;
|
|
7
|
+
constructor(scope: Construct, id: string, props: {
|
|
8
|
+
authToken?: ssm.ISecret;
|
|
9
|
+
timeout?: cdk.Duration;
|
|
10
|
+
vpc?: ec2.Vpc;
|
|
11
|
+
});
|
|
12
|
+
}
|
|
@@ -0,0 +1,77 @@
|
|
|
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
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
36
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
37
|
+
};
|
|
38
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
39
|
+
exports.RegistrationProvider = void 0;
|
|
40
|
+
const constructs_1 = require("constructs");
|
|
41
|
+
const lambda_node = __importStar(require("aws-cdk-lib/aws-lambda-nodejs"));
|
|
42
|
+
const node_path_1 = __importDefault(require("node:path"));
|
|
43
|
+
const lambda = __importStar(require("aws-cdk-lib/aws-lambda"));
|
|
44
|
+
const cdk = __importStar(require("aws-cdk-lib"));
|
|
45
|
+
const cr = __importStar(require("aws-cdk-lib/custom-resources"));
|
|
46
|
+
const DEFAULT_TIMEOUT = cdk.Duration.seconds(120);
|
|
47
|
+
class RegistrationProvider extends constructs_1.Construct {
|
|
48
|
+
constructor(scope, id, props) {
|
|
49
|
+
super(scope, id);
|
|
50
|
+
if (props.vpc) {
|
|
51
|
+
console.log("Using VPC!");
|
|
52
|
+
}
|
|
53
|
+
const registrationHandler = new lambda_node.NodejsFunction(this, "RegistrationHandler", {
|
|
54
|
+
description: "Restate custom registration handler",
|
|
55
|
+
entry: node_path_1.default.join(__dirname, "register-service-handler/index.js"),
|
|
56
|
+
architecture: lambda.Architecture.ARM_64,
|
|
57
|
+
runtime: lambda.Runtime.NODEJS_LATEST,
|
|
58
|
+
memorySize: 128,
|
|
59
|
+
timeout: props.timeout ?? DEFAULT_TIMEOUT,
|
|
60
|
+
environment: {
|
|
61
|
+
NODE_OPTIONS: "--enable-source-maps",
|
|
62
|
+
},
|
|
63
|
+
bundling: {
|
|
64
|
+
minify: false,
|
|
65
|
+
sourceMap: true,
|
|
66
|
+
},
|
|
67
|
+
...(props.vpc ? { vpc: props.vpc, subnets: props.vpc.privateSubnets } : {}),
|
|
68
|
+
});
|
|
69
|
+
props.authToken?.grantRead(registrationHandler);
|
|
70
|
+
const registrationProvider = new cr.Provider(this, "RegistrationProvider", {
|
|
71
|
+
onEventHandler: registrationHandler,
|
|
72
|
+
});
|
|
73
|
+
this.serviceToken = registrationProvider.serviceToken;
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
exports.RegistrationProvider = RegistrationProvider;
|
|
77
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVnaXN0cmF0aW9uLXByb3ZpZGVyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vbGliL3Jlc3RhdGUtY29uc3RydWN0cy9yZWdpc3RyYXRpb24tcHJvdmlkZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7Ozs7Ozs7R0FTRzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFFSCwyQ0FBdUM7QUFFdkMsMkVBQTZEO0FBQzdELDBEQUE2QjtBQUM3QiwrREFBaUQ7QUFDakQsaURBQW1DO0FBQ25DLGlFQUFtRDtBQUduRCxNQUFNLGVBQWUsR0FBRyxHQUFHLENBQUMsUUFBUSxDQUFDLE9BQU8sQ0FBQyxHQUFHLENBQUMsQ0FBQztBQUVsRCxNQUFhLG9CQUFxQixTQUFRLHNCQUFTO0lBR2pELFlBQVksS0FBZ0IsRUFBRSxFQUFVLEVBQUUsS0FBeUU7UUFDakgsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsQ0FBQztRQUVqQixJQUFJLEtBQUssQ0FBQyxHQUFHLEVBQUU7WUFDYixPQUFPLENBQUMsR0FBRyxDQUFDLFlBQVksQ0FBQyxDQUFDO1NBQzNCO1FBRUQsTUFBTSxtQkFBbUIsR0FBRyxJQUFJLFdBQVcsQ0FBQyxjQUFjLENBQUMsSUFBSSxFQUFFLHFCQUFxQixFQUFFO1lBQ3RGLFdBQVcsRUFBRSxxQ0FBcUM7WUFDbEQsS0FBSyxFQUFFLG1CQUFJLENBQUMsSUFBSSxDQUFDLFNBQVMsRUFBRSxtQ0FBbUMsQ0FBQztZQUNoRSxZQUFZLEVBQUUsTUFBTSxDQUFDLFlBQVksQ0FBQyxNQUFNO1lBQ3hDLE9BQU8sRUFBRSxNQUFNLENBQUMsT0FBTyxDQUFDLGFBQWE7WUFDckMsVUFBVSxFQUFFLEdBQUc7WUFDZixPQUFPLEVBQUUsS0FBSyxDQUFDLE9BQU8sSUFBSSxlQUFlO1lBQ3pDLFdBQVcsRUFBRTtnQkFDWCxZQUFZLEVBQUUsc0JBQXNCO2FBQ3JDO1lBQ0QsUUFBUSxFQUFFO2dCQUNSLE1BQU0sRUFBRSxLQUFLO2dCQUNiLFNBQVMsRUFBRSxJQUFJO2FBQ2hCO1lBQ0QsR0FBRyxDQUFDLEtBQUssQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLEVBQUUsR0FBRyxFQUFFLEtBQUssQ0FBQyxHQUFHLEVBQUUsT0FBTyxFQUFFLEtBQUssQ0FBQyxHQUFHLENBQUMsY0FBYyxFQUFFLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQztTQUM1RSxDQUFDLENBQUM7UUFDSCxLQUFLLENBQUMsU0FBUyxFQUFFLFNBQVMsQ0FBQyxtQkFBbUIsQ0FBQyxDQUFDO1FBRWhELE1BQU0sb0JBQW9CLEdBQUcsSUFBSSxFQUFFLENBQUMsUUFBUSxDQUFDLElBQUksRUFBRSxzQkFBc0IsRUFBRTtZQUN6RSxjQUFjLEVBQUUsbUJBQW1CO1NBQ3BDLENBQUMsQ0FBQztRQUNILElBQUksQ0FBQyxZQUFZLEdBQUcsb0JBQW9CLENBQUMsWUFBWSxDQUFDO0lBQ3hELENBQUM7Q0FDRjtBQWpDRCxvREFpQ0MiLCJzb3VyY2VzQ29udGVudCI6WyIvKlxuICogQ29weXJpZ2h0IChjKSAyMDIzIC0gUmVzdGF0ZSBTb2Z0d2FyZSwgSW5jLiwgUmVzdGF0ZSBHbWJIXG4gKlxuICogVGhpcyBmaWxlIGlzIHBhcnQgb2YgdGhlIFJlc3RhdGUgU0RLIGZvciBOb2RlLmpzL1R5cGVTY3JpcHQsXG4gKiB3aGljaCBpcyByZWxlYXNlZCB1bmRlciB0aGUgTUlUIGxpY2Vuc2UuXG4gKlxuICogWW91IGNhbiBmaW5kIGEgY29weSBvZiB0aGUgbGljZW5zZSBpbiBmaWxlIExJQ0VOU0UgaW4gdGhlIHJvb3RcbiAqIGRpcmVjdG9yeSBvZiB0aGlzIHJlcG9zaXRvcnkgb3IgcGFja2FnZSwgb3IgYXRcbiAqIGh0dHBzOi8vZ2l0aHViLmNvbS9yZXN0YXRlZGV2L3Nkay10eXBlc2NyaXB0L2Jsb2IvbWFpbi9MSUNFTlNFXG4gKi9cblxuaW1wb3J0IHsgQ29uc3RydWN0IH0gZnJvbSBcImNvbnN0cnVjdHNcIjtcbmltcG9ydCAqIGFzIHNzbSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLXNlY3JldHNtYW5hZ2VyXCI7XG5pbXBvcnQgKiBhcyBsYW1iZGFfbm9kZSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWxhbWJkYS1ub2RlanNcIjtcbmltcG9ydCBwYXRoIGZyb20gXCJub2RlOnBhdGhcIjtcbmltcG9ydCAqIGFzIGxhbWJkYSBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWxhbWJkYVwiO1xuaW1wb3J0ICogYXMgY2RrIGZyb20gXCJhd3MtY2RrLWxpYlwiO1xuaW1wb3J0ICogYXMgY3IgZnJvbSBcImF3cy1jZGstbGliL2N1c3RvbS1yZXNvdXJjZXNcIjtcbmltcG9ydCAqIGFzIGVjMiBmcm9tIFwiYXdzLWNkay1saWIvYXdzLWVjMlwiO1xuXG5jb25zdCBERUZBVUxUX1RJTUVPVVQgPSBjZGsuRHVyYXRpb24uc2Vjb25kcygxMjApO1xuXG5leHBvcnQgY2xhc3MgUmVnaXN0cmF0aW9uUHJvdmlkZXIgZXh0ZW5kcyBDb25zdHJ1Y3Qge1xuICByZWFkb25seSBzZXJ2aWNlVG9rZW46IHN0cmluZztcblxuICBjb25zdHJ1Y3RvcihzY29wZTogQ29uc3RydWN0LCBpZDogc3RyaW5nLCBwcm9wczogeyBhdXRoVG9rZW4/OiBzc20uSVNlY3JldDsgdGltZW91dD86IGNkay5EdXJhdGlvbjsgdnBjPzogZWMyLlZwYyB9KSB7XG4gICAgc3VwZXIoc2NvcGUsIGlkKTtcblxuICAgIGlmIChwcm9wcy52cGMpIHtcbiAgICAgIGNvbnNvbGUubG9nKFwiVXNpbmcgVlBDIVwiKTtcbiAgICB9XG5cbiAgICBjb25zdCByZWdpc3RyYXRpb25IYW5kbGVyID0gbmV3IGxhbWJkYV9ub2RlLk5vZGVqc0Z1bmN0aW9uKHRoaXMsIFwiUmVnaXN0cmF0aW9uSGFuZGxlclwiLCB7XG4gICAgICBkZXNjcmlwdGlvbjogXCJSZXN0YXRlIGN1c3RvbSByZWdpc3RyYXRpb24gaGFuZGxlclwiLFxuICAgICAgZW50cnk6IHBhdGguam9pbihfX2Rpcm5hbWUsIFwicmVnaXN0ZXItc2VydmljZS1oYW5kbGVyL2luZGV4LmpzXCIpLFxuICAgICAgYXJjaGl0ZWN0dXJlOiBsYW1iZGEuQXJjaGl0ZWN0dXJlLkFSTV82NCxcbiAgICAgIHJ1bnRpbWU6IGxhbWJkYS5SdW50aW1lLk5PREVKU19MQVRFU1QsXG4gICAgICBtZW1vcnlTaXplOiAxMjgsXG4gICAgICB0aW1lb3V0OiBwcm9wcy50aW1lb3V0ID8/IERFRkFVTFRfVElNRU9VVCxcbiAgICAgIGVudmlyb25tZW50OiB7XG4gICAgICAgIE5PREVfT1BUSU9OUzogXCItLWVuYWJsZS1zb3VyY2UtbWFwc1wiLFxuICAgICAgfSxcbiAgICAgIGJ1bmRsaW5nOiB7XG4gICAgICAgIG1pbmlmeTogZmFsc2UsXG4gICAgICAgIHNvdXJjZU1hcDogdHJ1ZSxcbiAgICAgIH0sXG4gICAgICAuLi4ocHJvcHMudnBjID8geyB2cGM6IHByb3BzLnZwYywgc3VibmV0czogcHJvcHMudnBjLnByaXZhdGVTdWJuZXRzIH0gOiB7fSksXG4gICAgfSk7XG4gICAgcHJvcHMuYXV0aFRva2VuPy5ncmFudFJlYWQocmVnaXN0cmF0aW9uSGFuZGxlcik7XG5cbiAgICBjb25zdCByZWdpc3RyYXRpb25Qcm92aWRlciA9IG5ldyBjci5Qcm92aWRlcih0aGlzLCBcIlJlZ2lzdHJhdGlvblByb3ZpZGVyXCIsIHtcbiAgICAgIG9uRXZlbnRIYW5kbGVyOiByZWdpc3RyYXRpb25IYW5kbGVyLFxuICAgIH0pO1xuICAgIHRoaXMuc2VydmljZVRva2VuID0gcmVnaXN0cmF0aW9uUHJvdmlkZXIuc2VydmljZVRva2VuO1xuICB9XG59XG4iXX0=
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
import { Construct } from "constructs";
|
|
2
|
+
import * as cdk from "aws-cdk-lib";
|
|
3
|
+
import * as iam from "aws-cdk-lib/aws-iam";
|
|
4
|
+
import * as ssm from "aws-cdk-lib/aws-secretsmanager";
|
|
5
|
+
import { RestateInstance } from "./restate-instance";
|
|
6
|
+
export interface ManagedRestateProps {
|
|
7
|
+
/** Prefix for resources created by this construct that require unique names. */
|
|
8
|
+
prefix?: string;
|
|
9
|
+
/** ID of the Restate service cluster to which this service will be registered. */
|
|
10
|
+
clusterId: string;
|
|
11
|
+
/** Auth token to use with Restate cluster. Used to authenticate access to the meta endpoint for registration. */
|
|
12
|
+
authTokenSecretArn: string;
|
|
13
|
+
}
|
|
14
|
+
/**
|
|
15
|
+
* Models a Restate managed service cluster provided to the application. In the case of a managed service, this
|
|
16
|
+
* construct only creates an appropriately configured registration provider custom component for use by the service
|
|
17
|
+
* registry elsewhere, and creates the role assumed by the cluster. An appropriate trust policy will be added to this
|
|
18
|
+
* role that allows Restate to assume it from outside the deployment AWS account.
|
|
19
|
+
*/
|
|
20
|
+
export declare class RestateCloudEndpoint extends Construct implements RestateInstance {
|
|
21
|
+
readonly invokerRole: iam.Role;
|
|
22
|
+
readonly ingressEndpoint: string;
|
|
23
|
+
readonly metaEndpoint: string;
|
|
24
|
+
readonly authToken: ssm.ISecret;
|
|
25
|
+
readonly registrationProviderToken: cdk.CfnOutput;
|
|
26
|
+
constructor(scope: Construct, id: string, props: ManagedRestateProps);
|
|
27
|
+
}
|
|
@@ -0,0 +1,70 @@
|
|
|
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.RestateCloudEndpoint = void 0;
|
|
37
|
+
const constructs_1 = require("constructs");
|
|
38
|
+
const cdk = __importStar(require("aws-cdk-lib"));
|
|
39
|
+
const iam = __importStar(require("aws-cdk-lib/aws-iam"));
|
|
40
|
+
const ssm = __importStar(require("aws-cdk-lib/aws-secretsmanager"));
|
|
41
|
+
const registration_provider_1 = require("./registration-provider");
|
|
42
|
+
const RESTATE_INGRESS_PORT = 8080;
|
|
43
|
+
const RESTATE_META_PORT = 9070;
|
|
44
|
+
/**
|
|
45
|
+
* Models a Restate managed service cluster provided to the application. In the case of a managed service, this
|
|
46
|
+
* construct only creates an appropriately configured registration provider custom component for use by the service
|
|
47
|
+
* registry elsewhere, and creates the role assumed by the cluster. An appropriate trust policy will be added to this
|
|
48
|
+
* role that allows Restate to assume it from outside the deployment AWS account.
|
|
49
|
+
*/
|
|
50
|
+
class RestateCloudEndpoint extends constructs_1.Construct {
|
|
51
|
+
constructor(scope, id, props) {
|
|
52
|
+
super(scope, id);
|
|
53
|
+
this.invokerRole = new iam.Role(this, "ManagedServiceRole", {
|
|
54
|
+
description: "Role assumed by the Restate managed service to invoke our services",
|
|
55
|
+
assumedBy: new iam.ArnPrincipal("arn:aws:iam::663487780041:role/restate-dev"),
|
|
56
|
+
externalIds: [props.clusterId],
|
|
57
|
+
});
|
|
58
|
+
this.ingressEndpoint = `https://${props.clusterId}.dev.restate.cloud:${RESTATE_INGRESS_PORT}`;
|
|
59
|
+
this.metaEndpoint = `https://${props.clusterId}.dev.restate.cloud:${RESTATE_META_PORT}`;
|
|
60
|
+
this.authToken = ssm.Secret.fromSecretCompleteArn(this, "ClusterAuthToken", props.authTokenSecretArn);
|
|
61
|
+
const registrationProvider = new registration_provider_1.RegistrationProvider(this, "RegistrationProvider", { authToken: this.authToken });
|
|
62
|
+
this.registrationProviderToken = new cdk.CfnOutput(this, "RegistrationProviderToken", {
|
|
63
|
+
description: "Restate service registration provider custom component token used by registry to perform discovery",
|
|
64
|
+
exportName: [props.prefix, "RegistrationProviderToken"].filter(Boolean).join("-"),
|
|
65
|
+
value: registrationProvider.serviceToken,
|
|
66
|
+
});
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
exports.RestateCloudEndpoint = RestateCloudEndpoint;
|
|
70
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVzdGF0ZS1jbG91ZC1lbmRwb2ludC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uL2xpYi9yZXN0YXRlLWNvbnN0cnVjdHMvcmVzdGF0ZS1jbG91ZC1lbmRwb2ludC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUE7Ozs7Ozs7OztHQVNHOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUVILDJDQUF1QztBQUN2QyxpREFBbUM7QUFDbkMseURBQTJDO0FBQzNDLG9FQUFzRDtBQUV0RCxtRUFBK0Q7QUFFL0QsTUFBTSxvQkFBb0IsR0FBRyxJQUFJLENBQUM7QUFDbEMsTUFBTSxpQkFBaUIsR0FBRyxJQUFJLENBQUM7QUFhL0I7Ozs7O0dBS0c7QUFDSCxNQUFhLG9CQUFxQixTQUFRLHNCQUFTO0lBT2pELFlBQVksS0FBZ0IsRUFBRSxFQUFVLEVBQUUsS0FBMEI7UUFDbEUsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsQ0FBQztRQUVqQixJQUFJLENBQUMsV0FBVyxHQUFHLElBQUksR0FBRyxDQUFDLElBQUksQ0FBQyxJQUFJLEVBQUUsb0JBQW9CLEVBQUU7WUFDMUQsV0FBVyxFQUFFLG9FQUFvRTtZQUNqRixTQUFTLEVBQUUsSUFBSSxHQUFHLENBQUMsWUFBWSxDQUFDLDRDQUE0QyxDQUFDO1lBQzdFLFdBQVcsRUFBRSxDQUFDLEtBQUssQ0FBQyxTQUFTLENBQUM7U0FDL0IsQ0FBQyxDQUFDO1FBRUgsSUFBSSxDQUFDLGVBQWUsR0FBRyxXQUFXLEtBQUssQ0FBQyxTQUFTLHNCQUFzQixvQkFBb0IsRUFBRSxDQUFDO1FBQzlGLElBQUksQ0FBQyxZQUFZLEdBQUcsV0FBVyxLQUFLLENBQUMsU0FBUyxzQkFBc0IsaUJBQWlCLEVBQUUsQ0FBQztRQUN4RixJQUFJLENBQUMsU0FBUyxHQUFHLEdBQUcsQ0FBQyxNQUFNLENBQUMscUJBQXFCLENBQUMsSUFBSSxFQUFFLGtCQUFrQixFQUFFLEtBQUssQ0FBQyxrQkFBa0IsQ0FBQyxDQUFDO1FBRXRHLE1BQU0sb0JBQW9CLEdBQUcsSUFBSSw0Q0FBb0IsQ0FBQyxJQUFJLEVBQUUsc0JBQXNCLEVBQUUsRUFBRSxTQUFTLEVBQUUsSUFBSSxDQUFDLFNBQVMsRUFBRSxDQUFDLENBQUM7UUFDbkgsSUFBSSxDQUFDLHlCQUF5QixHQUFHLElBQUksR0FBRyxDQUFDLFNBQVMsQ0FBQyxJQUFJLEVBQUUsMkJBQTJCLEVBQUU7WUFDcEYsV0FBVyxFQUFFLG9HQUFvRztZQUNqSCxVQUFVLEVBQUUsQ0FBQyxLQUFLLENBQUMsTUFBTSxFQUFFLDJCQUEyQixDQUFDLENBQUMsTUFBTSxDQUFDLE9BQU8sQ0FBQyxDQUFDLElBQUksQ0FBQyxHQUFHLENBQUM7WUFDakYsS0FBSyxFQUFFLG9CQUFvQixDQUFDLFlBQVk7U0FDekMsQ0FBQyxDQUFDO0lBQ0wsQ0FBQztDQUNGO0FBM0JELG9EQTJCQyIsInNvdXJjZXNDb250ZW50IjpbIi8qXG4gKiBDb3B5cmlnaHQgKGMpIDIwMjMgLSBSZXN0YXRlIFNvZnR3YXJlLCBJbmMuLCBSZXN0YXRlIEdtYkhcbiAqXG4gKiBUaGlzIGZpbGUgaXMgcGFydCBvZiB0aGUgUmVzdGF0ZSBTREsgZm9yIE5vZGUuanMvVHlwZVNjcmlwdCxcbiAqIHdoaWNoIGlzIHJlbGVhc2VkIHVuZGVyIHRoZSBNSVQgbGljZW5zZS5cbiAqXG4gKiBZb3UgY2FuIGZpbmQgYSBjb3B5IG9mIHRoZSBsaWNlbnNlIGluIGZpbGUgTElDRU5TRSBpbiB0aGUgcm9vdFxuICogZGlyZWN0b3J5IG9mIHRoaXMgcmVwb3NpdG9yeSBvciBwYWNrYWdlLCBvciBhdFxuICogaHR0cHM6Ly9naXRodWIuY29tL3Jlc3RhdGVkZXYvc2RrLXR5cGVzY3JpcHQvYmxvYi9tYWluL0xJQ0VOU0VcbiAqL1xuXG5pbXBvcnQgeyBDb25zdHJ1Y3QgfSBmcm9tIFwiY29uc3RydWN0c1wiO1xuaW1wb3J0ICogYXMgY2RrIGZyb20gXCJhd3MtY2RrLWxpYlwiO1xuaW1wb3J0ICogYXMgaWFtIGZyb20gXCJhd3MtY2RrLWxpYi9hd3MtaWFtXCI7XG5pbXBvcnQgKiBhcyBzc20gZnJvbSBcImF3cy1jZGstbGliL2F3cy1zZWNyZXRzbWFuYWdlclwiO1xuaW1wb3J0IHsgUmVzdGF0ZUluc3RhbmNlIH0gZnJvbSBcIi4vcmVzdGF0ZS1pbnN0YW5jZVwiO1xuaW1wb3J0IHsgUmVnaXN0cmF0aW9uUHJvdmlkZXIgfSBmcm9tIFwiLi9yZWdpc3RyYXRpb24tcHJvdmlkZXJcIjtcblxuY29uc3QgUkVTVEFURV9JTkdSRVNTX1BPUlQgPSA4MDgwO1xuY29uc3QgUkVTVEFURV9NRVRBX1BPUlQgPSA5MDcwO1xuXG5leHBvcnQgaW50ZXJmYWNlIE1hbmFnZWRSZXN0YXRlUHJvcHMge1xuICAvKiogUHJlZml4IGZvciByZXNvdXJjZXMgY3JlYXRlZCBieSB0aGlzIGNvbnN0cnVjdCB0aGF0IHJlcXVpcmUgdW5pcXVlIG5hbWVzLiAqL1xuICBwcmVmaXg/OiBzdHJpbmc7XG5cbiAgLyoqIElEIG9mIHRoZSBSZXN0YXRlIHNlcnZpY2UgY2x1c3RlciB0byB3aGljaCB0aGlzIHNlcnZpY2Ugd2lsbCBiZSByZWdpc3RlcmVkLiAqL1xuICBjbHVzdGVySWQ6IHN0cmluZztcblxuICAvKiogQXV0aCB0b2tlbiB0byB1c2Ugd2l0aCBSZXN0YXRlIGNsdXN0ZXIuIFVzZWQgdG8gYXV0aGVudGljYXRlIGFjY2VzcyB0byB0aGUgbWV0YSBlbmRwb2ludCBmb3IgcmVnaXN0cmF0aW9uLiAqL1xuICBhdXRoVG9rZW5TZWNyZXRBcm46IHN0cmluZztcbn1cblxuLyoqXG4gKiBNb2RlbHMgYSBSZXN0YXRlIG1hbmFnZWQgc2VydmljZSBjbHVzdGVyIHByb3ZpZGVkIHRvIHRoZSBhcHBsaWNhdGlvbi4gSW4gdGhlIGNhc2Ugb2YgYSBtYW5hZ2VkIHNlcnZpY2UsIHRoaXNcbiAqIGNvbnN0cnVjdCBvbmx5IGNyZWF0ZXMgYW4gYXBwcm9wcmlhdGVseSBjb25maWd1cmVkIHJlZ2lzdHJhdGlvbiBwcm92aWRlciBjdXN0b20gY29tcG9uZW50IGZvciB1c2UgYnkgdGhlIHNlcnZpY2VcbiAqIHJlZ2lzdHJ5IGVsc2V3aGVyZSwgYW5kIGNyZWF0ZXMgdGhlIHJvbGUgYXNzdW1lZCBieSB0aGUgY2x1c3Rlci4gQW4gYXBwcm9wcmlhdGUgdHJ1c3QgcG9saWN5IHdpbGwgYmUgYWRkZWQgdG8gdGhpc1xuICogcm9sZSB0aGF0IGFsbG93cyBSZXN0YXRlIHRvIGFzc3VtZSBpdCBmcm9tIG91dHNpZGUgdGhlIGRlcGxveW1lbnQgQVdTIGFjY291bnQuXG4gKi9cbmV4cG9ydCBjbGFzcyBSZXN0YXRlQ2xvdWRFbmRwb2ludCBleHRlbmRzIENvbnN0cnVjdCBpbXBsZW1lbnRzIFJlc3RhdGVJbnN0YW5jZSB7XG4gIHJlYWRvbmx5IGludm9rZXJSb2xlOiBpYW0uUm9sZTtcbiAgcmVhZG9ubHkgaW5ncmVzc0VuZHBvaW50OiBzdHJpbmc7XG4gIHJlYWRvbmx5IG1ldGFFbmRwb2ludDogc3RyaW5nO1xuICByZWFkb25seSBhdXRoVG9rZW46IHNzbS5JU2VjcmV0O1xuICByZWFkb25seSByZWdpc3RyYXRpb25Qcm92aWRlclRva2VuOiBjZGsuQ2ZuT3V0cHV0O1xuXG4gIGNvbnN0cnVjdG9yKHNjb3BlOiBDb25zdHJ1Y3QsIGlkOiBzdHJpbmcsIHByb3BzOiBNYW5hZ2VkUmVzdGF0ZVByb3BzKSB7XG4gICAgc3VwZXIoc2NvcGUsIGlkKTtcblxuICAgIHRoaXMuaW52b2tlclJvbGUgPSBuZXcgaWFtLlJvbGUodGhpcywgXCJNYW5hZ2VkU2VydmljZVJvbGVcIiwge1xuICAgICAgZGVzY3JpcHRpb246IFwiUm9sZSBhc3N1bWVkIGJ5IHRoZSBSZXN0YXRlIG1hbmFnZWQgc2VydmljZSB0byBpbnZva2Ugb3VyIHNlcnZpY2VzXCIsXG4gICAgICBhc3N1bWVkQnk6IG5ldyBpYW0uQXJuUHJpbmNpcGFsKFwiYXJuOmF3czppYW06OjY2MzQ4Nzc4MDA0MTpyb2xlL3Jlc3RhdGUtZGV2XCIpLFxuICAgICAgZXh0ZXJuYWxJZHM6IFtwcm9wcy5jbHVzdGVySWRdLFxuICAgIH0pO1xuXG4gICAgdGhpcy5pbmdyZXNzRW5kcG9pbnQgPSBgaHR0cHM6Ly8ke3Byb3BzLmNsdXN0ZXJJZH0uZGV2LnJlc3RhdGUuY2xvdWQ6JHtSRVNUQVRFX0lOR1JFU1NfUE9SVH1gO1xuICAgIHRoaXMubWV0YUVuZHBvaW50ID0gYGh0dHBzOi8vJHtwcm9wcy5jbHVzdGVySWR9LmRldi5yZXN0YXRlLmNsb3VkOiR7UkVTVEFURV9NRVRBX1BPUlR9YDtcbiAgICB0aGlzLmF1dGhUb2tlbiA9IHNzbS5TZWNyZXQuZnJvbVNlY3JldENvbXBsZXRlQXJuKHRoaXMsIFwiQ2x1c3RlckF1dGhUb2tlblwiLCBwcm9wcy5hdXRoVG9rZW5TZWNyZXRBcm4pO1xuXG4gICAgY29uc3QgcmVnaXN0cmF0aW9uUHJvdmlkZXIgPSBuZXcgUmVnaXN0cmF0aW9uUHJvdmlkZXIodGhpcywgXCJSZWdpc3RyYXRpb25Qcm92aWRlclwiLCB7IGF1dGhUb2tlbjogdGhpcy5hdXRoVG9rZW4gfSk7XG4gICAgdGhpcy5yZWdpc3RyYXRpb25Qcm92aWRlclRva2VuID0gbmV3IGNkay5DZm5PdXRwdXQodGhpcywgXCJSZWdpc3RyYXRpb25Qcm92aWRlclRva2VuXCIsIHtcbiAgICAgIGRlc2NyaXB0aW9uOiBcIlJlc3RhdGUgc2VydmljZSByZWdpc3RyYXRpb24gcHJvdmlkZXIgY3VzdG9tIGNvbXBvbmVudCB0b2tlbiB1c2VkIGJ5IHJlZ2lzdHJ5IHRvIHBlcmZvcm0gZGlzY292ZXJ5XCIsXG4gICAgICBleHBvcnROYW1lOiBbcHJvcHMucHJlZml4LCBcIlJlZ2lzdHJhdGlvblByb3ZpZGVyVG9rZW5cIl0uZmlsdGVyKEJvb2xlYW4pLmpvaW4oXCItXCIpLFxuICAgICAgdmFsdWU6IHJlZ2lzdHJhdGlvblByb3ZpZGVyLnNlcnZpY2VUb2tlbixcbiAgICB9KTtcbiAgfVxufSJdfQ==
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import * as iam from "aws-cdk-lib/aws-iam";
|
|
2
|
+
import * as cdk from "aws-cdk-lib";
|
|
3
|
+
import * as ssm from "aws-cdk-lib/aws-secretsmanager";
|
|
4
|
+
/**
|
|
5
|
+
* Represents an instance of the Restate service. This could represent a self-hosted broker, or Restate's managed
|
|
6
|
+
* service.
|
|
7
|
+
*/
|
|
8
|
+
export interface RestateInstance {
|
|
9
|
+
readonly invokerRole: iam.IRole;
|
|
10
|
+
readonly metaEndpoint: string;
|
|
11
|
+
readonly authToken?: ssm.ISecret;
|
|
12
|
+
readonly registrationProviderToken: cdk.CfnOutput;
|
|
13
|
+
}
|
|
@@ -0,0 +1,3 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVzdGF0ZS1pbnN0YW5jZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uL2xpYi9yZXN0YXRlLWNvbnN0cnVjdHMvcmVzdGF0ZS1pbnN0YW5jZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgaWFtIGZyb20gXCJhd3MtY2RrLWxpYi9hd3MtaWFtXCI7XG5pbXBvcnQgKiBhcyBjZGsgZnJvbSBcImF3cy1jZGstbGliXCI7XG5pbXBvcnQgKiBhcyBzc20gZnJvbSBcImF3cy1jZGstbGliL2F3cy1zZWNyZXRzbWFuYWdlclwiO1xuXG4vKipcbiAqIFJlcHJlc2VudHMgYW4gaW5zdGFuY2Ugb2YgdGhlIFJlc3RhdGUgc2VydmljZS4gVGhpcyBjb3VsZCByZXByZXNlbnQgYSBzZWxmLWhvc3RlZCBicm9rZXIsIG9yIFJlc3RhdGUncyBtYW5hZ2VkXG4gKiBzZXJ2aWNlLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIFJlc3RhdGVJbnN0YW5jZSB7XG4gIHJlYWRvbmx5IGludm9rZXJSb2xlOiBpYW0uSVJvbGU7XG4gIHJlYWRvbmx5IG1ldGFFbmRwb2ludDogc3RyaW5nO1xuICByZWFkb25seSBhdXRoVG9rZW4/OiBzc20uSVNlY3JldDtcbiAgcmVhZG9ubHkgcmVnaXN0cmF0aW9uUHJvdmlkZXJUb2tlbjogY2RrLkNmbk91dHB1dDtcbn0iXX0=
|
|
@@ -3,17 +3,7 @@ import * as logs from "aws-cdk-lib/aws-logs";
|
|
|
3
3
|
import * as ec2 from "aws-cdk-lib/aws-ec2";
|
|
4
4
|
import * as iam from "aws-cdk-lib/aws-iam";
|
|
5
5
|
import * as cdk from "aws-cdk-lib";
|
|
6
|
-
import
|
|
7
|
-
import * as acm from "aws-cdk-lib/aws-certificatemanager";
|
|
8
|
-
/**
|
|
9
|
-
* Represents an instance of the Restate service. This could represent a self-hosted broker, or Restate's managed
|
|
10
|
-
* service.
|
|
11
|
-
*/
|
|
12
|
-
export interface RestateInstance {
|
|
13
|
-
readonly invokerRole: iam.Role;
|
|
14
|
-
readonly metaEndpoint: string;
|
|
15
|
-
readonly registrationProviderToken: cdk.CfnOutput;
|
|
16
|
-
}
|
|
6
|
+
import { RestateInstance } from "./restate-instance";
|
|
17
7
|
export declare enum TracingMode {
|
|
18
8
|
DISABLED = "DISABLED",
|
|
19
9
|
AWS_XRAY = "AWS_XRAY"
|
|
@@ -21,7 +11,7 @@ export declare enum TracingMode {
|
|
|
21
11
|
export interface RestateInstanceProps {
|
|
22
12
|
/** Log group for Restate service logs. */
|
|
23
13
|
logGroup: logs.LogGroup;
|
|
24
|
-
/** Tracing mode for Restate services.
|
|
14
|
+
/** Tracing mode for Restate services. Defaults to {@link TracingMode.DISABLED}. */
|
|
25
15
|
tracing?: TracingMode;
|
|
26
16
|
/** Prefix for resources created by this construct that require unique names. */
|
|
27
17
|
prefix?: string;
|
|
@@ -29,26 +19,19 @@ export interface RestateInstanceProps {
|
|
|
29
19
|
restateTag?: string;
|
|
30
20
|
/** Amazon Distro for Open Telemetry Docker image tag. Defaults to `latest`. */
|
|
31
21
|
adotTag?: string;
|
|
32
|
-
/** Optional certificate for ingress endpoint. If unspecified, a plain HTTP listener will be created. */
|
|
33
|
-
certificate?: acm.ICertificate;
|
|
34
22
|
}
|
|
35
23
|
/**
|
|
36
24
|
* Creates a Restate service deployment backed by a single EC2 instance,
|
|
37
|
-
* suitable for development and testing purposes.
|
|
25
|
+
* suitable for development and testing purposes. The instance will be created
|
|
26
|
+
* in a dedicated VPC (unless one is provided). EC2 instance will be allocated
|
|
27
|
+
* a public IP address.
|
|
38
28
|
*/
|
|
39
29
|
export declare class SingleNodeRestateInstance extends Construct implements RestateInstance {
|
|
40
30
|
readonly instance: ec2.Instance;
|
|
41
|
-
readonly invokerRole: iam.
|
|
31
|
+
readonly invokerRole: iam.IRole;
|
|
42
32
|
readonly vpc: ec2.Vpc;
|
|
43
|
-
readonly
|
|
44
|
-
readonly privateIngressEndpoint: string;
|
|
33
|
+
readonly ingressEndpoint: string;
|
|
45
34
|
readonly metaEndpoint: string;
|
|
46
|
-
readonly registrationProvider: cr.Provider;
|
|
47
35
|
readonly registrationProviderToken: cdk.CfnOutput;
|
|
48
36
|
constructor(scope: Construct, id: string, props: RestateInstanceProps);
|
|
49
|
-
/**
|
|
50
|
-
* Creates a custom resource provider to facilitate service discovery. Note that the custom resource event handler
|
|
51
|
-
* must be able to reach the Restate instance's meta endpoint - which is why it is deployed within the same VPC.
|
|
52
|
-
*/
|
|
53
|
-
private createRegistrationProvider;
|
|
54
37
|
}
|
|
@@ -32,22 +32,15 @@ var __importStar = (this && this.__importStar) || function (mod) {
|
|
|
32
32
|
__setModuleDefault(result, mod);
|
|
33
33
|
return result;
|
|
34
34
|
};
|
|
35
|
-
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
36
|
-
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
37
|
-
};
|
|
38
35
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
39
36
|
exports.SingleNodeRestateInstance = exports.TracingMode = void 0;
|
|
40
37
|
const constructs_1 = require("constructs");
|
|
41
38
|
const ec2 = __importStar(require("aws-cdk-lib/aws-ec2"));
|
|
42
39
|
const iam = __importStar(require("aws-cdk-lib/aws-iam"));
|
|
43
|
-
const elb_v2 = __importStar(require("aws-cdk-lib/aws-elasticloadbalancingv2"));
|
|
44
|
-
const aws_elasticloadbalancingv2_1 = require("aws-cdk-lib/aws-elasticloadbalancingv2");
|
|
45
|
-
const aws_elasticloadbalancingv2_targets_1 = require("aws-cdk-lib/aws-elasticloadbalancingv2-targets");
|
|
46
|
-
const lambda_node = __importStar(require("aws-cdk-lib/aws-lambda-nodejs"));
|
|
47
|
-
const node_path_1 = __importDefault(require("node:path"));
|
|
48
|
-
const lambda = __importStar(require("aws-cdk-lib/aws-lambda"));
|
|
49
40
|
const cdk = __importStar(require("aws-cdk-lib"));
|
|
50
|
-
const
|
|
41
|
+
const registration_provider_1 = require("./registration-provider");
|
|
42
|
+
const PUBLIC_INGRESS_PORT = 443;
|
|
43
|
+
const PUBLIC_META_PORT = 9073;
|
|
51
44
|
const RESTATE_INGRESS_PORT = 8080;
|
|
52
45
|
const RESTATE_META_PORT = 9070;
|
|
53
46
|
const RESTATE_DOCKER_DEFAULT_TAG = "latest";
|
|
@@ -59,25 +52,27 @@ var TracingMode;
|
|
|
59
52
|
})(TracingMode || (exports.TracingMode = TracingMode = {}));
|
|
60
53
|
/**
|
|
61
54
|
* Creates a Restate service deployment backed by a single EC2 instance,
|
|
62
|
-
* suitable for development and testing purposes.
|
|
55
|
+
* suitable for development and testing purposes. The instance will be created
|
|
56
|
+
* in a dedicated VPC (unless one is provided). EC2 instance will be allocated
|
|
57
|
+
* a public IP address.
|
|
63
58
|
*/
|
|
64
59
|
class SingleNodeRestateInstance extends constructs_1.Construct {
|
|
65
60
|
constructor(scope, id, props) {
|
|
66
61
|
super(scope, id);
|
|
67
|
-
this.vpc = new ec2.Vpc(this, "
|
|
62
|
+
this.vpc = new ec2.Vpc(this, "Vpc", {
|
|
68
63
|
maxAzs: 3,
|
|
64
|
+
createInternetGateway: true,
|
|
65
|
+
natGateways: 0,
|
|
69
66
|
});
|
|
70
67
|
this.invokerRole = new iam.Role(this, "InstanceRole", {
|
|
71
68
|
assumedBy: new iam.ServicePrincipal("ec2.amazonaws.com"),
|
|
72
|
-
managedPolicies: [
|
|
73
|
-
iam.ManagedPolicy.fromAwsManagedPolicyName("AmazonSSMManagedInstanceCore"),
|
|
74
|
-
],
|
|
69
|
+
managedPolicies: [iam.ManagedPolicy.fromAwsManagedPolicyName("AmazonSSMManagedInstanceCore")],
|
|
75
70
|
});
|
|
76
71
|
props.logGroup.grantWrite(this.invokerRole);
|
|
77
72
|
const restateTag = props.restateTag ?? RESTATE_DOCKER_DEFAULT_TAG;
|
|
78
73
|
const adotTag = props.adotTag ?? ADOT_DOCKER_DEFAULT_TAG;
|
|
79
74
|
const restateInitCommands = ec2.UserData.forLinux();
|
|
80
|
-
restateInitCommands.addCommands("yum update -y", "yum install -y docker", "systemctl enable docker.service", "systemctl start docker.service", [
|
|
75
|
+
restateInitCommands.addCommands("yum update -y", "yum install -y docker nginx", "systemctl enable docker.service", "systemctl start docker.service", [
|
|
81
76
|
"docker run --name adot --restart unless-stopped --detach",
|
|
82
77
|
" -p 4317:4317 -p 55680:55680 -p 8889:8888",
|
|
83
78
|
` public.ecr.aws/aws-observability/aws-otel-collector:${adotTag}`,
|
|
@@ -88,9 +83,14 @@ class SingleNodeRestateInstance extends constructs_1.Construct {
|
|
|
88
83
|
" -e RESTATE_OBSERVABILITY__TRACING__ENDPOINT=http://localhost:4317",
|
|
89
84
|
` --log-driver=awslogs --log-opt awslogs-group=${props.logGroup.logGroupName}`,
|
|
90
85
|
` docker.io/restatedev/restate:${restateTag}`,
|
|
91
|
-
].join("")
|
|
86
|
+
].join(""), "mkdir -p /etc/pki/private", [
|
|
87
|
+
"openssl req -new -x509 -nodes -sha256 -days 365 -extensions v3_ca",
|
|
88
|
+
" -subj '/C=DE/ST=Berlin/L=Berlin/O=restate.dev/OU=demo/CN=restate.example.com'",
|
|
89
|
+
" -newkey rsa:2048 -keyout /etc/pki/private/restate-selfsigned.key -out /etc/pki/private/restate-selfsigned.crt",
|
|
90
|
+
].join(""), ["cat << EOF > /etc/nginx/conf.d/restate-ingress.conf", NGINX_REVERSE_PROXY_CONFIG, "EOF"].join("\n"), "systemctl enable nginx", "systemctl start nginx");
|
|
92
91
|
const restateInstance = new ec2.Instance(this, "Host", {
|
|
93
92
|
vpc: this.vpc,
|
|
93
|
+
vpcSubnets: { subnetType: ec2.SubnetType.PUBLIC },
|
|
94
94
|
instanceType: new ec2.InstanceType("t4g.micro"),
|
|
95
95
|
machineImage: ec2.MachineImage.latestAmazonLinux2023({
|
|
96
96
|
cpuType: ec2.AmazonLinuxCpuType.ARM_64,
|
|
@@ -107,81 +107,57 @@ class SingleNodeRestateInstance extends constructs_1.Construct {
|
|
|
107
107
|
const restateInstanceSecurityGroup = new ec2.SecurityGroup(this, "RestateSecurityGroup", {
|
|
108
108
|
vpc: this.vpc,
|
|
109
109
|
securityGroupName: "RestateSecurityGroup",
|
|
110
|
-
description: "
|
|
111
|
-
});
|
|
112
|
-
const ingressLoadBalancer = new elb_v2.ApplicationLoadBalancer(this, "RestateAlb", {
|
|
113
|
-
vpc: this.vpc,
|
|
114
|
-
internetFacing: true,
|
|
115
|
-
});
|
|
116
|
-
const targetGroup = new elb_v2.ApplicationTargetGroup(this, "TargetGroup", {
|
|
117
|
-
vpc: this.vpc,
|
|
118
|
-
protocol: elb_v2.ApplicationProtocol.HTTP,
|
|
119
|
-
port: RESTATE_INGRESS_PORT,
|
|
120
|
-
targets: [new aws_elasticloadbalancingv2_targets_1.InstanceTarget(restateInstance)],
|
|
121
|
-
healthCheck: {
|
|
122
|
-
protocol: elb_v2.Protocol.HTTP,
|
|
123
|
-
path: "/grpc.health.v1.Health/Check",
|
|
124
|
-
interval: cdk.Duration.seconds(60),
|
|
125
|
-
},
|
|
126
|
-
});
|
|
127
|
-
ingressLoadBalancer.addListener("Listener", {
|
|
128
|
-
port: props.certificate ? 443 : 80,
|
|
129
|
-
protocol: props.certificate ? aws_elasticloadbalancingv2_1.ApplicationProtocol.HTTPS : aws_elasticloadbalancingv2_1.ApplicationProtocol.HTTP,
|
|
130
|
-
defaultTargetGroups: [targetGroup],
|
|
131
|
-
certificates: props.certificate ? [elb_v2.ListenerCertificate.fromCertificateManager(props.certificate)] : [],
|
|
132
|
-
});
|
|
133
|
-
const albSecurityGroup = new ec2.SecurityGroup(this, "AlbSecurityGroup", {
|
|
134
|
-
vpc: this.vpc,
|
|
135
|
-
description: "ALB security group",
|
|
136
|
-
allowAllOutbound: false,
|
|
137
|
-
});
|
|
138
|
-
albSecurityGroup.addEgressRule(ec2.Peer.anyIpv4(), ec2.Port.tcp(RESTATE_INGRESS_PORT), "Allow outbound HTTP traffic to Restate ingress");
|
|
139
|
-
ingressLoadBalancer.addSecurityGroup(albSecurityGroup);
|
|
140
|
-
restateInstanceSecurityGroup.addIngressRule(albSecurityGroup, ec2.Port.tcp(RESTATE_INGRESS_PORT), "Allow traffic from ALB to Restate ingress");
|
|
141
|
-
// These rules allow the service registration component to trigger service discovery as needed; the requests
|
|
142
|
-
// originate from a VPC-bound Lambda function that backs the custom resource.
|
|
143
|
-
this.vpc.privateSubnets.forEach((subnet) => {
|
|
144
|
-
restateInstanceSecurityGroup.addIngressRule(ec2.Peer.ipv4(subnet.ipv4CidrBlock), ec2.Port.tcp(RESTATE_META_PORT), "Allow traffic from the VPC to Restate meta");
|
|
145
|
-
});
|
|
146
|
-
this.vpc.privateSubnets.forEach((subnet) => {
|
|
147
|
-
restateInstanceSecurityGroup.addIngressRule(ec2.Peer.ipv4(subnet.ipv4CidrBlock), ec2.Port.tcp(RESTATE_INGRESS_PORT), "Allow traffic from the VPC to Restate ingress");
|
|
110
|
+
description: "Restate service ACLs",
|
|
148
111
|
});
|
|
149
112
|
restateInstance.addSecurityGroup(restateInstanceSecurityGroup);
|
|
150
|
-
|
|
151
|
-
|
|
113
|
+
restateInstanceSecurityGroup.addIngressRule(ec2.Peer.anyIpv4(), ec2.Port.tcp(443), "Allow traffic from anywhere to Restate ingress");
|
|
114
|
+
restateInstanceSecurityGroup.addIngressRule(ec2.Peer.anyIpv4(), ec2.Port.tcp(9073), "Allow traffic from anywhere to Restate meta");
|
|
115
|
+
const registrationProvider = new registration_provider_1.RegistrationProvider(this, "RegistrationProvider", {});
|
|
152
116
|
this.registrationProviderToken = new cdk.CfnOutput(this, "RegistrationProviderToken", {
|
|
153
117
|
description: "Custom resource provider service token, needed by the Restate service registry component to trigger discovery",
|
|
154
118
|
exportName: [props.prefix, "RegistrationProviderToken"].join("-"),
|
|
155
119
|
value: registrationProvider.serviceToken,
|
|
156
120
|
});
|
|
157
|
-
this.
|
|
158
|
-
this.
|
|
159
|
-
this.metaEndpoint = `http://${this.instance.instancePrivateDnsName}:${RESTATE_META_PORT}`;
|
|
160
|
-
}
|
|
161
|
-
/**
|
|
162
|
-
* Creates a custom resource provider to facilitate service discovery. Note that the custom resource event handler
|
|
163
|
-
* must be able to reach the Restate instance's meta endpoint - which is why it is deployed within the same VPC.
|
|
164
|
-
*/
|
|
165
|
-
createRegistrationProvider() {
|
|
166
|
-
const registrationHandler = new lambda_node.NodejsFunction(this, "RegistrationHandler", {
|
|
167
|
-
description: "Restate custom registration handler",
|
|
168
|
-
entry: node_path_1.default.join(__dirname, "register-service-handler/index.js"),
|
|
169
|
-
architecture: lambda.Architecture.ARM_64,
|
|
170
|
-
runtime: lambda.Runtime.NODEJS_LATEST,
|
|
171
|
-
memorySize: 128,
|
|
172
|
-
timeout: cdk.Duration.seconds(60),
|
|
173
|
-
environment: {
|
|
174
|
-
NODE_OPTIONS: "--enable-source-maps",
|
|
175
|
-
},
|
|
176
|
-
vpc: this.vpc,
|
|
177
|
-
vpcSubnets: {
|
|
178
|
-
subnets: this.vpc.privateSubnets,
|
|
179
|
-
},
|
|
180
|
-
});
|
|
181
|
-
return new cr.Provider(this, "RegistrationProvider", {
|
|
182
|
-
onEventHandler: registrationHandler,
|
|
183
|
-
});
|
|
121
|
+
this.ingressEndpoint = `https://${restateInstance.instancePublicDnsName}${PUBLIC_INGRESS_PORT == 443 ? "" : `:${PUBLIC_INGRESS_PORT}`}`;
|
|
122
|
+
this.metaEndpoint = `https://${restateInstance.instancePublicDnsName}:${PUBLIC_META_PORT}`;
|
|
184
123
|
}
|
|
185
124
|
}
|
|
186
125
|
exports.SingleNodeRestateInstance = SingleNodeRestateInstance;
|
|
187
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"single-node-restate-instance.js","sourceRoot":"","sources":["../lib/restate-constructs/single-node-restate-instance.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 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 elb_v2 from \"aws-cdk-lib/aws-elasticloadbalancingv2\";\nimport { ApplicationProtocol } from \"aws-cdk-lib/aws-elasticloadbalancingv2\";\nimport { InstanceTarget } from \"aws-cdk-lib/aws-elasticloadbalancingv2-targets\";\nimport * as lambda_node from \"aws-cdk-lib/aws-lambda-nodejs\";\nimport path from \"node:path\";\nimport * as lambda from \"aws-cdk-lib/aws-lambda\";\nimport * as cdk from \"aws-cdk-lib\";\nimport * as cr from \"aws-cdk-lib/custom-resources\";\nimport * as acm from \"aws-cdk-lib/aws-certificatemanager\";\n\nconst RESTATE_INGRESS_PORT = 8080;\nconst RESTATE_META_PORT = 9070;\nconst RESTATE_DOCKER_DEFAULT_TAG = \"latest\";\nconst ADOT_DOCKER_DEFAULT_TAG = \"latest\";\n\n/**\n * Represents an instance of the Restate service. This could represent a self-hosted broker, or Restate's managed\n * service.\n */\nexport interface RestateInstance {\n  readonly invokerRole: iam.Role;\n  readonly metaEndpoint: string;\n  readonly registrationProviderToken: cdk.CfnOutput;\n}\n\nexport enum TracingMode {\n  DISABLED = \"DISABLED\",\n  AWS_XRAY = \"AWS_XRAY\",\n}\n\nexport interface RestateInstanceProps {\n  /** Log group for Restate service logs. */\n  logGroup: logs.LogGroup;\n\n  /** Tracing mode for Restate services. Disabled by default. */\n  tracing?: TracingMode;\n\n  /** Prefix for resources created by this construct that require unique names. */\n  prefix?: 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  /** Optional certificate for ingress endpoint. If unspecified, a plain HTTP listener will be created. */\n  certificate?: acm.ICertificate;\n}\n\n/**\n * Creates a Restate service deployment backed by a single EC2 instance,\n * suitable for development and testing purposes.\n */\nexport class SingleNodeRestateInstance extends Construct implements RestateInstance {\n  readonly instance: ec2.Instance;\n  readonly invokerRole: iam.Role;\n  readonly vpc: ec2.Vpc;\n\n  readonly publicIngressEndpoint: string;\n  readonly privateIngressEndpoint: string;\n  readonly metaEndpoint: string;\n  readonly registrationProvider: cr.Provider;\n  readonly registrationProviderToken: cdk.CfnOutput;\n\n  constructor(scope: Construct, id: string, props: RestateInstanceProps) {\n    super(scope, id);\n\n    this.vpc = new ec2.Vpc(this, \"RestateVpc\", {\n      maxAzs: 3,\n    });\n\n    this.invokerRole = new iam.Role(this, \"InstanceRole\", {\n      assumedBy: new iam.ServicePrincipal(\"ec2.amazonaws.com\"),\n      managedPolicies: [\n        iam.ManagedPolicy.fromAwsManagedPolicyName(\"AmazonSSMManagedInstanceCore\"),\n      ],\n    });\n    props.logGroup.grantWrite(this.invokerRole);\n\n    const restateTag = props.restateTag ?? RESTATE_DOCKER_DEFAULT_TAG;\n    const adotTag = props.adotTag ?? ADOT_DOCKER_DEFAULT_TAG;\n    const restateInitCommands = ec2.UserData.forLinux();\n    restateInitCommands.addCommands(\n      \"yum update -y\",\n      \"yum install -y docker\",\n      \"systemctl enable docker.service\",\n      \"systemctl start docker.service\",\n      [\n        \"docker run --name adot --restart unless-stopped --detach\",\n        \" -p 4317:4317 -p 55680:55680 -p 8889:8888\",\n        ` public.ecr.aws/aws-observability/aws-otel-collector:${adotTag}`,\n      ].join(\"\"),\n      [\n        \"docker run --name restate --restart unless-stopped --detach\",\n        \" --volume /var/restate:/target --network=host\",\n        \" -e RESTATE_OBSERVABILITY__LOG__FORMAT=Json -e RUST_LOG=info,restate_worker::partition=warn\",\n        \" -e RESTATE_OBSERVABILITY__TRACING__ENDPOINT=http://localhost:4317\",\n        ` --log-driver=awslogs --log-opt awslogs-group=${props.logGroup.logGroupName}`,\n        ` docker.io/restatedev/restate:${restateTag}`,\n      ].join(\"\"),\n    );\n\n    const restateInstance = new ec2.Instance(this, \"Host\", {\n      vpc: this.vpc,\n      instanceType: new ec2.InstanceType(\"t4g.micro\"),\n      machineImage: ec2.MachineImage.latestAmazonLinux2023({\n        cpuType: ec2.AmazonLinuxCpuType.ARM_64,\n      }),\n      role: this.invokerRole,\n      userData: restateInitCommands,\n    });\n    this.instance = restateInstance;\n\n    // We start the ADOT collector regardless, and only control whether they will be published to X-Ray via instance\n    // role permissions. This way historic traces will be buffered on the host, even if tracing is disabled initially.\n    if (props.tracing === TracingMode.AWS_XRAY) {\n      restateInstance.role.addManagedPolicy(iam.ManagedPolicy.fromAwsManagedPolicyName(\"AWSXrayWriteOnlyAccess\"));\n    }\n\n    const restateInstanceSecurityGroup = new ec2.SecurityGroup(this, \"RestateSecurityGroup\", {\n      vpc: this.vpc,\n      securityGroupName: \"RestateSecurityGroup\",\n      description: \"Allow inbound traffic to Restate\",\n    });\n\n    const ingressLoadBalancer = new elb_v2.ApplicationLoadBalancer(this, \"RestateAlb\", {\n      vpc: this.vpc,\n      internetFacing: true,\n    });\n    const targetGroup = new elb_v2.ApplicationTargetGroup(this, \"TargetGroup\", {\n      vpc: this.vpc,\n      protocol: elb_v2.ApplicationProtocol.HTTP,\n      port: RESTATE_INGRESS_PORT,\n      targets: [new InstanceTarget(restateInstance)],\n      healthCheck: {\n        protocol: elb_v2.Protocol.HTTP,\n        path: \"/grpc.health.v1.Health/Check\",\n        interval: cdk.Duration.seconds(60),\n      },\n    });\n    ingressLoadBalancer.addListener(\"Listener\", {\n      port: props.certificate ? 443 : 80,\n      protocol: props.certificate ? ApplicationProtocol.HTTPS : ApplicationProtocol.HTTP,\n      defaultTargetGroups: [targetGroup],\n      certificates: props.certificate ? [elb_v2.ListenerCertificate.fromCertificateManager(props.certificate)] : [],\n    });\n\n    const albSecurityGroup = new ec2.SecurityGroup(this, \"AlbSecurityGroup\", {\n      vpc: this.vpc,\n      description: \"ALB security group\",\n      allowAllOutbound: false,\n    });\n    albSecurityGroup.addEgressRule(ec2.Peer.anyIpv4(), ec2.Port.tcp(RESTATE_INGRESS_PORT), \"Allow outbound HTTP traffic to Restate ingress\");\n    ingressLoadBalancer.addSecurityGroup(albSecurityGroup);\n\n    restateInstanceSecurityGroup.addIngressRule(albSecurityGroup, ec2.Port.tcp(RESTATE_INGRESS_PORT), \"Allow traffic from ALB to Restate ingress\");\n\n    // These rules allow the service registration component to trigger service discovery as needed; the requests\n    // originate from a VPC-bound Lambda function that backs the custom resource.\n    this.vpc.privateSubnets.forEach((subnet) => {\n      restateInstanceSecurityGroup.addIngressRule(ec2.Peer.ipv4(subnet.ipv4CidrBlock), ec2.Port.tcp(RESTATE_META_PORT), \"Allow traffic from the VPC to Restate meta\");\n    });\n    this.vpc.privateSubnets.forEach((subnet) => {\n      restateInstanceSecurityGroup.addIngressRule(ec2.Peer.ipv4(subnet.ipv4CidrBlock), ec2.Port.tcp(RESTATE_INGRESS_PORT), \"Allow traffic from the VPC to Restate ingress\");\n    });\n    restateInstance.addSecurityGroup(restateInstanceSecurityGroup);\n\n    const registrationProvider = this.createRegistrationProvider();\n    this.registrationProvider = registrationProvider;\n    this.registrationProviderToken = new cdk.CfnOutput(this, \"RegistrationProviderToken\", {\n      description: \"Custom resource provider service token, needed by the Restate service registry component to trigger discovery\",\n      exportName: [props.prefix, \"RegistrationProviderToken\"].join(\"-\"),\n      value: registrationProvider.serviceToken,\n    });\n\n    this.publicIngressEndpoint = `${props.certificate ? \"https\" : \"http\"}://${ingressLoadBalancer.loadBalancerDnsName}`;\n    this.privateIngressEndpoint = `http://${this.instance.instancePrivateDnsName}:${RESTATE_INGRESS_PORT}`;\n    this.metaEndpoint = `http://${this.instance.instancePrivateDnsName}:${RESTATE_META_PORT}`;\n  }\n\n  /**\n   * Creates a custom resource provider to facilitate service discovery. Note that the custom resource event handler\n   * must be able to reach the Restate instance's meta endpoint - which is why it is deployed within the same VPC.\n   */\n  private createRegistrationProvider() {\n    const registrationHandler = new lambda_node.NodejsFunction(this, \"RegistrationHandler\", {\n      description: \"Restate custom registration handler\",\n      entry: path.join(__dirname, \"register-service-handler/index.js\"),\n      architecture: lambda.Architecture.ARM_64,\n      runtime: lambda.Runtime.NODEJS_LATEST,\n      memorySize: 128,\n      timeout: cdk.Duration.seconds(60),\n      environment: {\n        NODE_OPTIONS: \"--enable-source-maps\",\n      },\n      vpc: this.vpc,\n      vpcSubnets: {\n        subnets: this.vpc.privateSubnets,\n      },\n    });\n\n    return new cr.Provider(this, \"RegistrationProvider\", {\n      onEventHandler: registrationHandler,\n    });\n  }\n}"]}
|
|
126
|
+
const NGINX_REVERSE_PROXY_CONFIG = [
|
|
127
|
+
"server {",
|
|
128
|
+
" listen 443 ssl http2;",
|
|
129
|
+
" listen [::]:443 ssl http2;",
|
|
130
|
+
" server_name _;",
|
|
131
|
+
" root /usr/share/nginx/html;",
|
|
132
|
+
"",
|
|
133
|
+
' ssl_certificate "/etc/pki/private/restate-selfsigned.crt";',
|
|
134
|
+
' ssl_certificate_key "/etc/pki/private/restate-selfsigned.key";',
|
|
135
|
+
" ssl_session_cache shared:SSL:1m;",
|
|
136
|
+
" ssl_session_timeout 10m;",
|
|
137
|
+
" ssl_ciphers PROFILE=SYSTEM;",
|
|
138
|
+
" ssl_prefer_server_ciphers on;",
|
|
139
|
+
"",
|
|
140
|
+
" location / {",
|
|
141
|
+
` proxy_pass http://localhost:${RESTATE_INGRESS_PORT};`,
|
|
142
|
+
" }",
|
|
143
|
+
"}",
|
|
144
|
+
"",
|
|
145
|
+
"server {",
|
|
146
|
+
" listen 9073 ssl http2;",
|
|
147
|
+
" listen [::]:9073 ssl http2;",
|
|
148
|
+
" server_name _;",
|
|
149
|
+
" root /usr/share/nginx/html;",
|
|
150
|
+
"",
|
|
151
|
+
' ssl_certificate "/etc/pki/private/restate-selfsigned.crt";',
|
|
152
|
+
' ssl_certificate_key "/etc/pki/private/restate-selfsigned.key";',
|
|
153
|
+
" ssl_session_cache shared:SSL:1m;",
|
|
154
|
+
" ssl_session_timeout 10m;",
|
|
155
|
+
" ssl_ciphers PROFILE=SYSTEM;",
|
|
156
|
+
" ssl_prefer_server_ciphers on;",
|
|
157
|
+
"",
|
|
158
|
+
" location / {",
|
|
159
|
+
` proxy_pass http://localhost:${RESTATE_META_PORT};`,
|
|
160
|
+
" }",
|
|
161
|
+
"}",
|
|
162
|
+
].join("\n");
|
|
163
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"single-node-restate-instance.js","sourceRoot":"","sources":["../lib/restate-constructs/single-node-restate-instance.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 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 cdk from \"aws-cdk-lib\";\nimport { RestateInstance } from \"./restate-instance\";\nimport { RegistrationProvider } from \"./registration-provider\";\n\nconst PUBLIC_INGRESS_PORT = 443;\nconst PUBLIC_META_PORT = 9073;\nconst RESTATE_INGRESS_PORT = 8080;\nconst RESTATE_META_PORT = 9070;\nconst RESTATE_DOCKER_DEFAULT_TAG = \"latest\";\nconst ADOT_DOCKER_DEFAULT_TAG = \"latest\";\n\nexport enum TracingMode {\n  DISABLED = \"DISABLED\",\n  AWS_XRAY = \"AWS_XRAY\",\n}\n\nexport interface RestateInstanceProps {\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  /** 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/**\n * Creates a Restate service deployment backed by a single EC2 instance,\n * suitable for development and testing purposes. The instance will be created\n * in a dedicated VPC (unless one is provided). EC2 instance will be allocated\n * a public IP address.\n */\nexport class SingleNodeRestateInstance extends Construct implements RestateInstance {\n  readonly instance: ec2.Instance;\n  readonly invokerRole: iam.IRole;\n  readonly vpc: ec2.Vpc;\n\n  readonly ingressEndpoint: string;\n  readonly metaEndpoint: string;\n  readonly registrationProviderToken: cdk.CfnOutput;\n\n  constructor(scope: Construct, id: string, props: RestateInstanceProps) {\n    super(scope, id);\n\n    this.vpc = new ec2.Vpc(this, \"Vpc\", {\n      maxAzs: 3,\n      createInternetGateway: true,\n      natGateways: 0,\n    });\n\n    this.invokerRole = new iam.Role(this, \"InstanceRole\", {\n      assumedBy: new iam.ServicePrincipal(\"ec2.amazonaws.com\"),\n      managedPolicies: [iam.ManagedPolicy.fromAwsManagedPolicyName(\"AmazonSSMManagedInstanceCore\")],\n    });\n    props.logGroup.grantWrite(this.invokerRole);\n\n    const restateTag = props.restateTag ?? RESTATE_DOCKER_DEFAULT_TAG;\n    const adotTag = props.adotTag ?? ADOT_DOCKER_DEFAULT_TAG;\n    const restateInitCommands = ec2.UserData.forLinux();\n    restateInitCommands.addCommands(\n      \"yum update -y\",\n      \"yum install -y docker nginx\",\n\n      \"systemctl enable docker.service\",\n      \"systemctl start docker.service\",\n      [\n        \"docker run --name adot --restart unless-stopped --detach\",\n        \" -p 4317:4317 -p 55680:55680 -p 8889:8888\",\n        ` public.ecr.aws/aws-observability/aws-otel-collector:${adotTag}`,\n      ].join(\"\"),\n      [\n        \"docker run --name restate --restart unless-stopped --detach\",\n        \" --volume /var/restate:/target --network=host\",\n        \" -e RESTATE_OBSERVABILITY__LOG__FORMAT=Json -e RUST_LOG=info,restate_worker::partition=warn\",\n        \" -e RESTATE_OBSERVABILITY__TRACING__ENDPOINT=http://localhost:4317\",\n        ` --log-driver=awslogs --log-opt awslogs-group=${props.logGroup.logGroupName}`,\n        ` docker.io/restatedev/restate:${restateTag}`,\n      ].join(\"\"),\n\n      \"mkdir -p /etc/pki/private\",\n      [\n        \"openssl req -new -x509 -nodes -sha256 -days 365 -extensions v3_ca\",\n        \" -subj '/C=DE/ST=Berlin/L=Berlin/O=restate.dev/OU=demo/CN=restate.example.com'\",\n        \" -newkey rsa:2048 -keyout /etc/pki/private/restate-selfsigned.key -out /etc/pki/private/restate-selfsigned.crt\",\n      ].join(\"\"),\n      [\"cat << EOF > /etc/nginx/conf.d/restate-ingress.conf\", NGINX_REVERSE_PROXY_CONFIG, \"EOF\"].join(\"\\n\"),\n      \"systemctl enable nginx\",\n      \"systemctl start nginx\",\n    );\n\n    const restateInstance = new ec2.Instance(this, \"Host\", {\n      vpc: this.vpc,\n      vpcSubnets: { subnetType: ec2.SubnetType.PUBLIC },\n      instanceType: new ec2.InstanceType(\"t4g.micro\"),\n      machineImage: ec2.MachineImage.latestAmazonLinux2023({\n        cpuType: ec2.AmazonLinuxCpuType.ARM_64,\n      }),\n      role: this.invokerRole,\n      userData: restateInitCommands,\n    });\n    this.instance = restateInstance;\n\n    // We start the ADOT collector regardless, and only control whether they will be published to X-Ray via instance\n    // role permissions. This way historic traces will be buffered on the host, even if tracing is disabled initially.\n    if (props.tracing === TracingMode.AWS_XRAY) {\n      restateInstance.role.addManagedPolicy(iam.ManagedPolicy.fromAwsManagedPolicyName(\"AWSXrayWriteOnlyAccess\"));\n    }\n\n    const restateInstanceSecurityGroup = new ec2.SecurityGroup(this, \"RestateSecurityGroup\", {\n      vpc: this.vpc,\n      securityGroupName: \"RestateSecurityGroup\",\n      description: \"Restate service ACLs\",\n    });\n    restateInstance.addSecurityGroup(restateInstanceSecurityGroup);\n\n    restateInstanceSecurityGroup.addIngressRule(\n      ec2.Peer.anyIpv4(),\n      ec2.Port.tcp(443),\n      \"Allow traffic from anywhere to Restate ingress\",\n    );\n    restateInstanceSecurityGroup.addIngressRule(\n      ec2.Peer.anyIpv4(),\n      ec2.Port.tcp(9073),\n      \"Allow traffic from anywhere to Restate meta\",\n    );\n\n    const registrationProvider = new RegistrationProvider(this, \"RegistrationProvider\", {});\n    this.registrationProviderToken = new cdk.CfnOutput(this, \"RegistrationProviderToken\", {\n      description:\n        \"Custom resource provider service token, needed by the Restate service registry component to trigger discovery\",\n      exportName: [props.prefix, \"RegistrationProviderToken\"].join(\"-\"),\n      value: registrationProvider.serviceToken,\n    });\n\n    this.ingressEndpoint = `https://${restateInstance.instancePublicDnsName}${\n      PUBLIC_INGRESS_PORT == 443 ? \"\" : `:${PUBLIC_INGRESS_PORT}`\n    }`;\n    this.metaEndpoint = `https://${restateInstance.instancePublicDnsName}:${PUBLIC_META_PORT}`;\n  }\n}\n\nconst NGINX_REVERSE_PROXY_CONFIG = [\n  \"server {\",\n  \"  listen 443 ssl http2;\",\n  \"  listen [::]:443 ssl http2;\",\n  \"  server_name _;\",\n  \"  root /usr/share/nginx/html;\",\n  \"\",\n  '  ssl_certificate \"/etc/pki/private/restate-selfsigned.crt\";',\n  '  ssl_certificate_key \"/etc/pki/private/restate-selfsigned.key\";',\n  \"  ssl_session_cache shared:SSL:1m;\",\n  \"  ssl_session_timeout 10m;\",\n  \"  ssl_ciphers PROFILE=SYSTEM;\",\n  \"  ssl_prefer_server_ciphers on;\",\n  \"\",\n  \"  location / {\",\n  `    proxy_pass http://localhost:${RESTATE_INGRESS_PORT};`,\n  \"  }\",\n  \"}\",\n  \"\",\n  \"server {\",\n  \"  listen 9073 ssl http2;\",\n  \"  listen [::]:9073 ssl http2;\",\n  \"  server_name _;\",\n  \"  root /usr/share/nginx/html;\",\n  \"\",\n  '  ssl_certificate \"/etc/pki/private/restate-selfsigned.crt\";',\n  '  ssl_certificate_key \"/etc/pki/private/restate-selfsigned.key\";',\n  \"  ssl_session_cache shared:SSL:1m;\",\n  \"  ssl_session_timeout 10m;\",\n  \"  ssl_ciphers PROFILE=SYSTEM;\",\n  \"  ssl_prefer_server_ciphers on;\",\n  \"\",\n  \"  location / {\",\n  `    proxy_pass http://localhost:${RESTATE_META_PORT};`,\n  \"  }\",\n  \"}\",\n].join(\"\\n\");\n"]}
|
package/package.json
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@restatedev/restate-cdk",
|
|
3
3
|
"description": "Restate.dev CDK constructs",
|
|
4
|
-
"version": "0.
|
|
4
|
+
"version": "0.4.1",
|
|
5
5
|
"author": "Restate Developers",
|
|
6
6
|
"license": "MIT",
|
|
7
7
|
"email": "code@restate.dev",
|
|
@@ -30,12 +30,14 @@
|
|
|
30
30
|
"aws-cdk": "2.108.0",
|
|
31
31
|
"esbuild": "^0.19.6",
|
|
32
32
|
"jest": "^29.7.0",
|
|
33
|
+
"prettier": "^3.1.0",
|
|
33
34
|
"ts-jest": "^29.1.1",
|
|
34
35
|
"ts-node": "^10.9.1",
|
|
35
36
|
"typescript": "~5.2.2"
|
|
36
37
|
},
|
|
37
|
-
"
|
|
38
|
-
"aws-
|
|
38
|
+
"peerDependencies": {
|
|
39
|
+
"@aws-sdk/client-secrets-manager": "^3.462.0",
|
|
40
|
+
"aws-cdk-lib": "^2.113.0",
|
|
39
41
|
"constructs": "^10.0.0",
|
|
40
42
|
"node-fetch": "^3.3.2",
|
|
41
43
|
"source-map-support": "^0.5.21"
|