Clustering

NATS Server Clustering

NATS supports running each server in clustered mode. You can cluster servers together for high volume messaging systems and resiliency and high availability.

NATS servers achieve this by gossiping about and connecting to, all of the servers they know, thus dynamically forming a full mesh. Once clients connect or re-connect to a particular server, they are informed about current cluster members. Because of this behavior, a cluster can grow, shrink and self heal. The full mesh does not necessarily have to be explicitly configured either.

Note that NATS clustered servers have a forwarding limit of one hop. This means that each nats-server instance will only forward messages that it has received from a client to the immediately adjacent nats-server instances to which it has routes. Messages received from a route will only be distributed to local clients.

For the cluster to successfully form a full mesh and NATS to function as intended and described throughout the documentation - temporary errors permitting - it is necessary that servers can connect to each other and that clients can connect to each server in the cluster.

Cluster URLs

In addition to a port to listen for clients, nats-server listens on a “cluster” URL (the -cluster option). Additional nats-server servers can then add that URL to their -routes argument to join the cluster. These options can also be specified in a config file, but only the command-line version is shown in this overview for simplicity.

Running a Simple Cluster

Here is a simple cluster running on the same machine:

  1. # Server A - the 'seed server'
  2. > nats-server -p 4222 -cluster nats://localhost:4248 --cluster_name test-cluster
  3. # Server B
  4. > nats-server -p 5222 -cluster nats://localhost:5248 -routes nats://localhost:4248 --cluster_name test-cluster
  5. # Check the output of the server for the selected client and route ports.
  6. # Server C
  7. > nats-server -p 6222 -cluster nats://localhost:6248 -routes nats://localhost:4248 --cluster_name test-cluster
  8. # Check the output of the server for the selected client and route ports.

Each server has a client and cluster port specified. Servers with the routes option establish a route to the seed server. Because the clustering protocol gossips members of the cluster, all servers are able to discover other server in the cluster. When a server is discovered, the discovering server will automatically attempt to connect to it in order to form a full mesh. Typically only one instance of the server will run per machine, so you can reuse the client port (4222) and the cluster port (4248), and simply the route to the host/port of the seed server.

Similarly, clients connecting to any server in the cluster will discover other servers in the cluster. If the connection to the server is interrupted, the client will attempt to connect to all other known servers.

There is no explicit configuration for seed server. They simply serve as the starting point for server discovery by other members of the cluster as well as clients. As such these are the servers that clients have in their list of connect urls and cluster members have in their list of routes. They reduce configuration as not every server needs to be in these lists. But the ability for other server and clients to successfully connect depends on seed server running. If multiple seed server are used, they make use of the routes option as well, so they can establish routes to one another.

Command Line Options

The following cluster options are supported:

  1. --routes [rurl-1, rurl-2] Routes to solicit and connect
  2. --cluster nats://host:port Cluster URL for solicited routes

When a NATS server routes to a specified URL, it will advertise its own cluster URL to all other servers in the route effectively creating a routing mesh to all other servers.

Note: when using the -routes option, you must also specify a -cluster option.

Clustering can also be configured using the server config file.

Three Server Cluster Example

The following example demonstrates how to run a cluster of 3 servers on the same host. We will start with the seed server and use the -D command line parameter to produce debug information.

  1. nats-server -p 4222 -cluster nats://localhost:5222 -D

Alternatively, you could use a configuration file, let’s call it seed.conf, with a content similar to this:

  1. # Cluster Seed Node
  2. listen: 127.0.0.1:4222
  3. http: 8222
  4. cluster {
  5. listen: 127.0.0.1:4248
  6. }

And start the server like this:

  1. nats-server -config ./seed.conf -D

