@restatedev/restate-cdk 0.1.0 → 0.3.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -0
- package/README.md +5 -1
- package/dist/index.js +11 -1
- package/dist/lambda-service-registry.js +11 -1
- package/dist/register-service-handler/index.js +11 -1
- package/dist/single-node-restate-instance.d.ts +20 -5
- package/dist/single-node-restate-instance.js +40 -9
- package/package.json +6 -2
- package/test/.keep +0 -0
package/LICENSE
ADDED
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
MIT License
|
|
2
|
+
|
|
3
|
+
Copyright (c) 2023 - Restate Software, Inc., Restate GmbH
|
|
4
|
+
|
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
6
|
+
of this software and associated documentation files (the "Software"), to deal
|
|
7
|
+
in the Software without restriction, including without limitation the rights
|
|
8
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
9
|
+
copies of the Software, and to permit persons to whom the Software is
|
|
10
|
+
furnished to do so, subject to the following conditions:
|
|
11
|
+
|
|
12
|
+
The above copyright notice and this permission notice shall be included in all
|
|
13
|
+
copies or substantial portions of the Software.
|
|
14
|
+
|
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
16
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
17
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
18
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
19
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
20
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
21
|
+
SOFTWARE
|
package/README.md
CHANGED
|
@@ -1,3 +1,7 @@
|
|
|
1
|
+
[](https://docs.restate.dev)
|
|
2
|
+
[](https://discord.gg/skW3AZ6uGd)
|
|
3
|
+
[](https://twitter.com/intent/follow?screen_name=restatedev)
|
|
4
|
+
|
|
1
5
|
# Restate CDK support
|
|
2
6
|
|
|
3
|
-
CDK construct library for deploying [Restate](https://restate.dev) and Restate services on AWS.
|
|
7
|
+
CDK construct library for deploying [Restate](https://restate.dev) and Restate services on AWS.
|
package/dist/index.js
CHANGED
|
@@ -1,4 +1,14 @@
|
|
|
1
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
|
+
*/
|
|
2
12
|
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
13
|
if (k2 === undefined) k2 = k;
|
|
4
14
|
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
@@ -16,4 +26,4 @@ var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
|
16
26
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
17
27
|
__exportStar(require("./lambda-service-registry"), exports);
|
|
18
28
|
__exportStar(require("./single-node-restate-instance"), exports);
|
|
19
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
29
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9saWIvcmVzdGF0ZS1jb25zdHJ1Y3RzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQTs7Ozs7Ozs7O0dBU0c7Ozs7Ozs7Ozs7Ozs7Ozs7QUFFSCw0REFBMEM7QUFDMUMsaUVBQStDIiwic291cmNlc0NvbnRlbnQiOlsiLypcbiAqIENvcHlyaWdodCAoYykgMjAyMyAtIFJlc3RhdGUgU29mdHdhcmUsIEluYy4sIFJlc3RhdGUgR21iSFxuICpcbiAqIFRoaXMgZmlsZSBpcyBwYXJ0IG9mIHRoZSBSZXN0YXRlIFNESyBmb3IgTm9kZS5qcy9UeXBlU2NyaXB0LFxuICogd2hpY2ggaXMgcmVsZWFzZWQgdW5kZXIgdGhlIE1JVCBsaWNlbnNlLlxuICpcbiAqIFlvdSBjYW4gZmluZCBhIGNvcHkgb2YgdGhlIGxpY2Vuc2UgaW4gZmlsZSBMSUNFTlNFIGluIHRoZSByb290XG4gKiBkaXJlY3Rvcnkgb2YgdGhpcyByZXBvc2l0b3J5IG9yIHBhY2thZ2UsIG9yIGF0XG4gKiBodHRwczovL2dpdGh1Yi5jb20vcmVzdGF0ZWRldi9zZGstdHlwZXNjcmlwdC9ibG9iL21haW4vTElDRU5TRVxuICovXG5cbmV4cG9ydCAqIGZyb20gXCIuL2xhbWJkYS1zZXJ2aWNlLXJlZ2lzdHJ5XCI7XG5leHBvcnQgKiBmcm9tIFwiLi9zaW5nbGUtbm9kZS1yZXN0YXRlLWluc3RhbmNlXCI7Il19
|
|
@@ -1,4 +1,14 @@
|
|
|
1
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
|
+
*/
|
|
2
12
|
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
13
|
if (k2 === undefined) k2 = k;
|
|
4
14
|
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
@@ -82,4 +92,4 @@ class RestateServiceRegistrar extends constructs_1.Construct {
|
|
|
82
92
|
});
|
|
83
93
|
}
|
|
84
94
|
}
|
|
85
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
95
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lambda-service-registry.js","sourceRoot":"","sources":["../lib/restate-constructs/lambda-service-registry.ts"],"names":[],"mappings":";AAAA;;;;;;;;;GASG;;;;;;;;;;;;;;;;;;;;;;;;;;AAEH,iDAAmC;AACnC,yDAA2C;AAE3C,2CAAuC;AA4BvC;;;;GAIG;AACH,MAAa,qBAAsB,SAAQ,sBAAS;IAIlD,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAiC;QACzE,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,IAAI,CAAC,eAAe,GAAG,KAAK,CAAC,eAAe,CAAC;QAC7C,IAAI,CAAC,yBAAyB,GAAG,KAAK,CAAC,yBAAyB,CAAC;IACnE,CAAC;IAEM,QAAQ,CAAC,OAA2B;QACzC,MAAM,mBAAmB,GAAG,IAAI,GAAG,CAAC,MAAM,CAAC,IAAI,EAAE,qBAAqB,EAAE;YACtE,UAAU,EAAE;gBACV,IAAI,GAAG,CAAC,eAAe,CAAC;oBACtB,GAAG,EAAE,+BAA+B;oBACpC,OAAO,EAAE,CAAC,uBAAuB,CAAC;oBAClC,SAAS,EAAE,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,eAAe,CAAC;yBAC3C,GAAG,CAAC,OAAO,CAAC,EAAE,CAAC,OAAO,CAAC,WAAW,GAAG,IAAI,CAAC;iBAC9C,CAAC;aACH;SACF,CAAC,CAAC;QAEH,MAAM,WAAW,GAAG,GAAG,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,EAAE,aAAa,EAAE,OAAO,CAAC,cAAc,CAAC,CAAC;QACtF,WAAW,CAAC,kBAAkB,CAAC,mBAAmB,CAAC,CAAC;QAEpD,KAAK,MAAM,CAAC,IAAI,EAAE,OAAO,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,eAAe,CAAC,EAAE;YAClE,IAAI,CAAC,eAAe,CAAC,OAAO,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE,EAAE,mBAAmB,CAAC,CAAC;SACvE;IACH,CAAC;IAEO,eAAe,CAAC,OAA2B,EAAE,OAGpD,EAAE,mBAA+B;QAChC,MAAM,SAAS,GAAG,IAAI,uBAAuB,CAAC,IAAI,EAAE,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,GAAG,WAAW,EAAE;YACzF,OAAO;YACP,OAAO;YACP,YAAY,EAAE,IAAI,CAAC,yBAAyB;SAC7C,CAAC,CAAC;QAEH,6GAA6G;QAC7G,iGAAiG;QACjG,SAAS,CAAC,IAAI,CAAC,aAAa,CAAC,mBAAmB,CAAC,CAAC;IACpD,CAAC;CACF;AA7CD,sDA6CC;AAED,MAAM,uBAAwB,SAAQ,sBAAS;IAC7C,YAAY,KAAgB,EAAE,EAAU,EAC5B,KAOC;QAEX,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,IAAI,GAAG,CAAC,cAAc,CAAC,IAAI,EAAE,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,GAAG,WAAW,EAAE;YACxE,YAAY,EAAE,KAAK,CAAC,YAAY;YAChC,YAAY,EAAE,iCAAiC;YAC/C,UAAU,EAAE;gBACV,WAAW,EAAE,KAAK,CAAC,OAAO,CAAC,IAAI;gBAC/B,YAAY,EAAE,KAAK,CAAC,OAAO,CAAC,YAAY;gBACxC,gBAAgB,EAAE,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,cAAc,CAAC,WAAW;gBAClE,aAAa,EAAE,GAAG,CAAC,aAAa,CAAC,OAAO;aACR;SACnC,CAAC,CAAC;IACL,CAAC;CACF","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\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}\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  registrationProviderToken: string;\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    this.serviceHandlers = props.serviceHandlers;\n    this.registrationProviderToken = props.registrationProviderToken;\n  }\n\n  public register(restate: RestateInstanceRef) {\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    const invokerRole = iam.Role.fromRoleArn(this, \"InvokerRole\", restate.invokerRoleArn);\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        serviceLambdaArn: props.service.handler.currentVersion.functionArn,\n        removalPolicy: cdk.RemovalPolicy.DESTROY,\n      } satisfies RegistrationProperties,\n    });\n  }\n}"]}
|
|
@@ -1,4 +1,14 @@
|
|
|
1
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
|
+
*/
|
|
2
12
|
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
13
|
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
14
|
};
|
|
@@ -96,4 +106,4 @@ const handler = async function (event) {
|
|
|
96
106
|
};
|
|
97
107
|
};
|
|
98
108
|
exports.handler = handler;
|
|
99
|
-
//# 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":["import { 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  };"]}
|
|
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  };"]}
|
|
@@ -4,6 +4,7 @@ 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
6
|
import * as cr from "aws-cdk-lib/custom-resources";
|
|
7
|
+
import * as acm from "aws-cdk-lib/aws-certificatemanager";
|
|
7
8
|
/**
|
|
8
9
|
* Represents an instance of the Restate service. This could represent a self-hosted broker, or Restate's managed
|
|
9
10
|
* service.
|
|
@@ -13,6 +14,24 @@ export interface RestateInstance {
|
|
|
13
14
|
readonly metaEndpoint: string;
|
|
14
15
|
readonly registrationProviderToken: cdk.CfnOutput;
|
|
15
16
|
}
|
|
17
|
+
export declare enum TracingMode {
|
|
18
|
+
DISABLED = "DISABLED",
|
|
19
|
+
AWS_XRAY = "AWS_XRAY"
|
|
20
|
+
}
|
|
21
|
+
export interface RestateInstanceProps {
|
|
22
|
+
/** Log group for Restate service logs. */
|
|
23
|
+
logGroup: logs.LogGroup;
|
|
24
|
+
/** Tracing mode for Restate services. Disabled by default. */
|
|
25
|
+
tracing?: TracingMode;
|
|
26
|
+
/** Prefix for resources created by this construct that require unique names. */
|
|
27
|
+
prefix?: string;
|
|
28
|
+
/** Restate Docker image tag. Defaults to `latest`. */
|
|
29
|
+
restateTag?: string;
|
|
30
|
+
/** Amazon Distro for Open Telemetry Docker image tag. Defaults to `latest`. */
|
|
31
|
+
adotTag?: string;
|
|
32
|
+
/** Optional certificate for ingress endpoint. If unspecified, a plain HTTP listener will be created. */
|
|
33
|
+
certificate?: acm.ICertificate;
|
|
34
|
+
}
|
|
16
35
|
/**
|
|
17
36
|
* Creates a Restate service deployment backed by a single EC2 instance,
|
|
18
37
|
* suitable for development and testing purposes.
|
|
@@ -26,11 +45,7 @@ export declare class SingleNodeRestateInstance extends Construct implements Rest
|
|
|
26
45
|
readonly metaEndpoint: string;
|
|
27
46
|
readonly registrationProvider: cr.Provider;
|
|
28
47
|
readonly registrationProviderToken: cdk.CfnOutput;
|
|
29
|
-
constructor(scope: Construct, id: string, props:
|
|
30
|
-
prefix?: string;
|
|
31
|
-
restateTag?: string;
|
|
32
|
-
logGroup: logs.LogGroup;
|
|
33
|
-
});
|
|
48
|
+
constructor(scope: Construct, id: string, props: RestateInstanceProps);
|
|
34
49
|
/**
|
|
35
50
|
* Creates a custom resource provider to facilitate service discovery. Note that the custom resource event handler
|
|
36
51
|
* must be able to reach the Restate instance's meta endpoint - which is why it is deployed within the same VPC.
|
|
@@ -1,4 +1,14 @@
|
|
|
1
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
|
+
*/
|
|
2
12
|
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
13
|
if (k2 === undefined) k2 = k;
|
|
4
14
|
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
@@ -26,11 +36,12 @@ var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
|
26
36
|
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
27
37
|
};
|
|
28
38
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
29
|
-
exports.SingleNodeRestateInstance = void 0;
|
|
39
|
+
exports.SingleNodeRestateInstance = exports.TracingMode = void 0;
|
|
30
40
|
const constructs_1 = require("constructs");
|
|
31
41
|
const ec2 = __importStar(require("aws-cdk-lib/aws-ec2"));
|
|
32
42
|
const iam = __importStar(require("aws-cdk-lib/aws-iam"));
|
|
33
43
|
const elb_v2 = __importStar(require("aws-cdk-lib/aws-elasticloadbalancingv2"));
|
|
44
|
+
const aws_elasticloadbalancingv2_1 = require("aws-cdk-lib/aws-elasticloadbalancingv2");
|
|
34
45
|
const aws_elasticloadbalancingv2_targets_1 = require("aws-cdk-lib/aws-elasticloadbalancingv2-targets");
|
|
35
46
|
const lambda_node = __importStar(require("aws-cdk-lib/aws-lambda-nodejs"));
|
|
36
47
|
const node_path_1 = __importDefault(require("node:path"));
|
|
@@ -40,6 +51,12 @@ const cr = __importStar(require("aws-cdk-lib/custom-resources"));
|
|
|
40
51
|
const RESTATE_INGRESS_PORT = 8080;
|
|
41
52
|
const RESTATE_META_PORT = 9070;
|
|
42
53
|
const RESTATE_DOCKER_DEFAULT_TAG = "latest";
|
|
54
|
+
const ADOT_DOCKER_DEFAULT_TAG = "latest";
|
|
55
|
+
var TracingMode;
|
|
56
|
+
(function (TracingMode) {
|
|
57
|
+
TracingMode["DISABLED"] = "DISABLED";
|
|
58
|
+
TracingMode["AWS_XRAY"] = "AWS_XRAY";
|
|
59
|
+
})(TracingMode || (exports.TracingMode = TracingMode = {}));
|
|
43
60
|
/**
|
|
44
61
|
* Creates a Restate service deployment backed by a single EC2 instance,
|
|
45
62
|
* suitable for development and testing purposes.
|
|
@@ -58,13 +75,19 @@ class SingleNodeRestateInstance extends constructs_1.Construct {
|
|
|
58
75
|
});
|
|
59
76
|
props.logGroup.grantWrite(this.invokerRole);
|
|
60
77
|
const restateTag = props.restateTag ?? RESTATE_DOCKER_DEFAULT_TAG;
|
|
78
|
+
const adotTag = props.adotTag ?? ADOT_DOCKER_DEFAULT_TAG;
|
|
61
79
|
const restateInitCommands = ec2.UserData.forLinux();
|
|
62
|
-
restateInitCommands.addCommands("
|
|
63
|
-
"
|
|
80
|
+
restateInitCommands.addCommands("yum update -y", "yum install -y docker", "systemctl enable docker.service", "systemctl start docker.service", [
|
|
81
|
+
"docker run --name adot --restart unless-stopped --detach",
|
|
82
|
+
" -p 4317:4317 -p 55680:55680 -p 8889:8888",
|
|
83
|
+
` public.ecr.aws/aws-observability/aws-otel-collector:${adotTag}`,
|
|
84
|
+
].join(""), [
|
|
85
|
+
"docker run --name restate --restart unless-stopped --detach",
|
|
64
86
|
" --volume /var/restate:/target --network=host",
|
|
65
87
|
" -e RESTATE_OBSERVABILITY__LOG__FORMAT=Json -e RUST_LOG=info,restate_worker::partition=warn",
|
|
88
|
+
" -e RESTATE_OBSERVABILITY__TRACING__ENDPOINT=http://localhost:4317",
|
|
66
89
|
` --log-driver=awslogs --log-opt awslogs-group=${props.logGroup.logGroupName}`,
|
|
67
|
-
`
|
|
90
|
+
` docker.io/restatedev/restate:${restateTag}`,
|
|
68
91
|
].join(""));
|
|
69
92
|
const restateInstance = new ec2.Instance(this, "Host", {
|
|
70
93
|
vpc: this.vpc,
|
|
@@ -76,6 +99,11 @@ class SingleNodeRestateInstance extends constructs_1.Construct {
|
|
|
76
99
|
userData: restateInitCommands,
|
|
77
100
|
});
|
|
78
101
|
this.instance = restateInstance;
|
|
102
|
+
// We start the ADOT collector regardless, and only control whether they will be published to X-Ray via instance
|
|
103
|
+
// role permissions. This way historic traces will be buffered on the host, even if tracing is disabled initially.
|
|
104
|
+
if (props.tracing === TracingMode.AWS_XRAY) {
|
|
105
|
+
restateInstance.role.addManagedPolicy(iam.ManagedPolicy.fromAwsManagedPolicyName("AWSXrayWriteOnlyAccess"));
|
|
106
|
+
}
|
|
79
107
|
const restateInstanceSecurityGroup = new ec2.SecurityGroup(this, "RestateSecurityGroup", {
|
|
80
108
|
vpc: this.vpc,
|
|
81
109
|
securityGroupName: "RestateSecurityGroup",
|
|
@@ -87,17 +115,20 @@ class SingleNodeRestateInstance extends constructs_1.Construct {
|
|
|
87
115
|
});
|
|
88
116
|
const targetGroup = new elb_v2.ApplicationTargetGroup(this, "TargetGroup", {
|
|
89
117
|
vpc: this.vpc,
|
|
118
|
+
protocol: elb_v2.ApplicationProtocol.HTTP,
|
|
90
119
|
port: RESTATE_INGRESS_PORT,
|
|
91
120
|
targets: [new aws_elasticloadbalancingv2_targets_1.InstanceTarget(restateInstance)],
|
|
92
121
|
healthCheck: {
|
|
93
|
-
path: "/grpc.health.v1.Health/Check",
|
|
94
122
|
protocol: elb_v2.Protocol.HTTP,
|
|
123
|
+
path: "/grpc.health.v1.Health/Check",
|
|
124
|
+
interval: cdk.Duration.seconds(60),
|
|
95
125
|
},
|
|
96
126
|
});
|
|
97
|
-
// TODO: Make this HTTPS (https://github.com/restatedev/restate-cdk-support/issues/2)
|
|
98
127
|
ingressLoadBalancer.addListener("Listener", {
|
|
99
|
-
port: 80,
|
|
128
|
+
port: props.certificate ? 443 : 80,
|
|
129
|
+
protocol: props.certificate ? aws_elasticloadbalancingv2_1.ApplicationProtocol.HTTPS : aws_elasticloadbalancingv2_1.ApplicationProtocol.HTTP,
|
|
100
130
|
defaultTargetGroups: [targetGroup],
|
|
131
|
+
certificates: props.certificate ? [elb_v2.ListenerCertificate.fromCertificateManager(props.certificate)] : [],
|
|
101
132
|
});
|
|
102
133
|
const albSecurityGroup = new ec2.SecurityGroup(this, "AlbSecurityGroup", {
|
|
103
134
|
vpc: this.vpc,
|
|
@@ -123,7 +154,7 @@ class SingleNodeRestateInstance extends constructs_1.Construct {
|
|
|
123
154
|
exportName: [props.prefix, "RegistrationProviderToken"].join("-"),
|
|
124
155
|
value: registrationProvider.serviceToken,
|
|
125
156
|
});
|
|
126
|
-
this.publicIngressEndpoint =
|
|
157
|
+
this.publicIngressEndpoint = `${props.certificate ? "https" : "http"}://${ingressLoadBalancer.loadBalancerDnsName}`;
|
|
127
158
|
this.privateIngressEndpoint = `http://${this.instance.instancePrivateDnsName}:${RESTATE_INGRESS_PORT}`;
|
|
128
159
|
this.metaEndpoint = `http://${this.instance.instancePrivateDnsName}:${RESTATE_META_PORT}`;
|
|
129
160
|
}
|
|
@@ -153,4 +184,4 @@ class SingleNodeRestateInstance extends constructs_1.Construct {
|
|
|
153
184
|
}
|
|
154
185
|
}
|
|
155
186
|
exports.SingleNodeRestateInstance = SingleNodeRestateInstance;
|
|
156
|
-
//# 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":["import { 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 { 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\";\n\nconst RESTATE_INGRESS_PORT = 8080;\nconst RESTATE_META_PORT = 9070;\nconst RESTATE_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\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: { prefix?: string, restateTag?: string, logGroup: logs.LogGroup }) {\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 restateInitCommands = ec2.UserData.forLinux();\n    restateInitCommands.addCommands(\n      \"sudo yum update -y\",\n      \"sudo yum install -y docker\",\n      \"sudo systemctl enable docker.service\",\n      \"sudo systemctl start docker.service\",\n      [\n        \"sudo docker run --name restate --restart on-failure:10 --detach\",\n        \" --volume /var/restate:/target --network=host\",\n        \" -e RESTATE_OBSERVABILITY__LOG__FORMAT=Json -e RUST_LOG=info,restate_worker::partition=warn\",\n        ` --log-driver=awslogs --log-opt awslogs-group=${props.logGroup.logGroupName}`,\n        ` ghcr.io/restatedev/restate-dist:${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    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      port: RESTATE_INGRESS_PORT,\n      targets: [new InstanceTarget(restateInstance)],\n      healthCheck: {\n        path: \"/grpc.health.v1.Health/Check\",\n        protocol: elb_v2.Protocol.HTTP,\n      },\n    });\n    // TODO: Make this HTTPS (https://github.com/restatedev/restate-cdk-support/issues/2)\n    ingressLoadBalancer.addListener(\"Listener\", {\n      port: 80,\n      defaultTargetGroups: [targetGroup],\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 = `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}"]}
|
|
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}"]}
|
package/package.json
CHANGED
|
@@ -1,10 +1,14 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@restatedev/restate-cdk",
|
|
3
3
|
"description": "Restate.dev CDK constructs",
|
|
4
|
-
"version": "0.
|
|
4
|
+
"version": "0.3.0",
|
|
5
5
|
"author": "Restate Developers",
|
|
6
6
|
"license": "MIT",
|
|
7
7
|
"email": "code@restate.dev",
|
|
8
|
+
"repository": {
|
|
9
|
+
"type": "git",
|
|
10
|
+
"url": "git+https://github.com/restatedev/cdk.git"
|
|
11
|
+
},
|
|
8
12
|
"publishConfig": {
|
|
9
13
|
"@restatedev:registry": "https://registry.npmjs.org"
|
|
10
14
|
},
|
|
@@ -15,7 +19,7 @@
|
|
|
15
19
|
"prebundle": "rm -rf dist",
|
|
16
20
|
"postbundle": "cd dist && zip -r index.zip index.js*",
|
|
17
21
|
"watch": "tsc -w",
|
|
18
|
-
"test": "jest",
|
|
22
|
+
"test": "jest --passWithNoTests",
|
|
19
23
|
"cdk": "cdk"
|
|
20
24
|
},
|
|
21
25
|
"devDependencies": {
|
package/test/.keep
ADDED
|
File without changes
|