@restatedev/restate-cdk 1.1.1 → 1.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,11 @@
1
+ name: "CLA Assistant"
2
+ on:
3
+ issue_comment:
4
+ types: [created]
5
+ pull_request_target:
6
+ types: [opened, closed, synchronize]
7
+
8
+ jobs:
9
+ CLAAssistant:
10
+ uses: restatedev/restate/.github/workflows/cla.yml@main
11
+ secrets: inherit
@@ -1,5 +1,9 @@
1
1
  name: PR Checks (npm)
2
2
 
3
+ permissions:
4
+ contents: read
5
+ pull-requests: read
6
+
3
7
  on:
4
8
  pull_request:
5
9
  types: [opened, synchronize, reopened, ready_for_review]
package/RELEASING.md ADDED
@@ -0,0 +1,3 @@
1
+ # Releasing the package
2
+
3
+ Create and publish a new release in [GitHub](https://github.com/restatedev/cdk/releases).
@@ -38,7 +38,7 @@ export interface SingleNodeRestateProps {
38
38
  tracing?: TracingMode;
39
39
  /** Prefix for resources created by this construct that require unique names. */
40
40
  prefix?: string;
41
- /** Restate Docker image name. Defaults to `latest`. */
41
+ /** Restate Docker image name. Defaults to `docker.restate.dev/restatedev/restate`. */
42
42
  restateImage?: string;
43
43
  /** Restate Docker image tag. Defaults to `latest`. */
44
44
  restateTag?: string;
@@ -121,21 +121,21 @@ class SingleNodeRestateDeployment extends constructs_1.Construct {
121
121
  const adotTag = props.adotTag ?? ADOT_DOCKER_DEFAULT_TAG;
122
122
  this.tlsEnabled = props.tlsTermination === TlsTermination.ON_HOST_SELF_SIGNED_CERTIFICATE;
123
123
  const envDefaults = {
124
- RESTATE_OBSERVABILITY__LOG__FORMAT: "Json",
124
+ RESTATE_LOG_FORMAT: "json",
125
125
  RUST_LOG: "info",
126
- RESTATE_OBSERVABILITY__TRACING__ENDPOINT: "http://localhost:4317",
126
+ RESTATE_TRACING_ENDPOINT: "http://localhost:4317",
127
127
  };
128
128
  const envArgs = Object.entries({ ...envDefaults, ...(props.environment ?? {}) })
129
129
  .map(([key, value]) => `-e ${key}="${value}"`)
130
130
  .join(" ");
131
131
  const initScript = ec2.UserData.forLinux();
132
- initScript.addCommands("set -euf -o pipefail", `yum install -y npm && npm install -gq @restatedev/restate@${restateTag}`, "yum install -y docker", this.mountDataVolumeScript(), "mkdir -p /etc/restate", ["cat << EOF > /etc/restate/config.toml", this.restateConfig(id, props), "EOF"].join("\n"), "systemctl start docker.service",
132
+ initScript.addCommands("set -euf -o pipefail", `yum install -y npm && npm install -gq @restatedev/restate@${restateTag} @restatedev/restatectl@${restateTag}`, "yum install -y docker", this.mountDataVolumeScript(), "mkdir -p /etc/restate", ["cat << EOF > /etc/restate/config.toml", this.restateConfig(id, props), "EOF"].join("\n"), "systemctl start docker.service",
133
133
  // Start the ADOT collector - needed for X-ray trace forwarding
134
- `if [ "$(docker ps -qa -f name=adot)" ]; then docker stop adot || true; docker rm adot; fi`, "docker run --name adot --restart on-failure --detach" +
134
+ `if [ "$(docker ps -qa -f name=adot)" ]; then docker stop adot || true; docker rm adot; fi`, "docker run --name adot --restart on-failure --detach --pull always" +
135
135
  " -p 4317:4317 -p 55680:55680 -p 8889:8888" +
136
136
  ` public.ecr.aws/aws-observability/aws-otel-collector:${adotTag}`,
137
137
  // Start the Restate server container
138
- `if [ "$(docker ps -qa -f name=restate)" ]; then docker stop restate || true; docker rm restate; fi`, "docker run --name restate --restart on-failure --detach" +
138
+ `if [ "$(docker ps -qa -f name=restate)" ]; then docker stop restate || true; docker rm restate; fi`, "docker run --name restate --restart on-failure --detach --pull always" +
139
139
  " --volume /etc/restate:/etc/restate" +
140
140
  " --volume /var/restate:/restate-data" +
141
141
  " --network=host" +
@@ -216,44 +216,22 @@ class SingleNodeRestateDeployment extends constructs_1.Construct {
216
216
  `roles = [`,
217
217
  ` "worker",`,
218
218
  ` "admin",`,
219
- ` "metadata-store",`,
219
+ ` "metadata-server",`,
220
+ ` "log-server",`,
220
221
  `]`,
221
222
  `node-name = "restate-0"`,
222
223
  `cluster-name = "${props.restateConfig?.clusterName ?? id}"`,
223
- `allow-bootstrap = true`,
224
- `bootstrap-num-partitions = ${props.restateConfig?.bootstrapNumPartitions ?? 4}`,
225
- `default-thread-pool-size = 3`,
226
- `storage-high-priority-bg-threads = 3`,
227
- `storage-low-priority-bg-threads = 3`,
224
+ `default-num-partitions = ${props.restateConfig?.bootstrapNumPartitions ?? 4}`,
228
225
  `rocksdb-total-memory-size = "${props.restateConfig?.rocksdb?.totalMemorySize?.toMebibytes() ?? 512.0 + " MB"}"`,
229
- `rocksdb-total-memtables-ratio = 0.60`,
230
- `rocksdb-bg-threads = 3`,
231
- `rocksdb-high-priority-bg-threads = 3`,
232
- ``,
233
- `[worker]`,
234
- `internal-queue-length = 1000`,
235
- ``,
236
- `[worker.storage]`,
237
- `rocksdb-max-background-jobs = 3`,
238
- `rocksdb-statistics-level = "except-detailed-timers"`,
239
- `num-partitions-to-share-memory-budget = 4`,
240
226
  ``,
241
227
  `[admin]`,
242
228
  `bind-address = "${this.tlsEnabled ? "127.0.0.1" : "0.0.0.0"}:${RESTATE_ADMIN_PORT}"`,
243
229
  ``,
244
230
  `[admin.query-engine]`,
245
- `memory-size = "50.0 MB"`,
246
- `query-parallelism = 4`,
231
+ `memory-size = "256.0 MiB"`,
247
232
  ``,
248
233
  `[ingress]`,
249
234
  `bind-address = "${this.tlsEnabled ? "127.0.0.1" : "0.0.0.0"}:${RESTATE_INGRESS_PORT}"`,
250
- `rocksdb-max-background-jobs = 3`,
251
- `rocksdb-statistics-level = "except-detailed-timers"`,
252
- `writer-batch-commit-count = 1000`,
253
- ``,
254
- `[metadata-store.rocksdb]`,
255
- `rocksdb-max-background-jobs = 1`,
256
- `rocksdb-statistics-level = "except-detailed-timers"`,
257
235
  ].join("\n"));
258
236
  }
259
237
  mountDataVolumeScript() {
@@ -329,4 +307,4 @@ fi
329
307
  }
330
308
  }
331
309
  exports.SingleNodeRestateDeployment = SingleNodeRestateDeployment;
332
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"single-node-restate-deployment.js","sourceRoot":"","sources":["../lib/restate-constructs/single-node-restate-deployment.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2023-2025 - Restate Software, Inc., Restate GmbH\n *\n * This file is part of the Restate CDK Construct Library,\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 { IRestateEnvironment } from \"./restate-environment\";\nimport { TracingMode } from \"./deployments-common\";\nimport * as cdk from \"aws-cdk-lib\";\nimport { RemovalPolicy } from \"aws-cdk-lib\";\n\nexport interface SingleNodeRestateProps {\n  /** EC2 instance type to use. */\n  instanceType?: ec2.InstanceType;\n\n  /** Machine image. Note: startup script expects yum-based package management. */\n  machineImage?: ec2.IMachineImage;\n\n  /** The VPC in which to launch the Restate host. */\n  vpc?: ec2.IVpc;\n\n  networkConfiguration?: {\n    /**\n     * Subnet type for the Restate host.\n     *\n     * Available options:\n     *  - [Default] {@link ec2.SubnetType.PRIVATE_WITH_EGRESS} will create the Restate instance with outbound internet\n     *    access only, so that it can invoke HTTP endpoints. The security groups {@link ingressSecurityGroup} and\n     *    {@link adminSecurityGroup} control inbound traffic to the service ingress and admin ports respectively.\n     *    Configure {@link ServiceDeployer} to use the latter, and set up ingress traffic routing outside of this\n     *    construct using the former.\n     *  - Insecure, internet-facing {@link ec2.SubnetType.PUBLIC} will also provision an nginx reverse proxy\n     *    and an HTTP listener with a self-signed certificate.\n     */\n    subnetType?: ec2.SubnetType.PRIVATE_WITH_EGRESS | ec2.SubnetType.PUBLIC;\n  };\n\n  /**\n   * Allow incoming ingress traffic from anywhere. Default: `false`. Alternatively, add rules to the\n   * `ingressSecurityGroup` directly.\n   */\n  publicIngress?: boolean;\n\n  /** Log group for Restate service logs. */\n  logGroup?: logs.LogGroup;\n\n  /** Tracing mode for Restate services. Defaults to {@link TracingMode.DISABLED}. */\n  tracing?: TracingMode;\n\n  /** Prefix for resources created by this construct that require unique names. */\n  prefix?: string;\n\n  /** Restate Docker image name. Defaults to `latest`. */\n  restateImage?: string;\n\n  /** Restate Docker image tag. Defaults to `latest`. */\n  restateTag?: string;\n\n  /**\n   * EBS data volume settings for Restate data storage. If not specified, a default 8GB volume will be created.\n   */\n  dataVolumeOptions?: ec2.EbsDeviceProps | undefined;\n\n  /** Restate high-level configuration options. Alternatively, you can set {@link restateConfigOverride}. */\n  restateConfig?: {\n    /** Defaults to the construct id if left unspecified. */\n    clusterName?: string;\n    /** Defaults to 4. Only takes effect on initial provisioning. */\n    bootstrapNumPartitions?: number;\n    /** RocksDB settings. */\n    rocksdb?: {\n      /** Defaults to 512 MB. */\n      totalMemorySize?: cdk.Size;\n    };\n  };\n\n  /**\n   * Environment properties to set for Restate. This is a simple way to pass custom configuration parameters.\n   */\n  environment?: {\n    [key: string]: string;\n  };\n\n  /**\n   * Completely override the Restate server configuration. Note that other Restate configuration options\n   * will effectively be ignored if this is set. See https://docs.restate.dev/operate/configuration/server/\n   * for details.\n   */\n  restateConfigOverride?: string;\n\n  /** Amazon Distro for Open Telemetry Docker image tag. Defaults to `latest`. */\n  adotTag?: string;\n\n  /**\n   * Removal policy for long-lived resources (storage, logs). Default: `cdk.RemovalPolicy.DESTROY`.\n   */\n  removalPolicy?: cdk.RemovalPolicy;\n\n  /**\n   * Control on-host TLS termination for ingress and admin ports. Defaults to `TlsTermination.NONE`. Currently, enabling\n   * TLS implicitly enables an `nginx` service to be configured on the host. Depending on the value of this option, the\n   * ingress security group will be configured to allow inbound traffic to the appropriate port - 8080 for no TLS, or\n   * 443 with TLS enabled.\n   */\n  tlsTermination?: TlsTermination;\n\n  /**\n   * The read timeout for proxied ingress requests. Default: 3600 seconds.\n   */\n  ingressProxyReadTimeout?: cdk.Duration;\n\n  /**\n   * Completely override the default `nginx` configuration for the ingress proxy. Note that other\n   * ingress proxy configuration options will effectively be ignored if this is set.\n   */\n  ingressNginxConfigOverride?: string;\n}\n\nexport enum TlsTermination {\n  /**\n   * Disabled (default); expose the `restate-server` HTTP ports directly.\n   */\n  DISABLED,\n\n  /**\n   * Use self-signed certificates for TLS termination on the ingress and admin ports. Convenient for quick testing,\n   * make sure you set `insecure` = `true` when using `ServiceDeployer` to accept this certificate.\n   */\n  ON_HOST_SELF_SIGNED_CERTIFICATE,\n}\n\nconst RESTATE_INGRESS_PORT = 8080;\nconst RESTATE_TLS_INGRESS_PORT = 443;\nconst RESTATE_ADMIN_PORT = 9070;\nconst RESTATE_TLS_ADMIN_PORT = 9073;\nconst RESTATE_IMAGE_DEFAULT = \"docker.io/restatedev/restate\";\nconst RESTATE_DOCKER_DEFAULT_TAG = \"latest\";\nconst ADOT_DOCKER_DEFAULT_TAG = \"latest\";\nconst DATA_DEVICE_NAME = \"/dev/sdd\";\n\n/**\n * Creates a Restate service deployment backed by a single EC2 instance, suitable for development and testing purposes.\n *\n * **Durability**\n *\n * Restate data will be stored in a separate EBS volume which you can configure explicitly via the `dataVolumeOptions`\n * property. Updating configuration settings may trigger instance reboot or replacement - consider snapshotting the data\n * volume prior to deployments, and enabling instance termination protection.\n *\n * **Security**\n *\n * The EC2 instance will be created in the default VPC unless otherwise specified. Two security groups are created,\n * `ingressSecurityGroup` and `adminSecurityGroup`, which control access to the Restate service ingress and admin ports\n * respectively. You must add appropriate rules or add other resources to these security groups to allow access.\n *\n * See {@link SingleNodeRestateProps} for available configuration options, and {@link ServiceDeployer} for deploying\n * Lambda handlers to environments.\n */\nexport class SingleNodeRestateDeployment extends Construct implements IRestateEnvironment {\n  readonly instance: ec2.Instance;\n  readonly instanceRole: iam.IRole;\n  readonly invokerRole: iam.IRole;\n  readonly vpc: ec2.IVpc;\n  readonly ingressSecurityGroup: ec2.ISecurityGroup;\n  readonly ingressPort: number;\n  readonly ingressUrl: string;\n  readonly adminSecurityGroup: ec2.ISecurityGroup;\n  readonly adminPort: number;\n  readonly adminUrl: string;\n  readonly tlsEnabled: boolean;\n\n  constructor(scope: Construct, id: string, props: SingleNodeRestateProps) {\n    super(scope, id);\n\n    this.vpc = props.vpc ?? ec2.Vpc.fromLookup(this, \"Vpc\", { isDefault: true });\n\n    const subnetType = props.networkConfiguration?.subnetType ?? ec2.SubnetType.PRIVATE_WITH_EGRESS;\n\n    this.instanceRole = new iam.Role(this, \"InstanceRole\", {\n      assumedBy: new iam.ServicePrincipal(\"ec2.amazonaws.com\"),\n      managedPolicies: [iam.ManagedPolicy.fromAwsManagedPolicyName(\"AmazonSSMManagedInstanceCore\")],\n    });\n\n    this.invokerRole = new iam.Role(this, \"InvokerRole\", {\n      assumedBy: this.instanceRole,\n    });\n\n    new iam.Policy(this, \"AssumeInvokerRolePolicy\", {\n      statements: [\n        new iam.PolicyStatement({\n          sid: \"AllowAssumeInvokerRole\",\n          actions: [\"sts:AssumeRole\"],\n          resources: [this.invokerRole.roleArn],\n        }),\n      ],\n    }).attachToRole(this.instanceRole);\n\n    const logGroup =\n      props.logGroup ??\n      new logs.LogGroup(this, \"Logs\", {\n        logGroupName: `/restate/${id}`,\n        retention: logs.RetentionDays.ONE_MONTH,\n        removalPolicy: props.removalPolicy ?? RemovalPolicy.DESTROY,\n      });\n    logGroup.grantWrite(this.instanceRole);\n\n    const restateImage = props.restateImage ?? RESTATE_IMAGE_DEFAULT;\n    const restateTag = props.restateTag ?? RESTATE_DOCKER_DEFAULT_TAG;\n    const adotTag = props.adotTag ?? ADOT_DOCKER_DEFAULT_TAG;\n\n    this.tlsEnabled = props.tlsTermination === TlsTermination.ON_HOST_SELF_SIGNED_CERTIFICATE;\n\n    const envDefaults = {\n      RESTATE_OBSERVABILITY__LOG__FORMAT: \"Json\",\n      RUST_LOG: \"info\",\n      RESTATE_OBSERVABILITY__TRACING__ENDPOINT: \"http://localhost:4317\",\n    };\n    const envArgs = Object.entries({ ...envDefaults, ...(props.environment ?? {}) })\n      .map(([key, value]) => `-e ${key}=\"${value}\"`)\n      .join(\" \");\n\n    const initScript = ec2.UserData.forLinux();\n    initScript.addCommands(\n      \"set -euf -o pipefail\",\n      `yum install -y npm && npm install -gq @restatedev/restate@${restateTag}`,\n      \"yum install -y docker\",\n      this.mountDataVolumeScript(),\n\n      \"mkdir -p /etc/restate\",\n      [\"cat << EOF > /etc/restate/config.toml\", this.restateConfig(id, props), \"EOF\"].join(\"\\n\"),\n\n      \"systemctl start docker.service\",\n\n      // Start the ADOT collector - needed for X-ray trace forwarding\n      `if [ \"$(docker ps -qa -f name=adot)\" ]; then docker stop adot || true; docker rm adot; fi`,\n      \"docker run --name adot --restart on-failure --detach\" +\n        \" -p 4317:4317 -p 55680:55680 -p 8889:8888\" +\n        ` public.ecr.aws/aws-observability/aws-otel-collector:${adotTag}`,\n\n      // Start the Restate server container\n      `if [ \"$(docker ps -qa -f name=restate)\" ]; then docker stop restate || true; docker rm restate; fi`,\n      \"docker run --name restate --restart on-failure --detach\" +\n        \" --volume /etc/restate:/etc/restate\" +\n        \" --volume /var/restate:/restate-data\" +\n        \" --network=host\" +\n        ` ${envArgs}` +\n        ` --log-driver=awslogs --log-opt awslogs-group=${logGroup.logGroupName}` +\n        ` ${restateImage}:${restateTag}` +\n        \" --config-file /etc/restate/config.toml\",\n    );\n\n    // Optionally, configure and start the nginx service\n    if (this.tlsEnabled) {\n      if (subnetType == ec2.SubnetType.PUBLIC) {\n        initScript.addCommands(\n          \"yum install -y nginx\",\n          \"mkdir -p /etc/pki/private\",\n          [\n            \"openssl req -new -x509 -nodes -sha256 -days 365 -extensions v3_ca\",\n            \" -subj '/C=DE/ST=Berlin/L=Berlin/O=restate.dev/OU=demo/CN=restate.example.com'\",\n            \" -newkey rsa:2048 -keyout /etc/pki/private/restate-selfsigned.key -out /etc/pki/private/restate-selfsigned.crt\",\n          ].join(\"\"),\n\n          [\"cat << EOF > /etc/nginx/conf.d/restate-ingress.conf\", this.ingressNginxConfig(props), \"EOF\"].join(\"\\n\"),\n          \"systemctl start nginx\",\n        );\n      }\n    }\n\n    const cloudConfig = ec2.UserData.custom([`cloud_final_modules:`, `- [scripts-user, always]`].join(\"\\n\"));\n\n    const userData = new ec2.MultipartUserData();\n    userData.addUserDataPart(cloudConfig, \"text/cloud-config\");\n    userData.addUserDataPart(initScript, \"text/x-shellscript\");\n\n    const restateInstance = new ec2.Instance(this, \"Host\", {\n      vpc: this.vpc,\n      vpcSubnets: { subnetType },\n      instanceType: props.instanceType ?? new ec2.InstanceType(\"t4g.micro\"),\n      machineImage:\n        props.machineImage ??\n        ec2.MachineImage.latestAmazonLinux2023({\n          cpuType: ec2.AmazonLinuxCpuType.ARM_64,\n        }),\n      role: this.instanceRole,\n      blockDevices: [\n        {\n          deviceName: DATA_DEVICE_NAME,\n          volume: {\n            ebsDevice: {\n              volumeSize: 8,\n              deleteOnTermination: props.removalPolicy === RemovalPolicy.DESTROY,\n              ...(props.dataVolumeOptions ?? {}),\n            },\n            virtualName: \"restate-data\",\n          },\n        },\n      ],\n      userData,\n    });\n\n    this.instance = restateInstance;\n\n    // We start the ADOT collector regardless, and control whether traces will be exported to X-Ray using instance role\n    // 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 ingressSecurityGroup = new ec2.SecurityGroup(this, \"IngressSecurityGroup\", {\n      vpc: this.vpc,\n      description: \"Restate Ingress ACLs\",\n    });\n    restateInstance.addSecurityGroup(ingressSecurityGroup);\n    const adminSecurityGroup = new ec2.SecurityGroup(this, \"AdminSecurityGroup\", {\n      vpc: this.vpc,\n      description: \"Restate Admin ACLs\",\n    });\n    restateInstance.addSecurityGroup(adminSecurityGroup);\n\n    this.ingressPort = this.tlsEnabled ? RESTATE_TLS_INGRESS_PORT : RESTATE_INGRESS_PORT;\n    this.adminPort = this.tlsEnabled ? RESTATE_TLS_ADMIN_PORT : RESTATE_ADMIN_PORT;\n\n    ingressSecurityGroup.addIngressRule(\n      (props.publicIngress ?? false) ? ec2.Peer.anyIpv4() : ingressSecurityGroup,\n      ec2.Port.tcp(this.ingressPort),\n      \"Restate ingress\",\n    );\n    adminSecurityGroup.addIngressRule(adminSecurityGroup, ec2.Port.tcp(this.adminPort), \"Restate admin\");\n\n    const protocol = this.tlsEnabled ? \"https\" : \"http\";\n    const hostname =\n      props.networkConfiguration?.subnetType !== ec2.SubnetType.PUBLIC\n        ? `${restateInstance.instancePrivateDnsName}`\n        : `${restateInstance.instancePublicDnsName}`;\n    this.ingressUrl = `${protocol}://${hostname}` + (this.ingressPort === 443 ? \"\" : `:${this.ingressPort}`);\n    this.adminUrl = `${protocol}://${hostname}:${this.adminPort}`;\n\n    this.ingressSecurityGroup = ingressSecurityGroup;\n    this.adminSecurityGroup = adminSecurityGroup;\n  }\n\n  protected restateConfig(id: string, props: SingleNodeRestateProps) {\n    return (\n      props.restateConfigOverride ??\n      [\n        `roles = [`,\n        `    \"worker\",`,\n        `    \"admin\",`,\n        `    \"metadata-store\",`,\n        `]`,\n        `node-name = \"restate-0\"`,\n        `cluster-name = \"${props.restateConfig?.clusterName ?? id}\"`,\n        `allow-bootstrap = true`,\n        `bootstrap-num-partitions = ${props.restateConfig?.bootstrapNumPartitions ?? 4}`,\n        `default-thread-pool-size = 3`,\n        `storage-high-priority-bg-threads = 3`,\n        `storage-low-priority-bg-threads = 3`,\n        `rocksdb-total-memory-size = \"${props.restateConfig?.rocksdb?.totalMemorySize?.toMebibytes() ?? 512.0 + \" MB\"}\"`,\n        `rocksdb-total-memtables-ratio = 0.60`,\n        `rocksdb-bg-threads = 3`,\n        `rocksdb-high-priority-bg-threads = 3`,\n        ``,\n        `[worker]`,\n        `internal-queue-length = 1000`,\n        ``,\n        `[worker.storage]`,\n        `rocksdb-max-background-jobs = 3`,\n        `rocksdb-statistics-level = \"except-detailed-timers\"`,\n        `num-partitions-to-share-memory-budget = 4`,\n        ``,\n        `[admin]`,\n        `bind-address = \"${this.tlsEnabled ? \"127.0.0.1\" : \"0.0.0.0\"}:${RESTATE_ADMIN_PORT}\"`,\n        ``,\n        `[admin.query-engine]`,\n        `memory-size = \"50.0 MB\"`,\n        `query-parallelism = 4`,\n        ``,\n        `[ingress]`,\n        `bind-address = \"${this.tlsEnabled ? \"127.0.0.1\" : \"0.0.0.0\"}:${RESTATE_INGRESS_PORT}\"`,\n        `rocksdb-max-background-jobs = 3`,\n        `rocksdb-statistics-level = \"except-detailed-timers\"`,\n        `writer-batch-commit-count = 1000`,\n        ``,\n        `[metadata-store.rocksdb]`,\n        `rocksdb-max-background-jobs = 1`,\n        `rocksdb-statistics-level = \"except-detailed-timers\"`,\n      ].join(\"\\n\")\n    );\n  }\n\n  protected mountDataVolumeScript() {\n    return `\nif mount | grep -qs '/var/restate'; then\n  echo \"/var/restate is mounted\"\nelse\n  if [ -d /var/restate ]; then\n    if [ \"$(ls -A /var/restate)\" ]; then\n      echo \"Data exists in /var/restate that is not on data volume; refusing to overwrite!\"\n      exit 1\n    fi\n  else\n    mkdir /var/restate\n  fi\n  if file -sL ${DATA_DEVICE_NAME} | grep -q ': data$'; then\n    mkfs -t xfs ${DATA_DEVICE_NAME}\n  fi\n  mount ${DATA_DEVICE_NAME} /var/restate\n  if ! grep -qs '/var/restate' /etc/fstab; then\n    echo \"${DATA_DEVICE_NAME} /var/restate xfs defaults 0 0\" >> /etc/fstab\n    echo \"Added entry for ${DATA_DEVICE_NAME} to /etc/fstab\"\n  else\n    echo \"Entry for ${DATA_DEVICE_NAME} already exists in /etc/fstab\"\n  fi\nfi\n`;\n  }\n\n  /**\n   * @param props construct properties\n   * @returns nginx configuration to use for ingress reverse proxy, formatted as a multi-line string\n   */\n  protected ingressNginxConfig(props: SingleNodeRestateProps) {\n    return (\n      props.ingressNginxConfigOverride ??\n      [\n        \"server {\",\n        `  listen ${RESTATE_TLS_INGRESS_PORT} ssl http2;`,\n        `  listen [::]:${RESTATE_TLS_INGRESS_PORT} ssl http2;`,\n        \"  server_name _;\",\n        \"  root /usr/share/nginx/html;\",\n        \"\",\n        '  ssl_certificate \"/etc/pki/private/restate-selfsigned.crt\";',\n        '  ssl_certificate_key \"/etc/pki/private/restate-selfsigned.key\";',\n        \"  ssl_session_cache shared:SSL:1m;\",\n        \"  ssl_session_timeout 10m;\",\n        \"  ssl_ciphers PROFILE=SYSTEM;\",\n        \"  ssl_prefer_server_ciphers on;\",\n        \"\",\n        \"  location / {\",\n        `    proxy_pass http://localhost:${RESTATE_INGRESS_PORT};`,\n        `    proxy_read_timeout ${props.ingressProxyReadTimeout?.toSeconds() ?? 3600};`,\n        \"  }\",\n        \"}\",\n        \"\",\n        \"server {\",\n        `  listen ${RESTATE_TLS_ADMIN_PORT} ssl http2;`,\n        `  listen [::]:${RESTATE_TLS_ADMIN_PORT} ssl http2;`,\n        \"  server_name _;\",\n        \"  root /usr/share/nginx/html;\",\n        \"\",\n        '  ssl_certificate \"/etc/pki/private/restate-selfsigned.crt\";',\n        '  ssl_certificate_key \"/etc/pki/private/restate-selfsigned.key\";',\n        \"  ssl_session_cache shared:SSL:1m;\",\n        \"  ssl_session_timeout 10m;\",\n        \"  ssl_ciphers PROFILE=SYSTEM;\",\n        \"  ssl_prefer_server_ciphers on;\",\n        \"\",\n        \"  location / {\",\n        `    proxy_pass http://localhost:${RESTATE_ADMIN_PORT};`,\n        \"  }\",\n        \"}\",\n      ].join(\"\\n\")\n    );\n  }\n}\n"]}
310
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"single-node-restate-deployment.js","sourceRoot":"","sources":["../lib/restate-constructs/single-node-restate-deployment.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2023-2025 - Restate Software, Inc., Restate GmbH\n *\n * This file is part of the Restate CDK Construct Library,\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 { IRestateEnvironment } from \"./restate-environment\";\nimport { TracingMode } from \"./deployments-common\";\nimport * as cdk from \"aws-cdk-lib\";\nimport { RemovalPolicy } from \"aws-cdk-lib\";\n\nexport interface SingleNodeRestateProps {\n  /** EC2 instance type to use. */\n  instanceType?: ec2.InstanceType;\n\n  /** Machine image. Note: startup script expects yum-based package management. */\n  machineImage?: ec2.IMachineImage;\n\n  /** The VPC in which to launch the Restate host. */\n  vpc?: ec2.IVpc;\n\n  networkConfiguration?: {\n    /**\n     * Subnet type for the Restate host.\n     *\n     * Available options:\n     *  - [Default] {@link ec2.SubnetType.PRIVATE_WITH_EGRESS} will create the Restate instance with outbound internet\n     *    access only, so that it can invoke HTTP endpoints. The security groups {@link ingressSecurityGroup} and\n     *    {@link adminSecurityGroup} control inbound traffic to the service ingress and admin ports respectively.\n     *    Configure {@link ServiceDeployer} to use the latter, and set up ingress traffic routing outside of this\n     *    construct using the former.\n     *  - Insecure, internet-facing {@link ec2.SubnetType.PUBLIC} will also provision an nginx reverse proxy\n     *    and an HTTP listener with a self-signed certificate.\n     */\n    subnetType?: ec2.SubnetType.PRIVATE_WITH_EGRESS | ec2.SubnetType.PUBLIC;\n  };\n\n  /**\n   * Allow incoming ingress traffic from anywhere. Default: `false`. Alternatively, add rules to the\n   * `ingressSecurityGroup` directly.\n   */\n  publicIngress?: boolean;\n\n  /** Log group for Restate service logs. */\n  logGroup?: logs.LogGroup;\n\n  /** Tracing mode for Restate services. Defaults to {@link TracingMode.DISABLED}. */\n  tracing?: TracingMode;\n\n  /** Prefix for resources created by this construct that require unique names. */\n  prefix?: string;\n\n  /** Restate Docker image name. Defaults to `docker.restate.dev/restatedev/restate`. */\n  restateImage?: string;\n\n  /** Restate Docker image tag. Defaults to `latest`. */\n  restateTag?: string;\n\n  /**\n   * EBS data volume settings for Restate data storage. If not specified, a default 8GB volume will be created.\n   */\n  dataVolumeOptions?: ec2.EbsDeviceProps | undefined;\n\n  /** Restate high-level configuration options. Alternatively, you can set {@link restateConfigOverride}. */\n  restateConfig?: {\n    /** Defaults to the construct id if left unspecified. */\n    clusterName?: string;\n    /** Defaults to 4. Only takes effect on initial provisioning. */\n    bootstrapNumPartitions?: number;\n    /** RocksDB settings. */\n    rocksdb?: {\n      /** Defaults to 512 MB. */\n      totalMemorySize?: cdk.Size;\n    };\n  };\n\n  /**\n   * Environment properties to set for Restate. This is a simple way to pass custom configuration parameters.\n   */\n  environment?: {\n    [key: string]: string;\n  };\n\n  /**\n   * Completely override the Restate server configuration. Note that other Restate configuration options\n   * will effectively be ignored if this is set. See https://docs.restate.dev/operate/configuration/server/\n   * for details.\n   */\n  restateConfigOverride?: string;\n\n  /** Amazon Distro for Open Telemetry Docker image tag. Defaults to `latest`. */\n  adotTag?: string;\n\n  /**\n   * Removal policy for long-lived resources (storage, logs). Default: `cdk.RemovalPolicy.DESTROY`.\n   */\n  removalPolicy?: cdk.RemovalPolicy;\n\n  /**\n   * Control on-host TLS termination for ingress and admin ports. Defaults to `TlsTermination.NONE`. Currently, enabling\n   * TLS implicitly enables an `nginx` service to be configured on the host. Depending on the value of this option, the\n   * ingress security group will be configured to allow inbound traffic to the appropriate port - 8080 for no TLS, or\n   * 443 with TLS enabled.\n   */\n  tlsTermination?: TlsTermination;\n\n  /**\n   * The read timeout for proxied ingress requests. Default: 3600 seconds.\n   */\n  ingressProxyReadTimeout?: cdk.Duration;\n\n  /**\n   * Completely override the default `nginx` configuration for the ingress proxy. Note that other\n   * ingress proxy configuration options will effectively be ignored if this is set.\n   */\n  ingressNginxConfigOverride?: string;\n}\n\nexport enum TlsTermination {\n  /**\n   * Disabled (default); expose the `restate-server` HTTP ports directly.\n   */\n  DISABLED,\n\n  /**\n   * Use self-signed certificates for TLS termination on the ingress and admin ports. Convenient for quick testing,\n   * make sure you set `insecure` = `true` when using `ServiceDeployer` to accept this certificate.\n   */\n  ON_HOST_SELF_SIGNED_CERTIFICATE,\n}\n\nconst RESTATE_INGRESS_PORT = 8080;\nconst RESTATE_TLS_INGRESS_PORT = 443;\nconst RESTATE_ADMIN_PORT = 9070;\nconst RESTATE_TLS_ADMIN_PORT = 9073;\nconst RESTATE_IMAGE_DEFAULT = \"docker.io/restatedev/restate\";\nconst RESTATE_DOCKER_DEFAULT_TAG = \"latest\";\nconst ADOT_DOCKER_DEFAULT_TAG = \"latest\";\nconst DATA_DEVICE_NAME = \"/dev/sdd\";\n\n/**\n * Creates a Restate service deployment backed by a single EC2 instance, suitable for development and testing purposes.\n *\n * **Durability**\n *\n * Restate data will be stored in a separate EBS volume which you can configure explicitly via the `dataVolumeOptions`\n * property. Updating configuration settings may trigger instance reboot or replacement - consider snapshotting the data\n * volume prior to deployments, and enabling instance termination protection.\n *\n * **Security**\n *\n * The EC2 instance will be created in the default VPC unless otherwise specified. Two security groups are created,\n * `ingressSecurityGroup` and `adminSecurityGroup`, which control access to the Restate service ingress and admin ports\n * respectively. You must add appropriate rules or add other resources to these security groups to allow access.\n *\n * See {@link SingleNodeRestateProps} for available configuration options, and {@link ServiceDeployer} for deploying\n * Lambda handlers to environments.\n */\nexport class SingleNodeRestateDeployment extends Construct implements IRestateEnvironment {\n  readonly instance: ec2.Instance;\n  readonly instanceRole: iam.IRole;\n  readonly invokerRole: iam.IRole;\n  readonly vpc: ec2.IVpc;\n  readonly ingressSecurityGroup: ec2.ISecurityGroup;\n  readonly ingressPort: number;\n  readonly ingressUrl: string;\n  readonly adminSecurityGroup: ec2.ISecurityGroup;\n  readonly adminPort: number;\n  readonly adminUrl: string;\n  readonly tlsEnabled: boolean;\n\n  constructor(scope: Construct, id: string, props: SingleNodeRestateProps) {\n    super(scope, id);\n\n    this.vpc = props.vpc ?? ec2.Vpc.fromLookup(this, \"Vpc\", { isDefault: true });\n\n    const subnetType = props.networkConfiguration?.subnetType ?? ec2.SubnetType.PRIVATE_WITH_EGRESS;\n\n    this.instanceRole = new iam.Role(this, \"InstanceRole\", {\n      assumedBy: new iam.ServicePrincipal(\"ec2.amazonaws.com\"),\n      managedPolicies: [iam.ManagedPolicy.fromAwsManagedPolicyName(\"AmazonSSMManagedInstanceCore\")],\n    });\n\n    this.invokerRole = new iam.Role(this, \"InvokerRole\", {\n      assumedBy: this.instanceRole,\n    });\n\n    new iam.Policy(this, \"AssumeInvokerRolePolicy\", {\n      statements: [\n        new iam.PolicyStatement({\n          sid: \"AllowAssumeInvokerRole\",\n          actions: [\"sts:AssumeRole\"],\n          resources: [this.invokerRole.roleArn],\n        }),\n      ],\n    }).attachToRole(this.instanceRole);\n\n    const logGroup =\n      props.logGroup ??\n      new logs.LogGroup(this, \"Logs\", {\n        logGroupName: `/restate/${id}`,\n        retention: logs.RetentionDays.ONE_MONTH,\n        removalPolicy: props.removalPolicy ?? RemovalPolicy.DESTROY,\n      });\n    logGroup.grantWrite(this.instanceRole);\n\n    const restateImage = props.restateImage ?? RESTATE_IMAGE_DEFAULT;\n    const restateTag = props.restateTag ?? RESTATE_DOCKER_DEFAULT_TAG;\n    const adotTag = props.adotTag ?? ADOT_DOCKER_DEFAULT_TAG;\n\n    this.tlsEnabled = props.tlsTermination === TlsTermination.ON_HOST_SELF_SIGNED_CERTIFICATE;\n\n    const envDefaults = {\n      RESTATE_LOG_FORMAT: \"json\",\n      RUST_LOG: \"info\",\n      RESTATE_TRACING_ENDPOINT: \"http://localhost:4317\",\n    };\n    const envArgs = Object.entries({ ...envDefaults, ...(props.environment ?? {}) })\n      .map(([key, value]) => `-e ${key}=\"${value}\"`)\n      .join(\" \");\n\n    const initScript = ec2.UserData.forLinux();\n    initScript.addCommands(\n      \"set -euf -o pipefail\",\n      `yum install -y npm && npm install -gq @restatedev/restate@${restateTag} @restatedev/restatectl@${restateTag}`,\n      \"yum install -y docker\",\n      this.mountDataVolumeScript(),\n\n      \"mkdir -p /etc/restate\",\n      [\"cat << EOF > /etc/restate/config.toml\", this.restateConfig(id, props), \"EOF\"].join(\"\\n\"),\n\n      \"systemctl start docker.service\",\n\n      // Start the ADOT collector - needed for X-ray trace forwarding\n      `if [ \"$(docker ps -qa -f name=adot)\" ]; then docker stop adot || true; docker rm adot; fi`,\n      \"docker run --name adot --restart on-failure --detach --pull always\" +\n        \" -p 4317:4317 -p 55680:55680 -p 8889:8888\" +\n        ` public.ecr.aws/aws-observability/aws-otel-collector:${adotTag}`,\n\n      // Start the Restate server container\n      `if [ \"$(docker ps -qa -f name=restate)\" ]; then docker stop restate || true; docker rm restate; fi`,\n      \"docker run --name restate --restart on-failure --detach --pull always\" +\n        \" --volume /etc/restate:/etc/restate\" +\n        \" --volume /var/restate:/restate-data\" +\n        \" --network=host\" +\n        ` ${envArgs}` +\n        ` --log-driver=awslogs --log-opt awslogs-group=${logGroup.logGroupName}` +\n        ` ${restateImage}:${restateTag}` +\n        \" --config-file /etc/restate/config.toml\",\n    );\n\n    // Optionally, configure and start the nginx service\n    if (this.tlsEnabled) {\n      if (subnetType == ec2.SubnetType.PUBLIC) {\n        initScript.addCommands(\n          \"yum install -y nginx\",\n          \"mkdir -p /etc/pki/private\",\n          [\n            \"openssl req -new -x509 -nodes -sha256 -days 365 -extensions v3_ca\",\n            \" -subj '/C=DE/ST=Berlin/L=Berlin/O=restate.dev/OU=demo/CN=restate.example.com'\",\n            \" -newkey rsa:2048 -keyout /etc/pki/private/restate-selfsigned.key -out /etc/pki/private/restate-selfsigned.crt\",\n          ].join(\"\"),\n\n          [\"cat << EOF > /etc/nginx/conf.d/restate-ingress.conf\", this.ingressNginxConfig(props), \"EOF\"].join(\"\\n\"),\n          \"systemctl start nginx\",\n        );\n      }\n    }\n\n    const cloudConfig = ec2.UserData.custom([`cloud_final_modules:`, `- [scripts-user, always]`].join(\"\\n\"));\n\n    const userData = new ec2.MultipartUserData();\n    userData.addUserDataPart(cloudConfig, \"text/cloud-config\");\n    userData.addUserDataPart(initScript, \"text/x-shellscript\");\n\n    const restateInstance = new ec2.Instance(this, \"Host\", {\n      vpc: this.vpc,\n      vpcSubnets: { subnetType },\n      instanceType: props.instanceType ?? new ec2.InstanceType(\"t4g.micro\"),\n      machineImage:\n        props.machineImage ??\n        ec2.MachineImage.latestAmazonLinux2023({\n          cpuType: ec2.AmazonLinuxCpuType.ARM_64,\n        }),\n      role: this.instanceRole,\n      blockDevices: [\n        {\n          deviceName: DATA_DEVICE_NAME,\n          volume: {\n            ebsDevice: {\n              volumeSize: 8,\n              deleteOnTermination: props.removalPolicy === RemovalPolicy.DESTROY,\n              ...(props.dataVolumeOptions ?? {}),\n            },\n            virtualName: \"restate-data\",\n          },\n        },\n      ],\n      userData,\n    });\n\n    this.instance = restateInstance;\n\n    // We start the ADOT collector regardless, and control whether traces will be exported to X-Ray using instance role\n    // 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 ingressSecurityGroup = new ec2.SecurityGroup(this, \"IngressSecurityGroup\", {\n      vpc: this.vpc,\n      description: \"Restate Ingress ACLs\",\n    });\n    restateInstance.addSecurityGroup(ingressSecurityGroup);\n    const adminSecurityGroup = new ec2.SecurityGroup(this, \"AdminSecurityGroup\", {\n      vpc: this.vpc,\n      description: \"Restate Admin ACLs\",\n    });\n    restateInstance.addSecurityGroup(adminSecurityGroup);\n\n    this.ingressPort = this.tlsEnabled ? RESTATE_TLS_INGRESS_PORT : RESTATE_INGRESS_PORT;\n    this.adminPort = this.tlsEnabled ? RESTATE_TLS_ADMIN_PORT : RESTATE_ADMIN_PORT;\n\n    ingressSecurityGroup.addIngressRule(\n      (props.publicIngress ?? false) ? ec2.Peer.anyIpv4() : ingressSecurityGroup,\n      ec2.Port.tcp(this.ingressPort),\n      \"Restate ingress\",\n    );\n    adminSecurityGroup.addIngressRule(adminSecurityGroup, ec2.Port.tcp(this.adminPort), \"Restate admin\");\n\n    const protocol = this.tlsEnabled ? \"https\" : \"http\";\n    const hostname =\n      props.networkConfiguration?.subnetType !== ec2.SubnetType.PUBLIC\n        ? `${restateInstance.instancePrivateDnsName}`\n        : `${restateInstance.instancePublicDnsName}`;\n    this.ingressUrl = `${protocol}://${hostname}` + (this.ingressPort === 443 ? \"\" : `:${this.ingressPort}`);\n    this.adminUrl = `${protocol}://${hostname}:${this.adminPort}`;\n\n    this.ingressSecurityGroup = ingressSecurityGroup;\n    this.adminSecurityGroup = adminSecurityGroup;\n  }\n\n  protected restateConfig(id: string, props: SingleNodeRestateProps) {\n    return (\n      props.restateConfigOverride ??\n      [\n        `roles = [`,\n        `    \"worker\",`,\n        `    \"admin\",`,\n        `    \"metadata-server\",`,\n        `    \"log-server\",`,\n        `]`,\n        `node-name = \"restate-0\"`,\n        `cluster-name = \"${props.restateConfig?.clusterName ?? id}\"`,\n        `default-num-partitions = ${props.restateConfig?.bootstrapNumPartitions ?? 4}`,\n        `rocksdb-total-memory-size = \"${props.restateConfig?.rocksdb?.totalMemorySize?.toMebibytes() ?? 512.0 + \" MB\"}\"`,\n        ``,\n        `[admin]`,\n        `bind-address = \"${this.tlsEnabled ? \"127.0.0.1\" : \"0.0.0.0\"}:${RESTATE_ADMIN_PORT}\"`,\n        ``,\n        `[admin.query-engine]`,\n        `memory-size = \"256.0 MiB\"`,\n        ``,\n        `[ingress]`,\n        `bind-address = \"${this.tlsEnabled ? \"127.0.0.1\" : \"0.0.0.0\"}:${RESTATE_INGRESS_PORT}\"`,\n      ].join(\"\\n\")\n    );\n  }\n\n  protected mountDataVolumeScript() {\n    return `\nif mount | grep -qs '/var/restate'; then\n  echo \"/var/restate is mounted\"\nelse\n  if [ -d /var/restate ]; then\n    if [ \"$(ls -A /var/restate)\" ]; then\n      echo \"Data exists in /var/restate that is not on data volume; refusing to overwrite!\"\n      exit 1\n    fi\n  else\n    mkdir /var/restate\n  fi\n  if file -sL ${DATA_DEVICE_NAME} | grep -q ': data$'; then\n    mkfs -t xfs ${DATA_DEVICE_NAME}\n  fi\n  mount ${DATA_DEVICE_NAME} /var/restate\n  if ! grep -qs '/var/restate' /etc/fstab; then\n    echo \"${DATA_DEVICE_NAME} /var/restate xfs defaults 0 0\" >> /etc/fstab\n    echo \"Added entry for ${DATA_DEVICE_NAME} to /etc/fstab\"\n  else\n    echo \"Entry for ${DATA_DEVICE_NAME} already exists in /etc/fstab\"\n  fi\nfi\n`;\n  }\n\n  /**\n   * @param props construct properties\n   * @returns nginx configuration to use for ingress reverse proxy, formatted as a multi-line string\n   */\n  protected ingressNginxConfig(props: SingleNodeRestateProps) {\n    return (\n      props.ingressNginxConfigOverride ??\n      [\n        \"server {\",\n        `  listen ${RESTATE_TLS_INGRESS_PORT} ssl http2;`,\n        `  listen [::]:${RESTATE_TLS_INGRESS_PORT} ssl http2;`,\n        \"  server_name _;\",\n        \"  root /usr/share/nginx/html;\",\n        \"\",\n        '  ssl_certificate \"/etc/pki/private/restate-selfsigned.crt\";',\n        '  ssl_certificate_key \"/etc/pki/private/restate-selfsigned.key\";',\n        \"  ssl_session_cache shared:SSL:1m;\",\n        \"  ssl_session_timeout 10m;\",\n        \"  ssl_ciphers PROFILE=SYSTEM;\",\n        \"  ssl_prefer_server_ciphers on;\",\n        \"\",\n        \"  location / {\",\n        `    proxy_pass http://localhost:${RESTATE_INGRESS_PORT};`,\n        `    proxy_read_timeout ${props.ingressProxyReadTimeout?.toSeconds() ?? 3600};`,\n        \"  }\",\n        \"}\",\n        \"\",\n        \"server {\",\n        `  listen ${RESTATE_TLS_ADMIN_PORT} ssl http2;`,\n        `  listen [::]:${RESTATE_TLS_ADMIN_PORT} ssl http2;`,\n        \"  server_name _;\",\n        \"  root /usr/share/nginx/html;\",\n        \"\",\n        '  ssl_certificate \"/etc/pki/private/restate-selfsigned.crt\";',\n        '  ssl_certificate_key \"/etc/pki/private/restate-selfsigned.key\";',\n        \"  ssl_session_cache shared:SSL:1m;\",\n        \"  ssl_session_timeout 10m;\",\n        \"  ssl_ciphers PROFILE=SYSTEM;\",\n        \"  ssl_prefer_server_ciphers on;\",\n        \"\",\n        \"  location / {\",\n        `    proxy_pass http://localhost:${RESTATE_ADMIN_PORT};`,\n        \"  }\",\n        \"}\",\n      ].join(\"\\n\")\n    );\n  }\n}\n"]}
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@restatedev/restate-cdk",
3
3
  "description": "Restate.dev CDK constructs",
