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



No account? Register here

Already registered? Sign in to your account now.

Sign in with GitHub
Sign in with Google
  • Comments

    Online: 1

Please sign in to be able to write comments.

by Gabriel Garrido