@restatedev/restate-cdk 1.0.1 → 1.1.0-rc.2
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/.github/workflows/pr-checks.yaml +21 -0
- package/.github/workflows/publish.yml +2 -2
- package/.github/workflows/test.yml +3 -3
- package/README.md +3 -0
- package/dist/deployments-common.js +11 -1
- package/dist/index.d.ts +1 -1
- package/dist/index.js +2 -2
- package/dist/register-service-handler/index.d.ts +19 -2
- package/dist/register-service-handler/index.js +34 -34
- package/dist/restate-cloud-environment.d.ts +47 -0
- package/dist/restate-cloud-environment.js +100 -0
- package/dist/restate-environment.d.ts +9 -6
- package/dist/restate-environment.js +16 -1
- package/dist/service-deployer.d.ts +76 -43
- package/dist/service-deployer.js +59 -23
- package/dist/single-node-restate-deployment.d.ts +102 -4
- package/dist/single-node-restate-deployment.js +241 -84
- package/jest.config.js +5 -5
- package/package.json +16 -13
- package/test/__snapshots__/restate-constructs.test.ts.snap +1241 -405
- package/test/handlers/handler.js +28 -0
- package/tsconfig.json +4 -14
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
name: PR Checks (npm)
|
|
2
|
+
|
|
3
|
+
on:
|
|
4
|
+
push:
|
|
5
|
+
branches: [main]
|
|
6
|
+
pull_request:
|
|
7
|
+
types: [opened, synchronize, reopened, ready_for_review]
|
|
8
|
+
workflow_dispatch:
|
|
9
|
+
|
|
10
|
+
jobs:
|
|
11
|
+
check:
|
|
12
|
+
runs-on: ubuntu-latest
|
|
13
|
+
steps:
|
|
14
|
+
- uses: actions/checkout@v4
|
|
15
|
+
- uses: actions/setup-node@v4
|
|
16
|
+
with:
|
|
17
|
+
node-version: "20"
|
|
18
|
+
- run: npm ci
|
|
19
|
+
- run: npm run lint
|
|
20
|
+
- run: npm run build
|
|
21
|
+
- run: npm run test
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
name: Publish package to NPM
|
|
2
2
|
on:
|
|
3
3
|
release:
|
|
4
|
-
types: [
|
|
4
|
+
types: [published]
|
|
5
5
|
jobs:
|
|
6
6
|
build:
|
|
7
7
|
runs-on: ubuntu-latest
|
|
@@ -14,7 +14,7 @@ jobs:
|
|
|
14
14
|
- uses: actions/setup-node@v4
|
|
15
15
|
with:
|
|
16
16
|
node-version: "20.x"
|
|
17
|
-
registry-url:
|
|
17
|
+
registry-url: "https://registry.npmjs.org"
|
|
18
18
|
- run: npm ci
|
|
19
19
|
- run: npm run build
|
|
20
20
|
- run: npm run test
|
|
@@ -2,7 +2,7 @@ name: Build and test
|
|
|
2
2
|
|
|
3
3
|
on:
|
|
4
4
|
push:
|
|
5
|
-
branches: [
|
|
5
|
+
branches: [main]
|
|
6
6
|
|
|
7
7
|
jobs:
|
|
8
8
|
build:
|
|
@@ -11,7 +11,7 @@ jobs:
|
|
|
11
11
|
runs-on: ubuntu-latest
|
|
12
12
|
strategy:
|
|
13
13
|
matrix:
|
|
14
|
-
node-version: [
|
|
14
|
+
node-version: [20.x]
|
|
15
15
|
|
|
16
16
|
steps:
|
|
17
17
|
- uses: actions/checkout@v4
|
|
@@ -19,7 +19,7 @@ jobs:
|
|
|
19
19
|
uses: actions/setup-node@v4
|
|
20
20
|
with:
|
|
21
21
|
node-version: ${{ matrix.node-version }}
|
|
22
|
-
registry-url:
|
|
22
|
+
registry-url: "https://registry.npmjs.org"
|
|
23
23
|
- run: npm ci
|
|
24
24
|
- run: npm run build
|
|
25
25
|
- run: npm run test
|
package/README.md
CHANGED
|
@@ -20,6 +20,8 @@ npm i @restatedev/restate-cdk
|
|
|
20
20
|
|
|
21
21
|
## Available constructs
|
|
22
22
|
|
|
23
|
+
- [`RestateCloudEnvironment`](./lib/restate-constructs/restate-cloud-environment.ts) - Supports deploying Restate
|
|
24
|
+
services to an existing [Restate Cloud](https://cloud.restate.dev) environment.
|
|
23
25
|
- [`SingleNodeRestateDeployment`](./lib/restate-constructs/single-node-restate-deployment.ts) - Deploys a self-hosted
|
|
24
26
|
Restate server running on Amazon EC2; this provides a basic single-node deployment targeted at development and testing
|
|
25
27
|
- [`ServiceDeployer`](./lib/restate-constructs/service-deployer.ts) - facilitates registration of Lambda-based service
|
|
@@ -32,5 +34,6 @@ the [Restate CDK documentation](https://docs.restate.dev/deploy/lambda/cdk).
|
|
|
32
34
|
|
|
33
35
|
You can use the following templates to bootstrap your own CDK projects:
|
|
34
36
|
|
|
37
|
+
- [e2e tests in this repository](test/e2e)
|
|
35
38
|
- [typescript-lambda-cdk](https://github.com/restatedev/examples/tree/main/templates/typescript-lambda-cdk)
|
|
36
39
|
- [kotlin-gradle-lambda-cdk](https://github.com/restatedev/examples/tree/main/templates/kotlin-gradle-lambda-cdk)
|
|
@@ -1,4 +1,14 @@
|
|
|
1
1
|
"use strict";
|
|
2
|
+
/*
|
|
3
|
+
* Copyright (c) 2024 - 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
|
+
*/
|
|
2
12
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
13
|
exports.TracingMode = void 0;
|
|
4
14
|
var TracingMode;
|
|
@@ -6,4 +16,4 @@ var TracingMode;
|
|
|
6
16
|
TracingMode["DISABLED"] = "DISABLED";
|
|
7
17
|
TracingMode["AWS_XRAY"] = "AWS_XRAY";
|
|
8
18
|
})(TracingMode || (exports.TracingMode = TracingMode = {}));
|
|
9
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
19
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGVwbG95bWVudHMtY29tbW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vbGliL3Jlc3RhdGUtY29uc3RydWN0cy9kZXBsb3ltZW50cy1jb21tb24udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7Ozs7Ozs7R0FTRzs7O0FBRUgsSUFBWSxXQUdYO0FBSEQsV0FBWSxXQUFXO0lBQ3JCLG9DQUFxQixDQUFBO0lBQ3JCLG9DQUFxQixDQUFBO0FBQ3ZCLENBQUMsRUFIVyxXQUFXLDJCQUFYLFdBQVcsUUFHdEIiLCJzb3VyY2VzQ29udGVudCI6WyIvKlxuICogQ29weXJpZ2h0IChjKSAyMDI0IC0gUmVzdGF0ZSBTb2Z0d2FyZSwgSW5jLiwgUmVzdGF0ZSBHbWJIXG4gKlxuICogVGhpcyBmaWxlIGlzIHBhcnQgb2YgdGhlIFJlc3RhdGUgU0RLIGZvciBOb2RlLmpzL1R5cGVTY3JpcHQsXG4gKiB3aGljaCBpcyByZWxlYXNlZCB1bmRlciB0aGUgTUlUIGxpY2Vuc2UuXG4gKlxuICogWW91IGNhbiBmaW5kIGEgY29weSBvZiB0aGUgbGljZW5zZSBpbiBmaWxlIExJQ0VOU0UgaW4gdGhlIHJvb3RcbiAqIGRpcmVjdG9yeSBvZiB0aGlzIHJlcG9zaXRvcnkgb3IgcGFja2FnZSwgb3IgYXRcbiAqIGh0dHBzOi8vZ2l0aHViLmNvbS9yZXN0YXRlZGV2L3Nkay10eXBlc2NyaXB0L2Jsb2IvbWFpbi9MSUNFTlNFXG4gKi9cblxuZXhwb3J0IGVudW0gVHJhY2luZ01vZGUge1xuICBESVNBQkxFRCA9IFwiRElTQUJMRURcIixcbiAgQVdTX1hSQVkgPSBcIkFXU19YUkFZXCIsXG59XG4iXX0=
|
package/dist/index.d.ts
CHANGED
package/dist/index.js
CHANGED
|
@@ -26,7 +26,7 @@ var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
|
26
26
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
27
27
|
__exportStar(require("./service-deployer"), exports);
|
|
28
28
|
__exportStar(require("./restate-environment"), exports);
|
|
29
|
+
__exportStar(require("./restate-cloud-environment"), exports);
|
|
29
30
|
__exportStar(require("./deployments-common"), exports);
|
|
30
31
|
__exportStar(require("./single-node-restate-deployment"), exports);
|
|
31
|
-
|
|
32
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9saWIvcmVzdGF0ZS1jb25zdHJ1Y3RzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQTs7Ozs7Ozs7O0dBU0c7Ozs7Ozs7Ozs7Ozs7Ozs7QUFFSCxxREFBbUM7QUFDbkMsd0RBQXNDO0FBQ3RDLHVEQUFxQztBQUNyQyxtRUFBaUQ7QUFDakQsK0RBQTZDIiwic291cmNlc0NvbnRlbnQiOlsiLypcbiAqIENvcHlyaWdodCAoYykgMjAyMyAtIFJlc3RhdGUgU29mdHdhcmUsIEluYy4sIFJlc3RhdGUgR21iSFxuICpcbiAqIFRoaXMgZmlsZSBpcyBwYXJ0IG9mIHRoZSBSZXN0YXRlIFNESyBmb3IgTm9kZS5qcy9UeXBlU2NyaXB0LFxuICogd2hpY2ggaXMgcmVsZWFzZWQgdW5kZXIgdGhlIE1JVCBsaWNlbnNlLlxuICpcbiAqIFlvdSBjYW4gZmluZCBhIGNvcHkgb2YgdGhlIGxpY2Vuc2UgaW4gZmlsZSBMSUNFTlNFIGluIHRoZSByb290XG4gKiBkaXJlY3Rvcnkgb2YgdGhpcyByZXBvc2l0b3J5IG9yIHBhY2thZ2UsIG9yIGF0XG4gKiBodHRwczovL2dpdGh1Yi5jb20vcmVzdGF0ZWRldi9zZGstdHlwZXNjcmlwdC9ibG9iL21haW4vTElDRU5TRVxuICovXG5cbmV4cG9ydCAqIGZyb20gXCIuL3NlcnZpY2UtZGVwbG95ZXJcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3Jlc3RhdGUtZW52aXJvbm1lbnRcIjtcbmV4cG9ydCAqIGZyb20gXCIuL2RlcGxveW1lbnRzLWNvbW1vblwiO1xuZXhwb3J0ICogZnJvbSBcIi4vc2luZ2xlLW5vZGUtcmVzdGF0ZS1kZXBsb3ltZW50XCI7XG5leHBvcnQgKiBmcm9tIFwiLi9mYXJnYXRlLXJlc3RhdGUtZGVwbG95bWVudFwiO1xuIl19
|
|
32
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9saWIvcmVzdGF0ZS1jb25zdHJ1Y3RzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQTs7Ozs7Ozs7O0dBU0c7Ozs7Ozs7Ozs7Ozs7Ozs7QUFFSCxxREFBbUM7QUFDbkMsd0RBQXNDO0FBQ3RDLDhEQUE0QztBQUM1Qyx1REFBcUM7QUFDckMsbUVBQWlEIiwic291cmNlc0NvbnRlbnQiOlsiLypcbiAqIENvcHlyaWdodCAoYykgMjAyMyAtIFJlc3RhdGUgU29mdHdhcmUsIEluYy4sIFJlc3RhdGUgR21iSFxuICpcbiAqIFRoaXMgZmlsZSBpcyBwYXJ0IG9mIHRoZSBSZXN0YXRlIFNESyBmb3IgTm9kZS5qcy9UeXBlU2NyaXB0LFxuICogd2hpY2ggaXMgcmVsZWFzZWQgdW5kZXIgdGhlIE1JVCBsaWNlbnNlLlxuICpcbiAqIFlvdSBjYW4gZmluZCBhIGNvcHkgb2YgdGhlIGxpY2Vuc2UgaW4gZmlsZSBMSUNFTlNFIGluIHRoZSByb290XG4gKiBkaXJlY3Rvcnkgb2YgdGhpcyByZXBvc2l0b3J5IG9yIHBhY2thZ2UsIG9yIGF0XG4gKiBodHRwczovL2dpdGh1Yi5jb20vcmVzdGF0ZWRldi9zZGstdHlwZXNjcmlwdC9ibG9iL21haW4vTElDRU5TRVxuICovXG5cbmV4cG9ydCAqIGZyb20gXCIuL3NlcnZpY2UtZGVwbG95ZXJcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3Jlc3RhdGUtZW52aXJvbm1lbnRcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3Jlc3RhdGUtY2xvdWQtZW52aXJvbm1lbnRcIjtcbmV4cG9ydCAqIGZyb20gXCIuL2RlcGxveW1lbnRzLWNvbW1vblwiO1xuZXhwb3J0ICogZnJvbSBcIi4vc2luZ2xlLW5vZGUtcmVzdGF0ZS1kZXBsb3ltZW50XCI7XG4iXX0=
|
|
@@ -1,16 +1,33 @@
|
|
|
1
1
|
import { Handler } from "aws-lambda/handler";
|
|
2
2
|
import { CloudFormationCustomResourceEvent } from "aws-lambda/trigger/cloudformation-custom-resource";
|
|
3
3
|
import * as cdk from "aws-cdk-lib";
|
|
4
|
+
/**
|
|
5
|
+
* Custom Resource event shape for registering Restate Lambda service handlers with a Restate environment.
|
|
6
|
+
*/
|
|
4
7
|
export interface RegistrationProperties {
|
|
5
|
-
|
|
8
|
+
/** Where to find the Restate admin endpoint. */
|
|
6
9
|
adminUrl?: string;
|
|
10
|
+
/**
|
|
11
|
+
* Optional service name to look for in the deployment. If more than one service is behind the same endpoint, any one
|
|
12
|
+
* should match. Leave unset to skip the check.
|
|
13
|
+
*/
|
|
14
|
+
servicePath?: string;
|
|
7
15
|
serviceLambdaArn?: string;
|
|
8
16
|
invokeRoleArn?: string;
|
|
9
|
-
|
|
17
|
+
/**
|
|
18
|
+
* Authentication token ARN to use with the admin endpoint. The secret value will be used as a bearer token, if set.
|
|
19
|
+
*/
|
|
10
20
|
authTokenSecretArn?: string;
|
|
21
|
+
/** Not used by the handler, purely used to trick CloudFormation to perform an update when it otherwise would not. */
|
|
11
22
|
configurationVersion?: string;
|
|
23
|
+
/**
|
|
24
|
+
* Whether to mark the service as private, and make it unavailable to be called via Restate ingress. If there are
|
|
25
|
+
* multiple services provided by the endpoint, they will all be marked as specified.
|
|
26
|
+
*/
|
|
12
27
|
private?: "true" | "false";
|
|
28
|
+
/** Whether to trust any certificate when connecting to the admin endpoint. */
|
|
13
29
|
insecure?: "true" | "false";
|
|
30
|
+
removalPolicy?: cdk.RemovalPolicy;
|
|
14
31
|
}
|
|
15
32
|
/**
|
|
16
33
|
* Custom Resource event handler for Restate service registration. This handler backs the custom resources created by
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
/*
|
|
3
|
-
* Copyright (c) 2023 - Restate Software, Inc., Restate GmbH
|
|
3
|
+
* Copyright (c) 2023-2024 - Restate Software, Inc., Restate GmbH
|
|
4
4
|
*
|
|
5
5
|
* This file is part of the Restate SDK for Node.js/TypeScript,
|
|
6
6
|
* which is released under the MIT license.
|
|
@@ -44,10 +44,8 @@ const https = __importStar(require("node:https"));
|
|
|
44
44
|
const http = __importStar(require("node:http"));
|
|
45
45
|
const MAX_HEALTH_CHECK_ATTEMPTS = 5; // This is intentionally quite long to allow some time for first-run EC2 and Docker boot up
|
|
46
46
|
const MAX_REGISTRATION_ATTEMPTS = 3;
|
|
47
|
-
// const INSECURE = true;
|
|
48
47
|
const DEPLOYMENTS_PATH = "deployments";
|
|
49
48
|
const SERVICES_PATH = "services";
|
|
50
|
-
const DEPLOYMENTS_PATH_LEGACY = "endpoints"; // temporarily fall back for legacy clusters
|
|
51
49
|
/**
|
|
52
50
|
* Custom Resource event handler for Restate service registration. This handler backs the custom resources created by
|
|
53
51
|
* {@link ServiceDeployer} to facilitate Lambda service handler discovery.
|
|
@@ -131,15 +129,15 @@ const handler = async function (event) {
|
|
|
131
129
|
console.log(`Retrying after ${waitTimeMillis} ms...`);
|
|
132
130
|
await sleep(waitTimeMillis);
|
|
133
131
|
}
|
|
134
|
-
|
|
132
|
+
const deploymentsUrl = `${props.adminUrl}/${DEPLOYMENTS_PATH}`;
|
|
135
133
|
const registrationRequest = JSON.stringify({
|
|
136
134
|
arn: props.serviceLambdaArn,
|
|
137
135
|
assume_role_arn: props.invokeRoleArn,
|
|
138
136
|
});
|
|
139
137
|
let failureReason;
|
|
140
138
|
attempt = 1;
|
|
141
|
-
console.log(`
|
|
142
|
-
while (true) {
|
|
139
|
+
console.log(`Registering services at ${deploymentsUrl}: ${registrationRequest}`);
|
|
140
|
+
registration_retry_loop: while (true) {
|
|
143
141
|
try {
|
|
144
142
|
console.log(`Making registration request #${attempt}...`);
|
|
145
143
|
const controller = new AbortController();
|
|
@@ -154,41 +152,43 @@ const handler = async function (event) {
|
|
|
154
152
|
},
|
|
155
153
|
agent: agentSelector,
|
|
156
154
|
}).finally(() => clearTimeout(registerCallTimeout));
|
|
157
|
-
if (registerDeploymentResponse.status == 404 && attempt == 1) {
|
|
158
|
-
deploymentsUrl = `${props.adminUrl}/${DEPLOYMENTS_PATH_LEGACY}`;
|
|
159
|
-
console.log(`Got 404, falling back to <0.7.0 legacy endpoint registration at: ${deploymentsUrl}`);
|
|
160
|
-
}
|
|
161
155
|
if (registerDeploymentResponse.status >= 200 && registerDeploymentResponse.status < 300) {
|
|
162
156
|
const response = (await registerDeploymentResponse.json());
|
|
163
|
-
|
|
164
|
-
if (!response?.services?.find((s) => s.name === props.servicePath)) {
|
|
157
|
+
if (props.servicePath && !response.services.find((s) => s.name === props.servicePath)) {
|
|
165
158
|
failureReason =
|
|
166
|
-
"
|
|
167
|
-
|
|
159
|
+
`"Registration succeeded, but none the services names in the deployment matched the specified name. " +
|
|
160
|
+
"Expected \"${props.servicePath}\"", got back: [` + response.services.map((svc) => svc?.name).join(", ");
|
|
161
|
+
`]`;
|
|
168
162
|
attempt = MAX_REGISTRATION_ATTEMPTS; // don't retry this
|
|
169
163
|
break;
|
|
170
164
|
}
|
|
171
|
-
console.log("Successful registration!");
|
|
165
|
+
console.log("Successful registration! Services: ", JSON.stringify(response.services));
|
|
172
166
|
const isPublic = (props.private ?? "false") === "false";
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
167
|
+
for (const service of response.services ?? []) {
|
|
168
|
+
if (service.public === isPublic) {
|
|
169
|
+
console.log(`Service ${service.name} is ${isPublic ? "public" : "private"}.`);
|
|
170
|
+
continue;
|
|
171
|
+
}
|
|
172
|
+
console.log(`Marking service ${service.name} as ${isPublic ? "public" : "private"}...`);
|
|
173
|
+
const controller = new AbortController();
|
|
174
|
+
const privateCallTimeout = setTimeout(() => controller.abort("timeout"), 10000);
|
|
175
|
+
const patchResponse = await (0, node_fetch_1.default)(`${props.adminUrl}/${SERVICES_PATH}/${service.name}`, {
|
|
176
|
+
signal: controller.signal,
|
|
177
|
+
method: "PATCH",
|
|
178
|
+
headers: {
|
|
179
|
+
"Content-Type": "application/json",
|
|
180
|
+
...authHeader,
|
|
181
|
+
},
|
|
182
|
+
body: JSON.stringify({ public: isPublic }),
|
|
183
|
+
agent: agentSelector,
|
|
184
|
+
}).finally(() => clearTimeout(privateCallTimeout));
|
|
185
|
+
console.log(`Got patch response back: ${patchResponse.status}`);
|
|
186
|
+
if (patchResponse.status != 200) {
|
|
187
|
+
failureReason = `Marking service as ${props.private ? "private" : "public"} failed: ${patchResponse.statusText} (${patchResponse.status})`;
|
|
188
|
+
break registration_retry_loop; // don't throw immediately - let retry loop decide whether to abort s
|
|
189
|
+
}
|
|
190
|
+
console.log(`Successfully marked service as ${isPublic ? "public" : "private"}.`);
|
|
190
191
|
}
|
|
191
|
-
console.log(`Successfully marked service as ${isPublic ? "public" : "private"}.`);
|
|
192
192
|
return; // Overall success!
|
|
193
193
|
}
|
|
194
194
|
else {
|
|
@@ -235,4 +235,4 @@ async function createAuthHeader(props) {
|
|
|
235
235
|
async function sleep(millis) {
|
|
236
236
|
return new Promise((resolve) => setTimeout(resolve, millis));
|
|
237
237
|
}
|
|
238
|
-
//# 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 \"node:https\";\nimport * as http from \"node:http\";\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 ServiceDeployer} to facilitate Lambda service handler discovery.\n */\nexport const handler: Handler<CloudFormationCustomResourceEvent, void> = async function (event) {\n  console.log({ event });\n\n  const props = event.ResourceProperties as RegistrationProperties;\n\n  const httpAgent = new http.Agent({\n    keepAlive: true,\n  });\n  const httpsAgent = new https.Agent({\n    keepAlive: true,\n    rejectUnauthorized: props.insecure !== \"true\",\n  });\n  const agentSelector = (url: URL) => {\n    if (url.protocol == \"http:\") {\n      return httpAgent;\n    } else {\n      return httpsAgent;\n    }\n  };\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: agentSelector,\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 authHeader = await createAuthHeader(props);\n\n  let attempt;\n\n  const healthCheckUrl = `${props.adminUrl}/health`;\n\n  attempt = 1;\n  console.log(`Performing health check against: ${healthCheckUrl}`);\n  while (true) {\n    console.log(`Making health check request #${attempt}...`);\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: agentSelector,\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  attempt = 1;\n  console.log(`Triggering registration at ${deploymentsUrl}: ${registrationRequest}`);\n  while (true) {\n    try {\n      console.log(`Making registration 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: agentSelector,\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?.find((s) => s.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\n          attempt = MAX_REGISTRATION_ATTEMPTS; // don't retry this\n          break;\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\"), 10_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: agentSelector,\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        const errorBody = await registerDeploymentResponse.text();\n        failureReason = `Registration failed (${registerDeploymentResponse.status}): ${errorBody}`;\n        console.log({\n          message: `Got error response from Restate.`,\n          code: registerDeploymentResponse.status,\n          body: errorBody,\n        });\n      }\n    } catch (e) {\n      console.error(`Registration failed: ${(e as Error)?.message} (attempt ${attempt})`);\n      failureReason = (e as Error)?.message;\n    }\n\n    if (attempt >= MAX_REGISTRATION_ATTEMPTS) {\n      failureReason = `Giving up after ${attempt} attempts. Last error: ${failureReason}`;\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"]}
|
|
238
|
+
//# 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-2024 - 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 \"node:https\";\nimport * as http from \"node:http\";\n\n/**\n * Custom Resource event shape for registering Restate Lambda service handlers with a Restate environment.\n */\nexport interface RegistrationProperties {\n  /** Where to find the Restate admin endpoint. */\n  adminUrl?: string;\n\n  /**\n   * Optional service name to look for in the deployment. If more than one service is behind the same endpoint, any one\n   * should match. Leave unset to skip the check.\n   */\n  servicePath?: string;\n\n  serviceLambdaArn?: string;\n\n  invokeRoleArn?: string;\n\n  /**\n   * Authentication token ARN to use with the admin endpoint. The secret value will be used as a bearer token, if set.\n   */\n  authTokenSecretArn?: string;\n\n  /** Not used by the handler, purely used to trick CloudFormation to perform an update when it otherwise would not. */\n  configurationVersion?: string;\n\n  /**\n   * Whether to mark the service as private, and make it unavailable to be called via Restate ingress. If there are\n   * multiple services provided by the endpoint, they will all be marked as specified.\n   */\n  private?: \"true\" | \"false\";\n\n  /** Whether to trust any certificate when connecting to the admin endpoint. */\n  insecure?: \"true\" | \"false\";\n\n  removalPolicy?: cdk.RemovalPolicy;\n}\n\ntype RegisterDeploymentResponse = {\n  id: string;\n  services: { name: string; revision: number; public: boolean }[];\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 DEPLOYMENTS_PATH = \"deployments\";\nconst SERVICES_PATH = \"services\";\n\n/**\n * Custom Resource event handler for Restate service registration. This handler backs the custom resources created by\n * {@link ServiceDeployer} to facilitate Lambda service handler discovery.\n */\nexport const handler: Handler<CloudFormationCustomResourceEvent, void> = async function (event) {\n  console.log({ event });\n\n  const props = event.ResourceProperties as RegistrationProperties;\n\n  const httpAgent = new http.Agent({\n    keepAlive: true,\n  });\n  const httpsAgent = new https.Agent({\n    keepAlive: true,\n    rejectUnauthorized: props.insecure !== \"true\",\n  });\n  const agentSelector = (url: URL) => {\n    if (url.protocol == \"http:\") {\n      return httpAgent;\n    } else {\n      return httpsAgent;\n    }\n  };\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: agentSelector,\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 authHeader = await createAuthHeader(props);\n\n  let attempt;\n\n  const healthCheckUrl = `${props.adminUrl}/health`;\n\n  attempt = 1;\n  console.log(`Performing health check against: ${healthCheckUrl}`);\n  while (true) {\n    console.log(`Making health check request #${attempt}...`);\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: agentSelector,\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  const 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  attempt = 1;\n  console.log(`Registering services at ${deploymentsUrl}: ${registrationRequest}`);\n\n  registration_retry_loop: while (true) {\n    try {\n      console.log(`Making registration 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: agentSelector,\n      }).finally(() => clearTimeout(registerCallTimeout));\n\n      if (registerDeploymentResponse.status >= 200 && registerDeploymentResponse.status < 300) {\n        const response = (await registerDeploymentResponse.json()) as RegisterDeploymentResponse;\n\n        if (props.servicePath && !response.services.find((s) => s.name === props.servicePath)) {\n          failureReason =\n            `\"Registration succeeded, but none the services names in the deployment matched the specified name. \" + \n            \"Expected \\\"${props.servicePath}\\\"\", got back: [` + response.services.map((svc) => svc?.name).join(\", \");\n          `]`;\n\n          attempt = MAX_REGISTRATION_ATTEMPTS; // don't retry this\n          break;\n        }\n\n        console.log(\"Successful registration! Services: \", JSON.stringify(response.services));\n\n        const isPublic = (props.private ?? \"false\") === \"false\";\n\n        for (const service of response.services ?? []) {\n          if (service.public === isPublic) {\n            console.log(`Service ${service.name} is ${isPublic ? \"public\" : \"private\"}.`);\n            continue;\n          }\n\n          console.log(`Marking service ${service.name} as ${isPublic ? \"public\" : \"private\"}...`);\n          const controller = new AbortController();\n          const privateCallTimeout = setTimeout(() => controller.abort(\"timeout\"), 10_000);\n          const patchResponse = await fetch(`${props.adminUrl}/${SERVICES_PATH}/${service.name}`, {\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: agentSelector,\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 registration_retry_loop; // 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\n        return; // Overall success!\n      } else {\n        const errorBody = await registerDeploymentResponse.text();\n        failureReason = `Registration failed (${registerDeploymentResponse.status}): ${errorBody}`;\n        console.log({\n          message: `Got error response from Restate.`,\n          code: registerDeploymentResponse.status,\n          body: errorBody,\n        });\n      }\n    } catch (e) {\n      console.error(`Registration failed: ${(e as Error)?.message} (attempt ${attempt})`);\n      failureReason = (e as Error)?.message;\n    }\n\n    if (attempt >= MAX_REGISTRATION_ATTEMPTS) {\n      failureReason = `Giving up after ${attempt} attempts. Last error: ${failureReason}`;\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"]}
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
import * as iam from "aws-cdk-lib/aws-iam";
|
|
2
|
+
import * as secrets from "aws-cdk-lib/aws-secretsmanager";
|
|
3
|
+
import { Construct } from "constructs";
|
|
4
|
+
import { IRestateEnvironment } from "./restate-environment";
|
|
5
|
+
/**
|
|
6
|
+
* Configuration for a Restate Cloud environment.
|
|
7
|
+
*/
|
|
8
|
+
export interface RestateCloudEnvironmentProps {
|
|
9
|
+
/**
|
|
10
|
+
* Unique id of the environment (including the `env_` prefix).
|
|
11
|
+
*/
|
|
12
|
+
readonly environmentId: EnvironmentId;
|
|
13
|
+
/**
|
|
14
|
+
* API key with administrative permissions. Used to manage services to the environment, see {@link ServiceDeployer}.
|
|
15
|
+
*/
|
|
16
|
+
readonly apiKey: secrets.ISecret;
|
|
17
|
+
}
|
|
18
|
+
/**
|
|
19
|
+
* A distinct Restate Cloud environment reference. This is a convenience utility for deploying to the
|
|
20
|
+
* [Restate Cloud](https://cloud.restate.dev/) hosted service.
|
|
21
|
+
*/
|
|
22
|
+
export declare class RestateCloudEnvironment extends Construct implements IRestateEnvironment {
|
|
23
|
+
readonly adminUrl: string;
|
|
24
|
+
readonly ingressUrl: string;
|
|
25
|
+
readonly authToken: secrets.ISecret;
|
|
26
|
+
readonly invokerRole: iam.IRole;
|
|
27
|
+
/**
|
|
28
|
+
* Constructs a Restate Cloud environment reference along with invoker. Note that this construct is only a pointer to
|
|
29
|
+
* an existing Restate Cloud environment and does not create it. However, it does create an invoker role that is used
|
|
30
|
+
* invoking Lambda service handlers. If you would prefer to directly manage the invoker role permissions, you can
|
|
31
|
+
* override the {@link createInvokerRole} method or construct one yourself and define the environment properties with
|
|
32
|
+
* {@link RestateEnvironment.fromAttributes} directly.
|
|
33
|
+
*
|
|
34
|
+
* @param scope parent construct
|
|
35
|
+
* @param id construct id
|
|
36
|
+
* @param props environment properties
|
|
37
|
+
* @returns Restate Cloud environment
|
|
38
|
+
*/
|
|
39
|
+
constructor(scope: Construct, id: string, props: RestateCloudEnvironmentProps);
|
|
40
|
+
/**
|
|
41
|
+
* This role is used by Restate to invoke Lambda service handlers; see https://docs.restate.dev/deploy/cloud for
|
|
42
|
+
* information on deploying services to Restate Cloud environments. For standalone environments, the EC2 instance
|
|
43
|
+
* profile can be used directly instead of creating a separate role.
|
|
44
|
+
*/
|
|
45
|
+
protected createInvokerRole(scope: Construct, props: RestateCloudEnvironmentProps): iam.IRole;
|
|
46
|
+
}
|
|
47
|
+
export type EnvironmentId = `env_${string}`;
|
|
@@ -0,0 +1,100 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/*
|
|
3
|
+
* Copyright (c) 2024 - 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.RestateCloudEnvironment = void 0;
|
|
37
|
+
const iam = __importStar(require("aws-cdk-lib/aws-iam"));
|
|
38
|
+
const constructs_1 = require("constructs");
|
|
39
|
+
/**
|
|
40
|
+
* A distinct Restate Cloud environment reference. This is a convenience utility for deploying to the
|
|
41
|
+
* [Restate Cloud](https://cloud.restate.dev/) hosted service.
|
|
42
|
+
*/
|
|
43
|
+
class RestateCloudEnvironment extends constructs_1.Construct {
|
|
44
|
+
/**
|
|
45
|
+
* Constructs a Restate Cloud environment reference along with invoker. Note that this construct is only a pointer to
|
|
46
|
+
* an existing Restate Cloud environment and does not create it. However, it does create an invoker role that is used
|
|
47
|
+
* invoking Lambda service handlers. If you would prefer to directly manage the invoker role permissions, you can
|
|
48
|
+
* override the {@link createInvokerRole} method or construct one yourself and define the environment properties with
|
|
49
|
+
* {@link RestateEnvironment.fromAttributes} directly.
|
|
50
|
+
*
|
|
51
|
+
* @param scope parent construct
|
|
52
|
+
* @param id construct id
|
|
53
|
+
* @param props environment properties
|
|
54
|
+
* @returns Restate Cloud environment
|
|
55
|
+
*/
|
|
56
|
+
constructor(scope, id, props) {
|
|
57
|
+
super(scope, id);
|
|
58
|
+
this.invokerRole = this.createInvokerRole(this, props);
|
|
59
|
+
this.authToken = props.apiKey;
|
|
60
|
+
this.adminUrl = adminEndpoint(RESTATE_CLOUD_REGION_US, props.environmentId);
|
|
61
|
+
this.ingressUrl = ingressEndpoint(RESTATE_CLOUD_REGION_US, props.environmentId);
|
|
62
|
+
}
|
|
63
|
+
/**
|
|
64
|
+
* This role is used by Restate to invoke Lambda service handlers; see https://docs.restate.dev/deploy/cloud for
|
|
65
|
+
* information on deploying services to Restate Cloud environments. For standalone environments, the EC2 instance
|
|
66
|
+
* profile can be used directly instead of creating a separate role.
|
|
67
|
+
*/
|
|
68
|
+
createInvokerRole(scope, props) {
|
|
69
|
+
const invokerRole = new iam.Role(scope, "InvokerRole", {
|
|
70
|
+
assumedBy: new iam.AccountPrincipal(CONFIG[RESTATE_CLOUD_REGION_US].accountId).withConditions({
|
|
71
|
+
StringEquals: {
|
|
72
|
+
"sts:ExternalId": props.environmentId,
|
|
73
|
+
"aws:PrincipalArn": CONFIG[RESTATE_CLOUD_REGION_US].principalArn,
|
|
74
|
+
},
|
|
75
|
+
}),
|
|
76
|
+
});
|
|
77
|
+
invokerRole.assumeRolePolicy.addStatements(new iam.PolicyStatement({
|
|
78
|
+
principals: [new iam.AccountPrincipal("654654156625")],
|
|
79
|
+
actions: ["sts:TagSession"],
|
|
80
|
+
}));
|
|
81
|
+
return invokerRole;
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
exports.RestateCloudEnvironment = RestateCloudEnvironment;
|
|
85
|
+
function adminEndpoint(region, environmentId) {
|
|
86
|
+
const bareEnvId = environmentId.replace(/^env_/, "");
|
|
87
|
+
return `https://${bareEnvId}.env.${region}.restate.cloud:9070`;
|
|
88
|
+
}
|
|
89
|
+
function ingressEndpoint(region, environmentId) {
|
|
90
|
+
const bareEnvId = environmentId.replace(/^env_/, "");
|
|
91
|
+
return `https://${bareEnvId}.env.${region}.restate.cloud`;
|
|
92
|
+
}
|
|
93
|
+
const RESTATE_CLOUD_REGION_US = "us";
|
|
94
|
+
const CONFIG = {
|
|
95
|
+
us: {
|
|
96
|
+
accountId: "654654156625",
|
|
97
|
+
principalArn: "arn:aws:iam::654654156625:role/RestateCloud",
|
|
98
|
+
},
|
|
99
|
+
};
|
|
100
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"restate-cloud-environment.js","sourceRoot":"","sources":["../lib/restate-constructs/restate-cloud-environment.ts"],"names":[],"mappings":";AAAA;;;;;;;;;GASG;;;;;;;;;;;;;;;;;;;;;;;;;;AAEH,yDAA2C;AAE3C,2CAAuC;AAmBvC;;;GAGG;AACH,MAAa,uBAAwB,SAAQ,sBAAS;IAMpD;;;;;;;;;;;OAWG;IACH,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAmC;QAC3E,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QACjB,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,iBAAiB,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;QACvD,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC,MAAM,CAAC;QAC9B,IAAI,CAAC,QAAQ,GAAG,aAAa,CAAC,uBAAuB,EAAE,KAAK,CAAC,aAAa,CAAC,CAAC;QAC5E,IAAI,CAAC,UAAU,GAAG,eAAe,CAAC,uBAAuB,EAAE,KAAK,CAAC,aAAa,CAAC,CAAC;IAClF,CAAC;IAED;;;;OAIG;IACO,iBAAiB,CAAC,KAAgB,EAAE,KAAmC;QAC/E,MAAM,WAAW,GAAG,IAAI,GAAG,CAAC,IAAI,CAAC,KAAK,EAAE,aAAa,EAAE;YACrD,SAAS,EAAE,IAAI,GAAG,CAAC,gBAAgB,CAAC,MAAM,CAAC,uBAAuB,CAAC,CAAC,SAAS,CAAC,CAAC,cAAc,CAAC;gBAC5F,YAAY,EAAE;oBACZ,gBAAgB,EAAE,KAAK,CAAC,aAAa;oBACrC,kBAAkB,EAAE,MAAM,CAAC,uBAAuB,CAAC,CAAC,YAAY;iBACjE;aACF,CAAC;SACH,CAAC,CAAC;QACH,WAAW,CAAC,gBAAiB,CAAC,aAAa,CACzC,IAAI,GAAG,CAAC,eAAe,CAAC;YACtB,UAAU,EAAE,CAAC,IAAI,GAAG,CAAC,gBAAgB,CAAC,cAAc,CAAC,CAAC;YACtD,OAAO,EAAE,CAAC,gBAAgB,CAAC;SAC5B,CAAC,CACH,CAAC;QACF,OAAO,WAAW,CAAC;IACrB,CAAC;CACF;AAhDD,0DAgDC;AAED,SAAS,aAAa,CAAC,MAA0B,EAAE,aAA4B;IAC7E,MAAM,SAAS,GAAG,aAAa,CAAC,OAAO,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC;IACrD,OAAO,WAAW,SAAS,QAAQ,MAAM,qBAAqB,CAAC;AACjE,CAAC;AAED,SAAS,eAAe,CAAC,MAA0B,EAAE,aAA4B;IAC/E,MAAM,SAAS,GAAG,aAAa,CAAC,OAAO,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC;IACrD,OAAO,WAAW,SAAS,QAAQ,MAAM,gBAAgB,CAAC;AAC5D,CAAC;AAUD,MAAM,uBAAuB,GAAG,IAAI,CAAC;AAErC,MAAM,MAAM,GAAG;IACb,EAAE,EAAE;QACF,SAAS,EAAE,cAAc;QACzB,YAAY,EAAE,6CAA6C;KAC5D;CAC0C,CAAC","sourcesContent":["/*\n * Copyright (c) 2024 - 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 iam from \"aws-cdk-lib/aws-iam\";\nimport * as secrets from \"aws-cdk-lib/aws-secretsmanager\";\nimport { Construct } from \"constructs\";\nimport { IRestateEnvironment, RestateEnvironment } from \"./restate-environment\";\nimport { ServiceDeployer } from \"./service-deployer\";\n\n/**\n * Configuration for a Restate Cloud environment.\n */\nexport interface RestateCloudEnvironmentProps {\n  /**\n   * Unique id of the environment (including the `env_` prefix).\n   */\n  readonly environmentId: EnvironmentId;\n\n  /**\n   * API key with administrative permissions. Used to manage services to the environment, see {@link ServiceDeployer}.\n   */\n  readonly apiKey: secrets.ISecret;\n}\n\n/**\n * A distinct Restate Cloud environment reference. This is a convenience utility for deploying to the\n * [Restate Cloud](https://cloud.restate.dev/) hosted service.\n */\nexport class RestateCloudEnvironment extends Construct implements IRestateEnvironment {\n  readonly adminUrl: string;\n  readonly ingressUrl: string;\n  readonly authToken: secrets.ISecret;\n  readonly invokerRole: iam.IRole;\n\n  /**\n   * Constructs a Restate Cloud environment reference along with invoker. Note that this construct is only a pointer to\n   * an existing Restate Cloud environment and does not create it. However, it does create an invoker role that is used\n   * invoking Lambda service handlers. If you would prefer to directly manage the invoker role permissions, you can\n   * override the {@link createInvokerRole} method or construct one yourself and define the environment properties with\n   * {@link RestateEnvironment.fromAttributes} directly.\n   *\n   * @param scope parent construct\n   * @param id construct id\n   * @param props environment properties\n   * @returns Restate Cloud environment\n   */\n  constructor(scope: Construct, id: string, props: RestateCloudEnvironmentProps) {\n    super(scope, id);\n    this.invokerRole = this.createInvokerRole(this, props);\n    this.authToken = props.apiKey;\n    this.adminUrl = adminEndpoint(RESTATE_CLOUD_REGION_US, props.environmentId);\n    this.ingressUrl = ingressEndpoint(RESTATE_CLOUD_REGION_US, props.environmentId);\n  }\n\n  /**\n   * This role is used by Restate to invoke Lambda service handlers; see https://docs.restate.dev/deploy/cloud for\n   * information on deploying services to Restate Cloud environments. For standalone environments, the EC2 instance\n   * profile can be used directly instead of creating a separate role.\n   */\n  protected createInvokerRole(scope: Construct, props: RestateCloudEnvironmentProps): iam.IRole {\n    const invokerRole = new iam.Role(scope, \"InvokerRole\", {\n      assumedBy: new iam.AccountPrincipal(CONFIG[RESTATE_CLOUD_REGION_US].accountId).withConditions({\n        StringEquals: {\n          \"sts:ExternalId\": props.environmentId,\n          \"aws:PrincipalArn\": CONFIG[RESTATE_CLOUD_REGION_US].principalArn,\n        },\n      }),\n    });\n    invokerRole.assumeRolePolicy!.addStatements(\n      new iam.PolicyStatement({\n        principals: [new iam.AccountPrincipal(\"654654156625\")],\n        actions: [\"sts:TagSession\"],\n      }),\n    );\n    return invokerRole;\n  }\n}\n\nfunction adminEndpoint(region: RestateCloudRegion, environmentId: EnvironmentId): string {\n  const bareEnvId = environmentId.replace(/^env_/, \"\");\n  return `https://${bareEnvId}.env.${region}.restate.cloud:9070`;\n}\n\nfunction ingressEndpoint(region: RestateCloudRegion, environmentId: EnvironmentId): string {\n  const bareEnvId = environmentId.replace(/^env_/, \"\");\n  return `https://${bareEnvId}.env.${region}.restate.cloud`;\n}\n\nexport type EnvironmentId = `env_${string}`;\ntype RestateCloudRegion = \"us\";\n\ninterface RegionConfig {\n  accountId: string;\n  principalArn: string;\n}\n\nconst RESTATE_CLOUD_REGION_US = \"us\";\n\nconst CONFIG = {\n  us: {\n    accountId: \"654654156625\",\n    principalArn: \"arn:aws:iam::654654156625:role/RestateCloud\",\n  },\n} as Record<RestateCloudRegion, RegionConfig>;\n"]}
|
|
@@ -1,7 +1,5 @@
|
|
|
1
1
|
import * as iam from "aws-cdk-lib/aws-iam";
|
|
2
|
-
import
|
|
3
|
-
import * as secretsmanager from "aws-cdk-lib/aws-secretsmanager";
|
|
4
|
-
import { ISecret } from "aws-cdk-lib/aws-secretsmanager";
|
|
2
|
+
import * as secrets from "aws-cdk-lib/aws-secretsmanager";
|
|
5
3
|
import { FunctionOptions } from "aws-cdk-lib/aws-lambda";
|
|
6
4
|
import { ServiceDeployer } from "./service-deployer";
|
|
7
5
|
/**
|
|
@@ -22,12 +20,17 @@ export interface IRestateEnvironment extends Pick<FunctionOptions, "vpc" | "vpcS
|
|
|
22
20
|
/**
|
|
23
21
|
* Authentication token to include as a bearer token in requests to the admin endpoint.
|
|
24
22
|
*/
|
|
25
|
-
readonly authToken?:
|
|
23
|
+
readonly authToken?: secrets.ISecret;
|
|
26
24
|
}
|
|
25
|
+
/**
|
|
26
|
+
* A reference to a Restate Environment that can be used as a target for deploying services. Use {@link fromAttributes}
|
|
27
|
+
* to instantiate an arbitrary pointer to an existing environment, or one of the {@link SingleNodeRestateDeployment} or
|
|
28
|
+
* {@link RestateCloudEnvironment} convenience classes.
|
|
29
|
+
*/
|
|
27
30
|
export declare class RestateEnvironment implements IRestateEnvironment {
|
|
28
31
|
readonly adminUrl: string;
|
|
29
|
-
readonly authToken?: ISecret;
|
|
30
|
-
readonly invokerRole?: IRole;
|
|
32
|
+
readonly authToken?: secrets.ISecret;
|
|
33
|
+
readonly invokerRole?: iam.IRole;
|
|
31
34
|
readonly serviceDeployer: ServiceDeployer;
|
|
32
35
|
private constructor();
|
|
33
36
|
static fromAttributes(props: IRestateEnvironment): IRestateEnvironment;
|