4
- "version": "1.1.1",
4
+ "version": "1.2.0",
5
5
  "author": "Restate Developers",
6
6
  "license": "MIT",
7
7
  "email": "code@restate.dev",
@@ -657,7 +657,7 @@ exports[`Restate constructs Create a self-hosted Restate environment deployed on
657
657
  set -euf -o pipefail
658
658
 
659
659
  yum install -y npm && npm install -gq
660
- @restatedev/restate@latest
660
+ @restatedev/restate@latest @restatedev/restatectl@latest
661
661
 
662
662
  yum install -y docker
663
663
 
@@ -693,45 +693,18 @@ exports[`Restate constructs Create a self-hosted Restate environment deployed on
693
693
  roles = [
694
694
  "worker",
695
695
  "admin",
696
- "metadata-store",
696
+ "metadata-server",
697
+ "log-server",
697
698
  ]
698
699
 
699
700
  node-name = "restate-0"
700
701
 
701
702
  cluster-name = "Restate"
702
703
 
703
- allow-bootstrap = true
704
-
705
- bootstrap-num-partitions = 4
706
-
707
- default-thread-pool-size = 3
708
-
709
- storage-high-priority-bg-threads = 3
710
-
711
- storage-low-priority-bg-threads = 3
704
+ default-num-partitions = 4
712
705
 
713
706
  rocksdb-total-memory-size = "512 MB"
714
707
 
715
- rocksdb-total-memtables-ratio = 0.60
716
-
717
- rocksdb-bg-threads = 3
718
-
719
- rocksdb-high-priority-bg-threads = 3
720
-
721
-
722
- [worker]
723
-
724
- internal-queue-length = 1000
725
-
726
-
727
- [worker.storage]
728
-
729
- rocksdb-max-background-jobs = 3
730
-
731
- rocksdb-statistics-level = "except-detailed-timers"
732
-
733
- num-partitions-to-share-memory-budget = 4
734
-
735
708
 
736
709
  [admin]
737
710
 
@@ -740,28 +713,13 @@ exports[`Restate constructs Create a self-hosted Restate environment deployed on
740
713
 
741
714
  [admin.query-engine]
742
715
 
743
- memory-size = "50.0 MB"
744
-
745
- query-parallelism = 4
716
+ memory-size = "256.0 MiB"
746
717
 
747
718
 
748
719
  [ingress]
749
720
 
750
721
  bind-address = "127.0.0.1:8080"
751
722
 
752
- rocksdb-max-background-jobs = 3
753
-
754
- rocksdb-statistics-level = "except-detailed-timers"
755
-
756
- writer-batch-commit-count = 1000
757
-
758
-
759
- [metadata-store.rocksdb]
760
-
761
- rocksdb-max-background-jobs = 1
762
-
763
- rocksdb-statistics-level = "except-detailed-timers"
764
-
765
723
  EOF
766
724
 
767
725
  systemctl start docker.service
@@ -769,19 +727,18 @@ exports[`Restate constructs Create a self-hosted Restate environment deployed on
769
727
  if [ "$(docker ps -qa -f name=adot)" ]; then docker stop
770
728
  adot || true; docker rm adot; fi
771
729
 
772
- docker run --name adot --restart on-failure --detach -p
773
- 4317:4317 -p 55680:55680 -p 8889:8888
730
+ docker run --name adot --restart on-failure --detach
731
+ --pull always -p 4317:4317 -p 55680:55680 -p 8889:8888
774
732
  public.ecr.aws/aws-observability/aws-otel-collector:latest
775
733
 
776
734
  if [ "$(docker ps -qa -f name=restate)" ]; then docker
777
735
  stop restate || true; docker rm restate; fi
778
736
 
779
737
  docker run --name restate --restart on-failure --detach
780
- --volume /etc/restate:/etc/restate --volume
781
- /var/restate:/restate-data --network=host -e
782
- RESTATE_OBSERVABILITY__LOG__FORMAT="Json" -e
783
- RUST_LOG="info" -e
784
- RESTATE_OBSERVABILITY__TRACING__ENDPOINT="http://localhost:4317"
738
+ --pull always --volume /etc/restate:/etc/restate
739
+ --volume /var/restate:/restate-data --network=host -e
740
+ RESTATE_LOG_FORMAT="json" -e RUST_LOG="info" -e
741
+ RESTATE_TRACING_ENDPOINT="http://localhost:4317"
785
742
  --log-driver=awslogs --log-opt awslogs-group=
786
743
  - Ref: RestateLogsFD86ECAE
787
744
  - ' docker.io/restatedev/restate:latest --config-file /etc/restate/config.toml'
@@ -997,6 +954,7 @@ exports[`Restate constructs Create a self-hosted Restate environment deployed on
997
954
 
998
955
  yum install -y npm && npm install -gq
999
956
  @restatedev/restate@custom-version
957
+ @restatedev/restatectl@custom-version
1000
958
 
1001
959
  yum install -y docker
1002
960
 
@@ -1032,45 +990,18 @@ exports[`Restate constructs Create a self-hosted Restate environment deployed on
1032
990
  roles = [
1033
991
  "worker",
1034
992
  "admin",
1035
- "metadata-store",
993
+ "metadata-server",
994
+ "log-server",
1036
995
  ]
1037
996
 
1038
997
  node-name = "restate-0"
1039
998
 
1040
999
  cluster-name = "Restate"
1041
1000
 
1042
- allow-bootstrap = true
1043
-
1044
- bootstrap-num-partitions = 4
1045
-
1046
- default-thread-pool-size = 3
1047
-
1048
- storage-high-priority-bg-threads = 3
1049
-
1050
- storage-low-priority-bg-threads = 3
1001
+ default-num-partitions = 4
1051
1002
 
1052
1003
  rocksdb-total-memory-size = "512 MB"
1053
1004
 
1054
- rocksdb-total-memtables-ratio = 0.60
1055
-
1056
- rocksdb-bg-threads = 3
1057
-
1058
- rocksdb-high-priority-bg-threads = 3
1059
-
1060
-
1061
- [worker]
1062
-
1063
- internal-queue-length = 1000
1064
-
1065
-
1066
- [worker.storage]
1067
-
1068
- rocksdb-max-background-jobs = 3
1069
-
1070
- rocksdb-statistics-level = "except-detailed-timers"
1071
-
1072
- num-partitions-to-share-memory-budget = 4
1073
-
1074
1005
 
1075
1006
  [admin]
1076
1007
 
@@ -1079,28 +1010,13 @@ exports[`Restate constructs Create a self-hosted Restate environment deployed on
1079
1010
 
1080
1011
  [admin.query-engine]
1081
1012
 
1082
- memory-size = "50.0 MB"
1083
-
1084
- query-parallelism = 4
1013
+ memory-size = "256.0 MiB"
1085
1014
 
1086
1015
 
1087
1016
  [ingress]
1088
1017
 
1089
1018
  bind-address = "0.0.0.0:8080"
1090
1019
 
1091
- rocksdb-max-background-jobs = 3
1092
-
1093
- rocksdb-statistics-level = "except-detailed-timers"
1094
-
1095
- writer-batch-commit-count = 1000
1096
-
1097
-
1098
- [metadata-store.rocksdb]
1099
-
1100
- rocksdb-max-background-jobs = 1
1101
-
1102
- rocksdb-statistics-level = "except-detailed-timers"
1103
-
1104
1020
  EOF
1105
1021
 
1106
1022
  systemctl start docker.service
@@ -1108,20 +1024,18 @@ exports[`Restate constructs Create a self-hosted Restate environment deployed on
1108
1024
  if [ "$(docker ps -qa -f name=adot)" ]; then docker stop
1109
1025
  adot || true; docker rm adot; fi
1110
1026
 
1111
- docker run --name adot --restart on-failure --detach -p
1112
- 4317:4317 -p 55680:55680 -p 8889:8888
1027
+ docker run --name adot --restart on-failure --detach
1028
+ --pull always -p 4317:4317 -p 55680:55680 -p 8889:8888
1113
1029
  public.ecr.aws/aws-observability/aws-otel-collector:latest
1114
1030
 
1115
1031
  if [ "$(docker ps -qa -f name=restate)" ]; then docker
1116
1032
  stop restate || true; docker rm restate; fi
1117
1033
 
1118
1034
  docker run --name restate --restart on-failure --detach
1119
- --volume /etc/restate:/etc/restate --volume
1120
- /var/restate:/restate-data --network=host -e
1121
- RESTATE_OBSERVABILITY__LOG__FORMAT="Json" -e
1122
- RUST_LOG="info" -e
1123
- RESTATE_OBSERVABILITY__TRACING__ENDPOINT="http://localhost:4317"
1124
- -e
1035
+ --pull always --volume /etc/restate:/etc/restate
1036
+ --volume /var/restate:/restate-data --network=host -e
1037
+ RESTATE_LOG_FORMAT="json" -e RUST_LOG="info" -e
1038
+ RESTATE_TRACING_ENDPOINT="http://localhost:4317" -e
1125
1039
  RESTATE_INGRESS__ADVERTISED_INGRESS_ENDPOINT="http://restate-ingress"
1126
1040
  --log-driver=awslogs --log-opt awslogs-group=
1127
1041
  - Ref: RestateLogsFD86ECAE
File without changes