Running a phoenix app in a multinode fashion in kubernetes
Introduction
In this brief update, I’m going to make all nodes able to connect and communicate with each other using the library libcluster with some basic configuration.
Upgrading packages
For that to work you need to add in mix.exs
:
{:libcluster, "~> 3.3"},
Update your lib/tr/application.ex
file to start libcluster
topologies = Application.get_env(:libcluster, :topologies, [])
children = [
# Start the Cluster supervisor for libcluster
{Cluster.Supervisor, [topologies, [name: Tr.ClusterSupervisor]]},
...
Then we need to update our config/prod.exs
file to tell libcluster what to look for in the cluster:
# Libcluster configuration
config :libcluster,
topologies: [
erlang_nodes_in_k8s: [
strategy: Elixir.Cluster.Strategy.Kubernetes.DNS,
config: [
service: "tr-cluster-svc",
application_name: "tr",
kubernetes_namespace: "tr",
polling_interval: 10_000
]
]
]
Dev config config/dev.exs
:
config :libcluster,
topologies: [
example: [
strategy: Cluster.Strategy.Epmd,
config: [hosts: [:"a@127.0.0.1", :"b@127.0.0.1"]],
connect: {:net_kernel, :connect_node, []},
disconnect: {:erlang, :disconnect_node, []},
list_nodes: {:erlang, :nodes, [:connected]}
]
]
That’s enough for elixir to try to find the other pods and attempt to connect to the nodes, however we need to allow
that communication and let the pods read the information from the kubernetes API, next up add these permissions to your
deployment 05-role.yaml
:
apiVersion: rbac.authorization.k8s.io/v1
kind: Role
metadata:
namespace: tr
name: pod-reader
rules:
- apiGroups: [""]
resources: ["endpoints"]
verbs: ["get", "watch", "list"]
---
apiVersion: rbac.authorization.k8s.io/v1
kind: RoleBinding
metadata:
name: read-pods
namespace: tr
subjects:
- kind: ServiceAccount
name: tr
roleRef:
kind: Role
name: pod-reader
apiGroup: rbac.authorization.k8s.io
Then in your 02-deployment.yaml
file:
env:
- name: POD_IP
valueFrom:
fieldRef:
fieldPath: status.podIP
And a service to make things easier 03-service.yaml
(epmd port):
apiVersion: v1
kind: Service
metadata:
name: tr-cluster-svc
namespace: tr
spec:
clusterIP: None
selector:
name: tr
This is to set the right environment variables for the application to use and to be able to connect to the other
nodes
.
Before moving on, make sure you generate the release files:
mix release.init
And now update your rel/env.sh.eex
file so it looks like this:
export RELEASE_DISTRIBUTION=name
export RELEASE_NODE=<%= @release.name %>@${POD_IP}
If everything went well, you should see something like this in the logs:
tr-deployment-6cf5c65b56-ndrgm tr 04:13:13.411 [info] [libcluster:erlang_nodes_in_k8s] connected to :"tr@10.42.1.217"
tr-deployment-6cf5c65b56-ndrgm tr 04:13:13.416 [info] [libcluster:erlang_nodes_in_k8s] connected to :"tr@10.42.3.185"
If you want to validate it locally, use this command instead from iex
:
❯ iex --name a@127.0.0.1 --cookie secret -S mix
❯ iex --name b@127.0.0.1 --cookie secret -S mix
iex(b@127.0.0.1)> Node.list()
Some useful links: https://hexdocs.pm/libcluster/readme.html https://hexdocs.pm/libcluster/Cluster.Strategy.Kubernetes.DNS.html https://brain.d.foundation/Engineering/Backend/libcluster+in+elixir
and last but not least good luck!
Closing notes
Let me know if there is anything that you would like to see implemented or tested, explored and what not in here…
Errata
If you spot any error or have any suggestion, please send me a message so it gets fixed.
Also, you can check the source code and changes in the sources here
-
Comments
Online: 1
Please sign in to be able to write comments.