This will produce an output similar to:

  1. [83329] 2020/02/12 16:04:52.369039 [INF] Starting nats-server version 2.1.4
  2. [83329] 2020/02/12 16:04:52.369130 [DBG] Go build version go1.13.6
  3. [83329] 2020/02/12 16:04:52.369133 [INF] Git commit [not set]
  4. [83329] 2020/02/12 16:04:52.369360 [INF] Starting http monitor on 127.0.0.1:8222
  5. [83329] 2020/02/12 16:04:52.369436 [INF] Listening for client connections on 127.0.0.1:4222
  6. [83329] 2020/02/12 16:04:52.369441 [INF] Server id is NDSGCS74MG5ZUMBOVWOUJ5S3HIOW
  7. [83329] 2020/02/12 16:04:52.369443 [INF] Server is ready
  8. [83329] 2020/02/12 16:04:52.369534 [INF] Listening for route connections on 127.0.0.1:4248

It is also possible to specify the hostname and port independently. At the minimum, the port is required. If you leave the hostname off it will bind to all the interfaces (‘0.0.0.0’).

  1. cluster {
  2. host: 127.0.0.1
  3. port: 4248
  4. }

Now let’s start two more servers, each one connecting to the seed server.

  1. nats-server -p 5222 -cluster nats://localhost:5248 -routes nats://localhost:4248 -D

When running on the same host, we need to pick different ports for the client connections -p, and for the port used to accept other routes -cluster. Note that -routes points to the -cluster address of the seed server (localhost:4248).

Here is the log produced. See how it connects and registers a route to the seed server (...GzM).

  1. [83330] 2020/02/12 16:05:09.661047 [INF] Starting nats-server version 2.1.4
  2. [83330] 2020/02/12 16:05:09.661123 [DBG] Go build version go1.13.6
  3. [83330] 2020/02/12 16:05:09.661125 [INF] Git commit [not set]
  4. [83330] 2020/02/12 16:05:09.661341 [INF] Listening for client connections on 0.0.0.0:5222
  5. [83330] 2020/02/12 16:05:09.661347 [INF] Server id is NAABC2CKRVPZBIECMLZZA6L3PK
  6. [83330] 2020/02/12 16:05:09.661349 [INF] Server is ready
  7. [83330] 2020/02/12 16:05:09.662429 [INF] Listening for route connections on localhost:5248
  8. [83330] 2020/02/12 16:05:09.662676 [DBG] Trying to connect to route on localhost:4248
  9. [83330] 2020/02/12 16:05:09.663308 [DBG] 127.0.0.1:4248 - rid:1 - Route connect msg sent
  10. [83330] 2020/02/12 16:05:09.663370 [INF] 127.0.0.1:4248 - rid:1 - Route connection created
  11. [83330] 2020/02/12 16:05:09.663537 [DBG] 127.0.0.1:4248 - rid:1 - Registering remote route "NDSGCS74MG5ZUMBOVWOUJ5S3HIOW"
  12. [83330] 2020/02/12 16:05:09.663549 [DBG] 127.0.0.1:4248 - rid:1 - Sent local subscriptions to route

From the seed’s server log, we see that the route is indeed accepted:

  1. [83329] 2020/02/12 16:05:09.663386 [INF] 127.0.0.1:62941 - rid:1 - Route connection created
  2. [83329] 2020/02/12 16:05:09.663665 [DBG] 127.0.0.1:62941 - rid:1 - Registering remote route "NAABC2CKRVPZBIECMLZZA6L3PK"
  3. [83329] 2020/02/12 16:05:09.663681 [DBG] 127.0.0.1:62941 - rid:1 - Sent local subscriptions to route

Finally, let’s start the third server:

  1. nats-server -p 6222 -cluster nats://localhost:6248 -routes nats://localhost:4248 -D

