diff --git a/.github/workflows/integration-tests-samples.yml b/.github/workflows/integration-tests-samples.yml
new file mode 100644
index 000000000..31269881b
--- /dev/null
+++ b/.github/workflows/integration-tests-samples.yml
@@ -0,0 +1,62 @@
+name: Integration test for akka-sample-cluster-kubernetes-scala
+
+on:
+ pull_request:
+ push:
+ branches:
+ - main
+ - release-*
+ tags-ignore: [ v.* ]
+ schedule:
+ - cron: '0 2 * * *' # every day 2am
+
+permissions:
+ contents: read
+
+jobs:
+ integration-test:
+ name: Integration Tests for akka-sample-cluster-kubernetes-scala
+ runs-on: ubuntu-22.04
+ steps:
+ - name: Checkout
+ # https://github.com/actions/checkout/releases
+ # v4.1.1
+ uses: actions/checkout@b4ffde65f46336ab88eb53be808477a3936bae11
+ with:
+ fetch-depth: 0
+
+ - name: Checkout GitHub merge
+ if: github.event.pull_request
+ run: |-
+ git fetch origin pull/${{ github.event.pull_request.number }}/merge:scratch
+ git checkout scratch
+
+ - name: Cache Coursier cache
+ # https://github.com/coursier/cache-action/releases
+ # v6.4.5
+ uses: coursier/cache-action@1ff273bff02a8787bc9f1877d347948af647956d
+
+ - name: Set up JDK 17
+ # https://github.com/coursier/setup-action/releases
+ # v1.3.5
+ uses: coursier/setup-action@7bde40eee928896f074dbb76d22dd772eed5c65f
+ with:
+ jvm: temurin:1.17.0
+
+ - name: Setup Minikube
+ # https://github.com/manusa/actions-setup-minikube/releases
+ # v2.7.1
+ uses: manusa/actions-setup-minikube@4582844dcacbf482729f8d7ef696f515d2141bb9
+ with:
+ minikube version: 'v1.32.0'
+ kubernetes version: 'v1.28.4'
+ driver: docker
+ start args: '--addons=ingress'
+
+ - name: Run Integration Tests
+ run: |-
+ ./integration-test/akka-sample-cluster-kubernetes-scala/test.sh
+
+ - name: Print logs on failure
+ if: ${{ failure() }}
+ run: find . -name "*.log" -exec ./scripts/cat-log.sh {} \;
diff --git a/.github/workflows/release.yml b/.github/workflows/release.yml
index 1ce4cac5e..07bb83d13 100644
--- a/.github/workflows/release.yml
+++ b/.github/workflows/release.yml
@@ -76,6 +76,7 @@ jobs:
- name: Publish
run: |-
+ scripts/prepare-downloads.sh
eval "$(ssh-agent -s)"
echo $AKKA_RSYNC_GUSTAV | base64 -d > .github/id_rsa
chmod 600 .github/id_rsa
diff --git a/.gitignore b/.gitignore
index c35fcc96c..cc2ca53ee 100755
--- a/.gitignore
+++ b/.gitignore
@@ -14,3 +14,6 @@ target/
.metals
.vscode
+
+# attachments created by scripts/prepare-downloads.sh
+docs/src/main/paradox/attachments
diff --git a/docs/release-train-issue-template.md b/docs/release-train-issue-template.md
index 6d6030cc5..0e3104417 100644
--- a/docs/release-train-issue-template.md
+++ b/docs/release-train-issue-template.md
@@ -15,6 +15,7 @@ Variables to be expanded in this template:
- [ ] Check that open PRs and issues assigned to the milestone are reasonable
- [ ] Update the Change date and version in the LICENSE file.
+- [ ] Update the Akka Management version in the samples to $VERSION$, otherwise the published zip files of the samples will have the old version.
- [ ] Create a new milestone for the [next version](https://github.com/akka/akka-management/milestones)
- [ ] Close the [$VERSION$ milestone](https://github.com/akka/akka-management/milestones?direction=asc&sort=due_date)
- [ ] Make sure all important PRs have been merged
@@ -46,7 +47,6 @@ For important patch releases, and only if critical issues have been fixed:
- [ ] Send a release notification to [Lightbend discuss](https://discuss.akka.io)
- [ ] Tweet using the [@akkateam](https://twitter.com/akkateam/) account (or ask someone to) about the new release
-- [ ] Announce on [Gitter akka/akka](https://gitter.im/akka/akka)
- [ ] Announce internally (with links to Tweet, discuss)
For minor or major releases:
@@ -56,6 +56,6 @@ For minor or major releases:
### Afterwards
- [ ] Update [akka-dependencies bom](https://github.com/lightbend/akka-dependencies) and version for [Akka module versions](https://doc.akka.io/docs/akka-dependencies/current/) in [akka-dependencies repo](https://github.com/akka/akka-dependencies)
-- [ ] Update [Akka Guide samples](https://github.com/akka/akka-platform-guide)
+- [ ] Update [Akka Guide samples](https://github.com/lightbend/akka-guide)
- Close this issue
diff --git a/docs/src/main/paradox/bootstrap/recipes.md b/docs/src/main/paradox/bootstrap/recipes.md
index a38514096..40aff8b65 100644
--- a/docs/src/main/paradox/bootstrap/recipes.md
+++ b/docs/src/main/paradox/bootstrap/recipes.md
@@ -3,11 +3,6 @@
A set of integration tests projects can be found in [integration-test folder of the Akka Management project](https://github.com/akka/akka-management/tree/main/integration-test).
These test various Akka management features together in various environments such as Kubernetes.
-The following samples exist as standalone projects:
-
-* [Akka Cluster bootstrap using the Kubernetes API with Java/Maven](https://github.com/akka/akka-sample-cluster-kubernetes-java)
-* [Akka Cluster bootstrap using DNS in Kubernetes](https://github.com/akka/akka-sample-cluster-kubernetes-dns-java)
-
## Local
To run Bootstrap locally without any dependencies such as DNS or Kubernetes see the @ref[`local` example](local-config.md)
@@ -25,7 +20,10 @@ The recommended approach is to:
### Example project
-To get started, it might be helpful to have a look at the [Akka Cluster on Kubernetes](https://developer.lightbend.com/start/?group=akka&project=akka-sample-cluster-kubernetes-java) example project.
+To get started, it might be helpful to have a look at the example projects.
+
+* [Akka Cluster bootstrap using the Kubernetes API with Java/Maven](../attachments/akka-sample-cluster-kubernetes-java.zip)
+* [Akka Cluster bootstrap using the Kubernetes API with Scala/sbt](../attachments/akka-sample-cluster-kubernetes-scala.zip)
### Kubernetes Deployment
diff --git a/docs/src/main/paradox/kubernetes-deployment/index.md b/docs/src/main/paradox/kubernetes-deployment/index.md
index d355f0a8a..9c1c4bb12 100644
--- a/docs/src/main/paradox/kubernetes-deployment/index.md
+++ b/docs/src/main/paradox/kubernetes-deployment/index.md
@@ -3,8 +3,8 @@
For this guide we will be using the Akka Cluster in Kubernetes sample.
It is available for both:
- * [Java](https://developer.lightbend.com/start/?group=akka&project=akka-sample-cluster-kubernetes-java)
- * [Scala](https://developer.lightbend.com/start/?group=akka&project=akka-sample-cluster-kubernetes-scala)
+* [Java](../attachments/akka-sample-cluster-kubernetes-java.zip)
+* [Scala](../attachments/akka-sample-cluster-kubernetes-scala.zip)
@@toc { depth=2 }
diff --git a/docs/src/main/paradox/kubernetes-deployment/preparing-for-production.md b/docs/src/main/paradox/kubernetes-deployment/preparing-for-production.md
index 50338c96d..79d0a8b96 100644
--- a/docs/src/main/paradox/kubernetes-deployment/preparing-for-production.md
+++ b/docs/src/main/paradox/kubernetes-deployment/preparing-for-production.md
@@ -8,8 +8,8 @@ In preparation for production, we need to do two main things:
The final configuration file and deployment spec are in the sample application.
In this guide we will show snippets. Locations of the samples:
-* [Java](https://developer.lightbend.com/start/?group=akka&project=akka-sample-cluster-kubernetes-java)
-* [Scala](https://developer.lightbend.com/start/?group=akka&project=akka-sample-cluster-kubernetes-scala)
+* [Java](../attachments/akka-sample-cluster-kubernetes-java.zip)
+* [Scala](../attachments/akka-sample-cluster-kubernetes-scala.zip)
## Deployment Spec
diff --git a/integration-test/akka-sample-cluster-kubernetes-scala/test.sh b/integration-test/akka-sample-cluster-kubernetes-scala/test.sh
new file mode 100755
index 000000000..5a130f873
--- /dev/null
+++ b/integration-test/akka-sample-cluster-kubernetes-scala/test.sh
@@ -0,0 +1,63 @@
+#!/bin/bash
+
+set -exu
+
+export NAMESPACE=appka-1
+export APP_NAME=appka
+export PROJECT_DIR=samples/akka-sample-cluster-kubernetes-scala
+export DEPLOYMENT=kubernetes/akka-cluster.yml
+
+eval $(minikube -p minikube docker-env)
+cd $PROJECT_DIR
+sbt Docker/publishLocal
+
+docker images | head
+
+kubectl create namespace $NAMESPACE || true
+kubectl -n $NAMESPACE apply -f $DEPLOYMENT
+
+for i in {1..10}
+do
+ echo "Waiting for pods to get ready..."
+ kubectl get pods -n $NAMESPACE
+ [ `kubectl get pods -n $NAMESPACE | grep Running | wc -l` -eq 3 ] && break
+ sleep 4
+done
+
+if [ $i -eq 10 ]
+then
+ echo "Pods did not get ready"
+ kubectl -n $NAMESPACE events $APP_NAME
+ kubectl -n $NAMESPACE describe deployment $APP_NAME
+ exit -1
+fi
+
+POD=$(kubectl get pods -n $NAMESPACE | grep $APP_NAME | grep Running | head -n1 | awk '{ print $1 }')
+
+for i in {1..15}
+do
+ echo "Checking for MemberUp logging..."
+ kubectl logs $POD -n $NAMESPACE | grep MemberUp || true
+ [ `kubectl logs -n $NAMESPACE $POD | grep MemberUp | wc -l` -eq 3 ] && break
+ sleep 3
+done
+
+echo "Logs"
+echo "=============================="
+for POD in $(kubectl get pods -n $NAMESPACE | grep $APP_NAME | grep Running | awk '{ print $1 }')
+do
+ echo "Logging for $POD"
+ kubectl logs $POD -n $NAMESPACE
+done
+
+
+if [ $i -eq 15 ]
+then
+ echo "No 3 MemberUp log events found"
+ echo "=============================="
+
+ exit -1
+fi
+
+
+
diff --git a/integration-test/scripts/kubernetes-test.sh b/integration-test/scripts/kubernetes-test.sh
index e7bdd45f8..c8d90f812 100755
--- a/integration-test/scripts/kubernetes-test.sh
+++ b/integration-test/scripts/kubernetes-test.sh
@@ -5,7 +5,7 @@ sbt $PROJECT_NAME/Docker/publishLocal
docker images | head
-kubectl create namespace akka-bootstrap-demo-ns || true
+kubectl create namespace $NAMESPACE || true
kubectl -n $NAMESPACE apply -f $DEPLOYMENT
for i in {1..10}
diff --git a/samples/akka-sample-cluster-kubernetes-java/README.md b/samples/akka-sample-cluster-kubernetes-java/README.md
new file mode 100644
index 000000000..7e896ebc9
--- /dev/null
+++ b/samples/akka-sample-cluster-kubernetes-java/README.md
@@ -0,0 +1,58 @@
+# akka-sample-cluster-kubernetes-java
+akka sample cluster with kubernetes discovery in scala
+
+This is an example SBT project showing how to create an Akka Cluster on
+Kubernetes.
+
+## Kubernetes Instructions
+
+## Starting
+
+First, package the application and make it available locally as a docker image:
+
+ mvn clean package docker:build
+
+Then `akka-cluster.yml` should be sufficient to deploy a 2-node Akka Cluster, after
+creating a namespace for it:
+
+ kubectl apply -f kubernetes/namespace.json
+ kubectl config set-context --current --namespace=appka-1
+ kubectl apply -f kubernetes/akka-cluster.yml
+
+To check what you have done in Kubernetes so far, you can do:
+
+ kubectl get deployments
+ kubectl get pods
+ kubectl get replicasets
+ kubectl cluster-info dump
+ kubectl logs appka-79c98cf745-abcdee # pod name
+
+Finally, create a service so that you can then test [http://127.0.0.1:8080](http://127.0.0.1:8080)
+for 'hello world':
+
+ kubectl expose deployment appka --type=LoadBalancer --name=appka-service
+ kubectl port-forward svc/appka-service 8080:8080
+
+To wipe everything clean and start over, do:
+
+ kubectl delete namespaces appka-1
+
+## Running in a real Kubernetes cluster
+
+#### Publish to a registry the cluster can access e.g. Dockerhub with the kubakka user
+
+The app image must be in a registry the cluster can see. The build.sbt uses DockerHub by default.
+Use `mvn -Ddocker.registry=$DOCKER_REPO_URL/$NAMESPACE` if your cluster can't access DockerHub.
+
+To push an image to docker hub run:
+
+ mvn -am -pl bootstrap-demo-kubernetes-api package docker:push
+
+And remove the `imagePullPolicy: Never` from the deployments. Then you can use the same `kubectl` commands
+as described in the [Starting](#starting) section.
+
+## How it works
+
+This example uses [Akka Cluster Bootstrap](https://doc.akka.io/docs/akka-management/current/bootstrap/index.html)
+to initialize the cluster, using the [Kubernetes API discovery mechanism](https://doc.akka.io/docs/akka-management/current/discovery/index.html#discovery-method-kubernetes-api)
+to find peer nodes.
diff --git a/samples/akka-sample-cluster-kubernetes-java/kubernetes/akka-cluster.yml b/samples/akka-sample-cluster-kubernetes-java/kubernetes/akka-cluster.yml
new file mode 100644
index 000000000..6e3995e31
--- /dev/null
+++ b/samples/akka-sample-cluster-kubernetes-java/kubernetes/akka-cluster.yml
@@ -0,0 +1,82 @@
+apiVersion: apps/v1
+kind: Deployment
+metadata:
+ labels:
+ app: appka
+ name: appka
+ namespace: appka-1
+spec:
+ replicas: 3
+ selector:
+ matchLabels:
+ app: appka
+ template:
+ metadata:
+ labels:
+ app: appka
+ actorSystemName: appka
+ spec:
+ containers:
+ - name: appka
+ image: akka-sample-cluster-kubernetes:latest
+ # remove for real clusters, useful for minikube
+ imagePullPolicy: Never
+ readinessProbe:
+ httpGet:
+ path: /ready
+ port: management
+ periodSeconds: 10
+ failureThreshold: 3
+ initialDelaySeconds: 10
+ livenessProbe:
+ httpGet:
+ path: "/alive"
+ port: management
+ periodSeconds: 10
+ failureThreshold: 5
+ initialDelaySeconds: 20
+ ports:
+ # akka remoting
+ - name: remoting
+ containerPort: 2552
+ protocol: TCP
+ # akka-management and bootstrap
+ - name: management
+ containerPort: 8558
+ protocol: TCP
+ - name: http
+ containerPort: 8080
+ protocol: TCP
+ env:
+ - name: NAMESPACE
+ valueFrom:
+ fieldRef:
+ fieldPath: metadata.namespace
+ - name: REQUIRED_CONTACT_POINT_NR
+ value: "2"
+
+---
+kind: Role
+apiVersion: rbac.authorization.k8s.io/v1
+metadata:
+ name: pod-reader
+ namespace: appka-1
+rules:
+ - apiGroups: [""] # "" indicates the core API group
+ resources: ["pods"]
+ verbs: ["get", "watch", "list"]
+---
+kind: RoleBinding
+apiVersion: rbac.authorization.k8s.io/v1
+metadata:
+ name: read-pods
+ namespace: appka-1
+subjects:
+ # Note the `name` line below. The first default refers to the namespace. The second refers to the service account name.
+ # For instance, `name: system:serviceaccount:myns:default` would refer to the default service account in namespace `myns`
+ - kind: User
+ name: system:serviceaccount:appka-1:default
+roleRef:
+ kind: Role
+ name: pod-reader
+ apiGroup: rbac.authorization.k8s.io
diff --git a/samples/akka-sample-cluster-kubernetes-java/kubernetes/namespace.json b/samples/akka-sample-cluster-kubernetes-java/kubernetes/namespace.json
new file mode 100644
index 000000000..466d18c92
--- /dev/null
+++ b/samples/akka-sample-cluster-kubernetes-java/kubernetes/namespace.json
@@ -0,0 +1,11 @@
+{
+ "kind": "Namespace",
+ "apiVersion": "v1",
+ "metadata": {
+ "name": "appka-1",
+ "labels": {
+ "name": "appka-1"
+ }
+ }
+}
+
diff --git a/samples/akka-sample-cluster-kubernetes-java/pom.xml b/samples/akka-sample-cluster-kubernetes-java/pom.xml
new file mode 100644
index 000000000..8f1ff2be5
--- /dev/null
+++ b/samples/akka-sample-cluster-kubernetes-java/pom.xml
@@ -0,0 +1,152 @@
+
+ 4.0.0
+ com.lightbend
+ akka-sample-cluster-kubernetes
+ 1.0
+ akka-sample-cluster-kubernetes-java
+ Akka Sample for forming a Cluster in Kubernetes
+
+
+ 17
+ 17
+ UTF-8
+ 2.9.2
+ 1.5.1
+ 10.6.2
+ 2.13
+ ${git.commit.time}-${git.commit.id.abbrev}
+
+
+
+
+ akka-repository
+ Akka library repository
+ https://repo.akka.io/maven
+
+
+
+
+ akka-repository
+ Akka library repository
+ https://repo.akka.io/maven
+
+
+
+
+
+ com.typesafe.akka
+ akka-http_${scala.binary.version}
+ ${akka-http.version}
+
+
+ com.typesafe.akka
+ akka-http-spray-json_${scala.binary.version}
+ ${akka-http.version}
+
+
+ com.typesafe.akka
+ akka-cluster-typed_${scala.binary.version}
+ ${akka.version}
+
+
+ com.typesafe.akka
+ akka-cluster-sharding-typed_${scala.binary.version}
+ ${akka.version}
+
+
+ com.typesafe.akka
+ akka-slf4j_${scala.binary.version}
+ ${akka.version}
+
+
+ com.typesafe.akka
+ akka-stream-typed_${scala.binary.version}
+ ${akka.version}
+
+
+ com.typesafe.akka
+ akka-discovery_${scala.binary.version}
+ ${akka.version}
+
+
+ ch.qos.logback
+ logback-classic
+ 1.2.13
+
+
+ com.lightbend.akka.management
+ akka-management-cluster-bootstrap_${scala.binary.version}
+ ${akka-management.version}
+
+
+ com.lightbend.akka.discovery
+ akka-discovery-kubernetes-api_${scala.binary.version}
+ ${akka-management.version}
+
+
+ com.lightbend.akka.management
+ akka-management-cluster-http_${scala.binary.version}
+ ${akka-management.version}
+
+
+
+
+
+ io.fabric8
+ docker-maven-plugin
+ 0.42.0
+
+
+
+ %a
+
+ docker.io/library/eclipse-temurin:17.0.8.1_1-jre
+
+ ${version.number}
+
+
+
+ java
+ -cp
+ /maven/*
+ akka.cluster.bootstrap.demo.DemoApp
+
+
+
+ artifact-with-dependencies
+
+
+
+
+
+
+
+ build-docker-image
+ package
+
+ build
+
+
+
+
+
+ pl.project13.maven
+ git-commit-id-plugin
+ 4.0.0
+
+
+ validate
+
+ revision
+
+
+
+
+ yyyyMMdd-HHmmss
+ ${project.basedir}/.git
+ false
+
+
+
+
+
diff --git a/samples/akka-sample-cluster-kubernetes-java/src/main/java/akka/cluster/bootstrap/demo/DemoApp.java b/samples/akka-sample-cluster-kubernetes-java/src/main/java/akka/cluster/bootstrap/demo/DemoApp.java
new file mode 100644
index 000000000..272c480bc
--- /dev/null
+++ b/samples/akka-sample-cluster-kubernetes-java/src/main/java/akka/cluster/bootstrap/demo/DemoApp.java
@@ -0,0 +1,73 @@
+/*
+ * Copyright (C) 2017 Lightbend Inc.
+ */
+package akka.cluster.bootstrap.demo;
+
+import akka.actor.typed.ActorRef;
+import akka.actor.typed.ActorSystem;
+import akka.actor.typed.Behavior;
+import akka.actor.typed.javadsl.Adapter;
+import akka.actor.typed.javadsl.Behaviors;
+import akka.cluster.ClusterEvent;
+import akka.cluster.typed.Cluster;
+import akka.cluster.typed.Subscribe;
+import akka.http.javadsl.ConnectHttp;
+import akka.http.javadsl.Http;
+import static akka.http.javadsl.server.Directives.*;
+import akka.management.cluster.bootstrap.ClusterBootstrap;
+import akka.management.scaladsl.AkkaManagement;
+import akka.stream.Materializer;
+
+public class DemoApp {
+
+ static class MemberEventLogger {
+ public static Behavior create() {
+ return Behaviors.setup(context -> {
+ Cluster cluster = Cluster.get(context.getSystem());
+
+ context.getLog().info("Started [{}], cluster.selfAddress = {})",
+ context.getSystem(),
+ cluster.selfMember().address());
+
+ cluster.subscriptions().tell(new Subscribe<>(context.getSelf(), ClusterEvent.MemberEvent.class));
+
+ return Behaviors.receiveMessage(event -> {
+ context.getLog().info("MemberEvent: {}", event);
+ return Behaviors.same();
+ });
+ });
+ }
+ }
+
+ static class Guardian {
+ public static Behavior create() {
+ return Behaviors.setup(context -> {
+ final akka.actor.ActorSystem classicSystem = Adapter.toClassic(context.getSystem());
+ Materializer mat = Materializer.matFromSystem(classicSystem);
+
+ Http.get(classicSystem).bindAndHandle(complete("Hello world")
+ .flow(classicSystem, mat), ConnectHttp.toHost("0.0.0.0", 8080), mat)
+ .whenComplete((binding, failure) -> {
+ if (failure == null) {
+ classicSystem.log().info("HTTP server now listening at port 8080");
+ } else {
+ classicSystem.log().error(failure, "Failed to bind HTTP server, terminating.");
+ classicSystem.terminate();
+ }
+ });
+
+ context.spawn(MemberEventLogger.create(), "listener");
+
+ AkkaManagement.get(classicSystem).start();
+ ClusterBootstrap.get(classicSystem).start();
+
+ return Behaviors.empty();
+ });
+ }
+ }
+
+ public static void main(String[] args) {
+ ActorSystem.create(Guardian.create(), "Appka");
+ }
+
+}
\ No newline at end of file
diff --git a/samples/akka-sample-cluster-kubernetes-java/src/main/java/akka/cluster/bootstrap/demo/DemoHealthCheck.java b/samples/akka-sample-cluster-kubernetes-java/src/main/java/akka/cluster/bootstrap/demo/DemoHealthCheck.java
new file mode 100644
index 000000000..b42430d92
--- /dev/null
+++ b/samples/akka-sample-cluster-kubernetes-java/src/main/java/akka/cluster/bootstrap/demo/DemoHealthCheck.java
@@ -0,0 +1,22 @@
+package akka.cluster.bootstrap.demo;
+
+import akka.actor.ActorSystem;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+import java.util.concurrent.CompletableFuture;
+import java.util.concurrent.CompletionStage;
+import java.util.function.Supplier;
+
+public class DemoHealthCheck implements Supplier> {
+ private final Logger log = LoggerFactory.getLogger(getClass());
+
+ public DemoHealthCheck(ActorSystem system) {
+ }
+
+ @Override
+ public CompletionStage get() {
+ log.info("DemoHealthCheck called");
+ return CompletableFuture.completedFuture(true);
+ }
+}
diff --git a/samples/akka-sample-cluster-kubernetes-java/src/main/resources/application.conf b/samples/akka-sample-cluster-kubernetes-java/src/main/resources/application.conf
new file mode 100644
index 000000000..8ab258e39
--- /dev/null
+++ b/samples/akka-sample-cluster-kubernetes-java/src/main/resources/application.conf
@@ -0,0 +1,36 @@
+akka {
+ loglevel = "DEBUG"
+ actor.provider = cluster
+
+ coordinated-shutdown.exit-jvm = on
+
+ cluster {
+ shutdown-after-unsuccessful-join-seed-nodes = 60s
+ downing-provider-class = "akka.cluster.sbr.SplitBrainResolverProvider"
+ }
+}
+
+#management-config
+akka.management {
+ cluster.bootstrap {
+ contact-point-discovery {
+ # For the kubernetes API this value is substributed into the %s in pod-label-selector
+ service-name = "appka"
+
+ # pick the discovery method you'd like to use:
+ discovery-method = kubernetes-api
+
+ required-contact-point-nr = 2
+ required-contact-point-nr = ${?REQUIRED_CONTACT_POINT_NR}
+ }
+ }
+}
+#management-config
+
+akka.management {
+ health-checks {
+ readiness-checks {
+ example-ready = "akka.cluster.bootstrap.demo.DemoHealthCheck"
+ }
+ }
+}
diff --git a/samples/akka-sample-cluster-kubernetes-java/src/main/resources/logback.xml b/samples/akka-sample-cluster-kubernetes-java/src/main/resources/logback.xml
new file mode 100644
index 000000000..058a3ff5e
--- /dev/null
+++ b/samples/akka-sample-cluster-kubernetes-java/src/main/resources/logback.xml
@@ -0,0 +1,21 @@
+
+
+
+
+ System.out
+
+ [%date{ISO8601}] [%level] [%logger] [%marker] [%thread] - %msg MDC: {%mdc}%n
+
+
+
+
+ 8192
+ true
+
+
+
+
+
+
+
+
diff --git a/samples/akka-sample-cluster-kubernetes-scala/README.md b/samples/akka-sample-cluster-kubernetes-scala/README.md
new file mode 100644
index 000000000..4df9ea723
--- /dev/null
+++ b/samples/akka-sample-cluster-kubernetes-scala/README.md
@@ -0,0 +1,60 @@
+# akka-sample-cluster-kubernetes-scala
+akka sample cluster with kubernetes discovery in scala
+
+This is an example SBT project showing how to create an Akka Cluster on
+Kubernetes.
+
+## Kubernetes Instructions
+
+## Starting
+
+First, package the application and make it available locally as a docker image:
+
+ sbt Docker/publishLocal
+
+Then `akka-cluster.yml` should be sufficient to deploy a 2-node Akka Cluster, after
+creating a namespace for it:
+
+ kubectl apply -f kubernetes/namespace.json
+ kubectl config set-context --current --namespace=appka-1
+ kubectl apply -f kubernetes/akka-cluster.yml
+
+To check what you have done in Kubernetes so far, you can do:
+
+ kubectl get deployments
+ kubectl get pods
+ kubectl get replicasets
+ kubectl cluster-info dump
+ kubectl logs appka-79c98cf745-abcdee # pod name
+
+Finally, create a service so that you can then test [http://127.0.0.1:8080](http://127.0.0.1:8080)
+for 'hello world':
+
+ kubectl expose deployment appka --type=LoadBalancer --name=appka-service
+ kubectl port-forward svc/appka-service 8080:8080
+
+To wipe everything clean and start over, do:
+
+ kubectl delete namespaces appka-1
+
+## Running in a real Kubernetes cluster
+
+#### Publish to a registry the cluster can access e.g. Dockerhub with the kubakka user
+
+The app image must be in a registry the cluster can see. The build.sbt uses DockerHub by default.
+Start with `sbt -Ddocker.registry=your-registry` if your cluster can't access DockerHub.
+
+The user for the registry is defined with `sbt -Ddocker.username=your-user`
+
+To push an image to docker hub run:
+
+ `sbt -Ddocker.username=your-user docker:publish`
+
+And remove the `imagePullPolicy: Never` from the deployments. Then you can use the same `kubectl` commands
+as described in the [Starting](#starting) section.
+
+## How it works
+
+This example uses [Akka Cluster Bootstrap](https://doc.akka.io/docs/akka-management/current/bootstrap/index.html)
+to initialize the cluster, using the [Kubernetes API discovery mechanism](https://doc.akka.io/docs/akka-management/current/discovery/index.html#discovery-method-kubernetes-api)
+to find peer nodes.
diff --git a/samples/akka-sample-cluster-kubernetes-scala/build.sbt b/samples/akka-sample-cluster-kubernetes-scala/build.sbt
new file mode 100644
index 000000000..a7e9c24ac
--- /dev/null
+++ b/samples/akka-sample-cluster-kubernetes-scala/build.sbt
@@ -0,0 +1,46 @@
+ThisBuild / organization := "com.lightbend"
+
+name := "akka-sample-cluster-kubernetes"
+
+scalaVersion := "2.13.13"
+lazy val akkaHttpVersion = "10.6.2"
+lazy val akkaVersion = "2.9.2"
+lazy val akkaManagementVersion = "1.5.1"
+
+// make version compatible with docker for publishing
+ThisBuild / dynverSeparator := "-"
+
+scalacOptions := Seq("-feature", "-unchecked", "-deprecation", "-encoding", "utf8")
+classLoaderLayeringStrategy := ClassLoaderLayeringStrategy.AllLibraryJars
+run / fork := true
+Compile / run / fork := true
+Compile / run / mainClass := Some("akka.sample.cluster.kubernetes.DemoApp")
+
+enablePlugins(JavaServerAppPackaging, DockerPlugin)
+
+dockerExposedPorts := Seq(8080, 8558, 25520)
+dockerUpdateLatest := true
+dockerUsername := sys.props.get("docker.username")
+dockerRepository := sys.props.get("docker.registry")
+dockerBaseImage := "adoptopenjdk:11-jre-hotspot"
+
+resolvers += "Akka library repository".at("https://repo.akka.io/maven")
+
+libraryDependencies ++= {
+ Seq(
+ "com.typesafe.akka" %% "akka-http" % akkaHttpVersion,
+ "com.typesafe.akka" %% "akka-http-spray-json" % akkaHttpVersion,
+ "com.typesafe.akka" %% "akka-cluster-typed" % akkaVersion,
+ "com.typesafe.akka" %% "akka-cluster-sharding-typed" % akkaVersion,
+ "com.typesafe.akka" %% "akka-stream-typed" % akkaVersion,
+ "com.typesafe.akka" %% "akka-discovery" % akkaVersion,
+ "ch.qos.logback" % "logback-classic" % "1.2.13",
+ "com.lightbend.akka.discovery" %% "akka-discovery-kubernetes-api" % akkaManagementVersion,
+ "com.lightbend.akka.management" %% "akka-management-cluster-bootstrap" % akkaManagementVersion,
+ "com.lightbend.akka.management" %% "akka-management-cluster-http" % akkaManagementVersion,
+ "com.typesafe.akka" %% "akka-testkit" % akkaVersion % "test",
+ "com.typesafe.akka" %% "akka-actor-testkit-typed" % akkaVersion % Test,
+ "com.typesafe.akka" %% "akka-http-testkit" % akkaHttpVersion % Test,
+ "com.typesafe.akka" %% "akka-testkit" % akkaVersion % Test,
+ "com.typesafe.akka" %% "akka-stream-testkit" % akkaVersion % Test)
+}
diff --git a/samples/akka-sample-cluster-kubernetes-scala/kubernetes/akka-cluster.yml b/samples/akka-sample-cluster-kubernetes-scala/kubernetes/akka-cluster.yml
new file mode 100644
index 000000000..703777919
--- /dev/null
+++ b/samples/akka-sample-cluster-kubernetes-scala/kubernetes/akka-cluster.yml
@@ -0,0 +1,78 @@
+apiVersion: apps/v1
+kind: Deployment
+metadata:
+ labels:
+ app: appka
+ name: appka
+ namespace: appka-1
+spec:
+ replicas: 3
+ selector:
+ matchLabels:
+ app: appka
+ template:
+ metadata:
+ labels:
+ app: appka
+ actorSystemName: appka
+ spec:
+ containers:
+ - name: appka
+ image: akka-sample-cluster-kubernetes:latest
+ # remove for real clusters, useful for minikube
+ imagePullPolicy: Never
+ readinessProbe:
+ httpGet:
+ path: /ready
+ port: management
+ periodSeconds: 10
+ failureThreshold: 3
+ initialDelaySeconds: 10
+ livenessProbe:
+ httpGet:
+ path: "/alive"
+ port: management
+ periodSeconds: 10
+ failureThreshold: 5
+ initialDelaySeconds: 20
+ ports:
+ # akka-management and bootstrap
+ - name: management
+ containerPort: 8558
+ protocol: TCP
+ - name: http
+ containerPort: 8080
+ protocol: TCP
+ env:
+ - name: NAMESPACE
+ valueFrom:
+ fieldRef:
+ fieldPath: metadata.namespace
+ - name: REQUIRED_CONTACT_POINT_NR
+ value: "3"
+
+---
+kind: Role
+apiVersion: rbac.authorization.k8s.io/v1
+metadata:
+ name: pod-reader
+ namespace: appka-1
+rules:
+- apiGroups: [""] # "" indicates the core API group
+ resources: ["pods"]
+ verbs: ["get", "watch", "list"]
+---
+kind: RoleBinding
+apiVersion: rbac.authorization.k8s.io/v1
+metadata:
+ name: read-pods
+ namespace: appka-1
+subjects:
+# Note the `name` line below. The first default refers to the namespace. The second refers to the service account name.
+# For instance, `name: system:serviceaccount:myns:default` would refer to the default service account in namespace `myns`
+- kind: User
+ name: system:serviceaccount:appka-1:default
+roleRef:
+ kind: Role
+ name: pod-reader
+ apiGroup: rbac.authorization.k8s.io
diff --git a/samples/akka-sample-cluster-kubernetes-scala/kubernetes/namespace.json b/samples/akka-sample-cluster-kubernetes-scala/kubernetes/namespace.json
new file mode 100644
index 000000000..466d18c92
--- /dev/null
+++ b/samples/akka-sample-cluster-kubernetes-scala/kubernetes/namespace.json
@@ -0,0 +1,11 @@
+{
+ "kind": "Namespace",
+ "apiVersion": "v1",
+ "metadata": {
+ "name": "appka-1",
+ "labels": {
+ "name": "appka-1"
+ }
+ }
+}
+
diff --git a/samples/akka-sample-cluster-kubernetes-scala/project/build.properties b/samples/akka-sample-cluster-kubernetes-scala/project/build.properties
new file mode 100644
index 000000000..abbbce5da
--- /dev/null
+++ b/samples/akka-sample-cluster-kubernetes-scala/project/build.properties
@@ -0,0 +1 @@
+sbt.version=1.9.8
diff --git a/samples/akka-sample-cluster-kubernetes-scala/project/plugins.sbt b/samples/akka-sample-cluster-kubernetes-scala/project/plugins.sbt
new file mode 100644
index 000000000..8bab11d60
--- /dev/null
+++ b/samples/akka-sample-cluster-kubernetes-scala/project/plugins.sbt
@@ -0,0 +1,4 @@
+resolvers += "Akka library repository".at("https://repo.akka.io/maven")
+
+addSbtPlugin("com.github.sbt" % "sbt-native-packager" % "1.9.16")
+addSbtPlugin("com.github.sbt" % "sbt-dynver" % "5.0.1")
diff --git a/samples/akka-sample-cluster-kubernetes-scala/src/main/resources/application.conf b/samples/akka-sample-cluster-kubernetes-scala/src/main/resources/application.conf
new file mode 100644
index 000000000..3c8624381
--- /dev/null
+++ b/samples/akka-sample-cluster-kubernetes-scala/src/main/resources/application.conf
@@ -0,0 +1,31 @@
+akka {
+ loglevel = "DEBUG"
+ actor.provider = cluster
+
+ coordinated-shutdown.exit-jvm = on
+
+ cluster {
+ shutdown-after-unsuccessful-join-seed-nodes = 60s
+ }
+}
+
+#management-config
+akka.management {
+ cluster.bootstrap {
+ contact-point-discovery {
+ # pick the discovery method you'd like to use:
+ discovery-method = kubernetes-api
+
+ required-contact-point-nr = ${REQUIRED_CONTACT_POINT_NR}
+ }
+ }
+}
+#management-config
+
+akka.management {
+ health-checks {
+ readiness-checks {
+ example-ready = "akka.sample.cluster.kubernetes.DemoHealthCheck"
+ }
+ }
+}
diff --git a/samples/akka-sample-cluster-kubernetes-scala/src/main/resources/logback.xml b/samples/akka-sample-cluster-kubernetes-scala/src/main/resources/logback.xml
new file mode 100644
index 000000000..058a3ff5e
--- /dev/null
+++ b/samples/akka-sample-cluster-kubernetes-scala/src/main/resources/logback.xml
@@ -0,0 +1,21 @@
+
+
+
+
+ System.out
+
+ [%date{ISO8601}] [%level] [%logger] [%marker] [%thread] - %msg MDC: {%mdc}%n
+
+
+
+
+ 8192
+ true
+
+
+
+
+
+
+
+
diff --git a/samples/akka-sample-cluster-kubernetes-scala/src/main/scala/akka/sample/cluster/kubernetes/DemoApp.scala b/samples/akka-sample-cluster-kubernetes-scala/src/main/scala/akka/sample/cluster/kubernetes/DemoApp.scala
new file mode 100644
index 000000000..8ce065843
--- /dev/null
+++ b/samples/akka-sample-cluster-kubernetes-scala/src/main/scala/akka/sample/cluster/kubernetes/DemoApp.scala
@@ -0,0 +1,37 @@
+package akka.sample.cluster.kubernetes
+
+import akka.actor.typed.ActorSystem
+import akka.actor.typed.scaladsl.Behaviors
+import akka.cluster.ClusterEvent
+import akka.cluster.typed.{ Cluster, Subscribe }
+import akka.http.scaladsl.Http
+import akka.http.scaladsl.server.Directives._
+import akka.management.cluster.bootstrap.ClusterBootstrap
+import akka.management.javadsl.AkkaManagement
+import akka.{ actor => classic }
+
+object DemoApp extends App {
+
+ ActorSystem[Nothing](Behaviors.setup[Nothing] { context =>
+ import akka.actor.typed.scaladsl.adapter._
+ implicit val classicSystem: classic.ActorSystem = context.system.toClassic
+ implicit val ec = context.system.executionContext
+
+ val cluster = Cluster(context.system)
+ context.log.info("Started [" + context.system + "], cluster.selfAddress = " + cluster.selfMember.address + ")")
+
+ Http().newServerAt("0.0.0.0", 8080).bind(complete("Hello world"))
+
+ // Create an actor that handles cluster domain events
+ val listener = context.spawn(Behaviors.receive[ClusterEvent.MemberEvent]((ctx, event) => {
+ ctx.log.info("MemberEvent: {}", event)
+ Behaviors.same
+ }), "listener")
+
+ Cluster(context.system).subscriptions ! Subscribe(listener, classOf[ClusterEvent.MemberEvent])
+
+ AkkaManagement.get(classicSystem).start()
+ ClusterBootstrap.get(classicSystem).start()
+ Behaviors.empty
+ }, "appka")
+}
diff --git a/samples/akka-sample-cluster-kubernetes-scala/src/main/scala/akka/sample/cluster/kubernetes/DemoHealthCheck.scala b/samples/akka-sample-cluster-kubernetes-scala/src/main/scala/akka/sample/cluster/kubernetes/DemoHealthCheck.scala
new file mode 100644
index 000000000..f29de3e67
--- /dev/null
+++ b/samples/akka-sample-cluster-kubernetes-scala/src/main/scala/akka/sample/cluster/kubernetes/DemoHealthCheck.scala
@@ -0,0 +1,16 @@
+package akka.sample.cluster.kubernetes
+
+import scala.concurrent.Future
+
+import akka.actor.ActorSystem
+import org.slf4j.LoggerFactory
+
+// Enabled in application.conf
+class DemoHealthCheck(system: ActorSystem) extends (() => Future[Boolean]) {
+ private val log = LoggerFactory.getLogger(getClass)
+
+ override def apply(): Future[Boolean] = {
+ log.info("DemoHealthCheck called")
+ Future.successful(true)
+ }
+}
diff --git a/scripts/prepare-downloads.sh b/scripts/prepare-downloads.sh
new file mode 100755
index 000000000..6b59a0707
--- /dev/null
+++ b/scripts/prepare-downloads.sh
@@ -0,0 +1,73 @@
+#!/usr/bin/env bash
+
+set -e
+
+declare -r samples_sources="${PWD}/samples"
+declare -r docs_attachments="${PWD}/docs/src/main/paradox/attachments"
+declare -r target_temporal_attachments="${PWD}/target/docs/_attachments"
+
+declare -r temporal_folder="${PWD}/target/zips"
+
+function sed_command() {
+ local platform="$(uname -s | tr '[:upper:]' '[:lower:]')"
+
+ if [ "${platform}" != "darwin" ]; then
+ echo "sed"
+ else
+ # using gnu-sed on Mac
+ echo "gsed"
+ fi
+}
+
+## Remove the tags used by Paradox snippets from the codebase in the current folder
+function removeTags() {
+ ## remove tags from code
+ find . -type f -print0 | xargs -0 $(sed_command) -i "s/\/\/ #.*//g"
+}
+
+
+## Cleanup the temporal folder from previous executions
+function prepareTemporalFolder() {
+ rm -rf ${temporal_folder}
+ mkdir -p ${temporal_folder}
+}
+
+## Copy a folder with some code into the temporal folder. The
+## copied folder will be renamed to the folder name we want the
+## user to see when unzipping the file.
+## source_name -> folder in `examples`
+## target_name -> folder name the user should see (must not use a numeric prefix of a laguage suffix)
+function fetchProject() {
+ source_name=$1
+ target_name=$2
+ echo "Fetching content from [$1] to [$2]"
+ cp -a ${source_name} ${temporal_folder}/${target_name}
+ rm -rf ${temporal_folder}/${target_name}/target
+}
+
+## Zip the contents in $temporal_folder and create the
+## attachment file (aka, the zip file on the appropriate location)
+function zipAndAttach() {
+ zip_name=$1
+ temporal_attachments=$2
+ echo "Preparing zip $1"
+ pushd ${temporal_folder}
+ removeTags
+ zip --quiet -r ${zip_name} *
+ cp ${zip_name} ${temporal_attachments}
+ echo "Prepared attachment at ${zip_name}"
+ popd
+}
+
+mkdir -p ${docs_attachments}
+mkdir -p ${target_temporal_attachments}
+
+## akka-sample-cluster-kubernetes-java zip file
+prepareTemporalFolder
+fetchProject ${samples_sources}/akka-sample-cluster-kubernetes-java akka-sample-cluster-kubernetes
+zipAndAttach ${docs_attachments}/akka-sample-cluster-kubernetes-java.zip ${target_temporal_attachments}
+
+## akka-sample-cluster-kubernetes-scala zip file
+prepareTemporalFolder
+fetchProject ${samples_sources}/akka-sample-cluster-kubernetes-scala akka-sample-cluster-kubernetes
+zipAndAttach ${docs_attachments}/akka-sample-cluster-kubernetes-scala.zip ${target_temporal_attachments}