Again, notice that we use a different client port and cluster address, but still point to the same seed server at the address nats://localhost:4248:

  1. [83331] 2020/02/12 16:05:12.838022 [INF] Listening for client connections on 0.0.0.0:6222
  2. [83331] 2020/02/12 16:05:12.838029 [INF] Server id is NBE7SLUDLFIMHS2U6347N3DQEJ
  3. [83331] 2020/02/12 16:05:12.838031 [INF] Server is ready
  4. ...
  5. [83331] 2020/02/12 16:05:12.839203 [INF] Listening for route connections on localhost:6248
  6. [83331] 2020/02/12 16:05:12.839453 [DBG] Trying to connect to route on localhost:4248
  7. [83331] 2020/02/12 16:05:12.840112 [DBG] 127.0.0.1:4248 - rid:1 - Route connect msg sent
  8. [83331] 2020/02/12 16:05:12.840198 [INF] 127.0.0.1:4248 - rid:1 - Route connection created
  9. [83331] 2020/02/12 16:05:12.840324 [DBG] 127.0.0.1:4248 - rid:1 - Registering remote route "NDSGCS74MG5ZUMBOVWOUJ5S3HIOW"
  10. [83331] 2020/02/12 16:05:12.840342 [DBG] 127.0.0.1:4248 - rid:1 - Sent local subscriptions to route
  11. [83331] 2020/02/12 16:05:12.840717 [INF] 127.0.0.1:62946 - rid:2 - Route connection created
  12. [83331] 2020/02/12 16:05:12.840906 [DBG] 127.0.0.1:62946 - rid:2 - Registering remote route "NAABC2CKRVPZBIECMLZZA6L3PK"
  13. [83331] 2020/02/12 16:05:12.840915 [DBG] 127.0.0.1:62946 - rid:2 - Sent local subscriptions to route

First a route is created to the seed server (...IOW) and after that, a route from ...3PK - which is the ID of the second server - is accepted.

The log from the seed server shows that it accepted the route from the third server:

  1. [83329] 2020/02/12 16:05:12.840111 [INF] 127.0.0.1:62945 - rid:2 - Route connection created
  2. [83329] 2020/02/12 16:05:12.840350 [DBG] 127.0.0.1:62945 - rid:2 - Registering remote route "NBE7SLUDLFIMHS2U6347N3DQEJ"
  3. [83329] 2020/02/12 16:05:12.840363 [DBG] 127.0.0.1:62945 - rid:2 - Sent local subscriptions to route

And the log from the second server shows that it connected to the third.

  1. [83330] 2020/02/12 16:05:12.840529 [DBG] Trying to connect to route on 127.0.0.1:6248
  2. [83330] 2020/02/12 16:05:12.840684 [DBG] 127.0.0.1:6248 - rid:2 - Route connect msg sent
  3. [83330] 2020/02/12 16:05:12.840695 [INF] 127.0.0.1:6248 - rid:2 - Route connection created
  4. [83330] 2020/02/12 16:05:12.840814 [DBG] 127.0.0.1:6248 - rid:2 - Registering remote route "NBE7SLUDLFIMHS2U6347N3DQEJ"
  5. [83330] 2020/02/12 16:05:12.840827 [DBG] 127.0.0.1:6248 - rid:2 - Sent local subscriptions to route

At this point, there is a full mesh cluster of NATS servers.

Testing the Cluster

Now, the following should work: make a subscription to the first server (port 4222). Then publish to each server (ports 4222, 5222, 6222). You should be able to receive messages without problems.

  1. nats-sub -s "nats://127.0.0.1:4222" hello &
  2. nats-pub -s "nats://127.0.0.1:4222" hello world_4222
  3. [#1] Received on [hello] : 'world_4222'
  4. nats-pub -s "nats://127.0.0.1:5222" hello world_5222
  5. [#2] Received on [hello] : 'world_5222'
  6. nats-pub -s "nats://127.0.0.1:6222" hello world_6222
  7. [#3] Received on [hello] : 'world_6222'
  8. nats-pub -s "nats://127.0.0.1:4222,nats://127.0.0.1:5222,nats://127.0.0.1:6222" hello whole_world
  9. [#4] Received on [hello] : 'whole_world'
  10. # A random server was picked: NATS server logs for the second server, port 5222
  11. [83330] 2020/02/12 16:22:56.384754 [DBG] 127.0.0.1:63210 - cid:9 - Client connection created
  12. [83330] 2020/02/12 16:22:56.386467 [DBG] 127.0.0.1:63210 - cid:9 - Client connection closed