To create a Swarm you require:

A minimum of 2 Docker nodes that can communicte on a network over:

Initalize the Swarm on the first manager node

A node with Docker installed is a single-enginge node and you need to confogure Docker to run in Swarm mode so it can act as a multi-engine node. Logon to the first Docker node that you want to be a manager.

Obviously the IP and join details in this example will be different per setup

$ docker swarm init --advertise-addr 192.168.0.13:2377 --listen-addr 192.168.0.13:2377
Swarm initialized: current node (uuukt57mhnr02p7q0wk8kw10l) is now a manager.

To add a worker to this swarm, run the following command:

    docker swarm join --token SWMTKN-1-0btobek7ebvbtk4c3ncs2laux2yqpz4roiufsl2cdwftt3dqat-a0uofotfgakzpixrt6ol6zm77 192.168.0.13:2377

To add a manager to this swarm, run 'docker swarm join-token manager' and follow the instructions.

Add additional manager nodes

On the original manager node you can run the following command to retireve the command and values that can be copy/pasted onto new manager nodes to join them.

$ docker swarm join-token manager
To add a manager to this swarm, run the following command:

    docker swarm join --token SWMTKN-1-0btobek7ebvbtk4c3ncs2laux2yqpz4roiufsl2cdwftt3dqat-7kz5u6t75ptrqpc3kygwh7ssi 192.168.0.13:2377

Now logon onto the additional manager nodes and paste the command.

You can check that your managers are registered by running the following command.

$ docker node ls
ID                            HOSTNAME   STATUS    AVAILABILITY   MANAGER STATUS   ENGINE VERSION
uuukt57mhnr02p7q0wk8kw10l *   node1      Ready     Active         Leader           20.10.17
iwft3pma29oo12nfddru1deid     node2      Ready     Active         Reachable        20.10.17
swhtvpxigheh744unqc09hj7z     node3      Ready     Active         Reachable        20.10.17

Add worker nodes

Repeat the same process on the worker nodes as the manager but you need the worker token instead. So on a manager node run the following command and copy the output to clipboard and paste it onto your worker nodes to join them.

$ docker swarm join-token worker
To add a worker to this swarm, run the following command:

    docker swarm join --token SWMTKN-1-0btobek7ebvbtk4c3ncs2laux2yqpz4roiufsl2cdwftt3dqat-a0uofotfgakzpixrt6ol6zm77 192.168.0.13:2377

On any of the manager nodes, check to make sure your worker nodes are joined. Any node with no data in the MANAGER STATUS column are worker nodes and the node with an * simply identifies to you which node you are logged into and running the command from.

$ docker node ls
ID                            HOSTNAME   STATUS    AVAILABILITY   MANAGER STATUS   ENGINE VERSION
uuukt57mhnr02p7q0wk8kw10l     node1      Ready     Active         Leader           20.10.17
iwft3pma29oo12nfddru1deid     node2      Ready     Active         Reachable        20.10.17
swhtvpxigheh744unqc09hj7z *   node3      Ready     Active         Reachable        20.10.17
7gr0zv9cw0dcwb6gg6mcxodxv     node4      Ready     Active                          20.10.17
blpcfyi7hole1dx792zq17pvt     node5      Ready     Active                          20.10.17

Create an overlay network

This command creats a new Docker overlay network called grinntec-net which is available to all Docker nodes in the same Swarm.

$ docker network create -d overlay grinntec-net
o8iaoig5den15ha3tvir17cpc

$ docker network ls
NETWORK ID     NAME              DRIVER    SCOPE
9ae4bca4962e   bridge            bridge    local
62dd78760eae   docker_gwbridge   bridge    local
o8iaoig5den1   grinntec-net      overlay   swarm
8c93d83c35df   host              host      local
adt8wii4ba2s   ingress           overlay   swarm
2c5a36e59aea   none              null      local

Deploy a container as a Service

Logged into one of the manager nodes run the following command to deploy an image to the worker nodes.

$ docker service create --name web-fe --network grinntec-net -p 8080:8080 --replicas 2 nginx
t9h19g5mhucy6v7evhmz344er
overall progress: 2 out of 2 tasks 
1/2: running   [==================================================>] 
2/2: running   [==================================================>] 
verify: Service converged

This command creates a new service with the name of web-fe and to work on the overlay network call grinntec-net, and told Docker to listen on port 8080 and translate that traffic to 8080, two replicas of the image are required and the image used in nginx.

The service is saved as a configuration of the desired state of this service. As the Swarm monitors the web-fe containers, if the observed state does not match the desired state then the Swarm will work to ensure the desired state is achieved. This is typiclly when a worker node is offine and the number of replcias as dropped below the required number as per the current desired state for this container web-fe.

You can see the observed state

$  docker service ls
ID             NAME      MODE         REPLICAS   IMAGE          PORTS
t9h19g5mhucy   web-fe    replicated   2/2        nginx:latest   *:8080->8080/tcp

Or a more detailed view using

$ docker service ps web-fe
ID             NAME       IMAGE          NODE      DESIRED STATE   CURRENT STATE           ERROR     PORTS
j22zc3m5pzpz   web-fe.1   nginx:latest   node4     Running         Running 8 minutes ago             
1t2ayyc4v12b   web-fe.2   nginx:latest   node3     Running         Running 8 minutes ago

Scale the service up

We’re seeing a lot of traffic, let’s scale up from 2 to 10 instances.

$ docker service scale web-fe=10
web-fe scaled to 10
overall progress: 10 out of 10 tasks 
1/10: running   [==================================================>] 
2/10: running   [==================================================>] 
3/10: running   [==================================================>] 
4/10: running   [==================================================>] 
5/10: running   [==================================================>] 
6/10: running   [==================================================>] 
7/10: running   [==================================================>] 
8/10: running   [==================================================>] 
9/10: running   [==================================================>] 
10/10: running   [==================================================>] 
verify: Service converged 

$ docker service ps web-fe
ID             NAME        IMAGE          NODE      DESIRED STATE   CURRENT STATE                ERROR     PORTS
2u7oj8ere0jh   web-fe.1    nginx:latest   node2     Running         Running 3 minutes ago                  
450ic1m7ij0f   web-fe.2    nginx:latest   node1     Running         Running 3 minutes ago                  
iaui9yyp9sar   web-fe.3    nginx:latest   node1     Running         Running about a minute ago             
m8htwrriw7zh   web-fe.4    nginx:latest   node3     Running         Running about a minute ago             
lbl27juxx05t   web-fe.5    nginx:latest   node4     Running         Running about a minute ago             
64sfczr1jr63   web-fe.6    nginx:latest   node3     Running         Running about a minute ago             
onogf97lkbpt   web-fe.7    nginx:latest   node4     Running         Running about a minute ago             
ihqio6csb2et   web-fe.8    nginx:latest   node5     Running         Running about a minute ago             
h4dx06cn1ewx   web-fe.9    nginx:latest   node5     Running         Running about a minute ago             
6sp9aux50ykg   web-fe.10   nginx:latest   node2     Running         Running about a minute ago

Update the Service with a new version

A new version of the image has been released so we need to push it out to all 10 instances of the Service without downtime. The followwing command will update the web-fe service to use the latest version of the nginx image, with a rolling update configuration that updates 2 tasks at a time and waits for 20 seconds between each update. This rolling update strategy helps to minimize downtime during the update process.

$ docker service update --image nginx:latest --update-parallelism 2 --update-delay 20s web-fe
web-fe
overall progress: 10 out of 10 tasks 
1/10: running   [==================================================>] 
2/10: running   [==================================================>] 
3/10: running   [==================================================>] 
4/10: running   [==================================================>] 
5/10: running   [==================================================>] 
6/10: running   [==================================================>] 
7/10: running   [==================================================>] 
8/10: running   [==================================================>] 
9/10: running   [==================================================>] 
10/10: running   [==================================================>] 
verify: Service converged 

State of the Service during the update. Some are shutdown whilst upgrading. But you have multiples running to allow for the desired state.

$ docker service ps web-fe
ID             NAME            IMAGE          NODE      DESIRED STATE   CURRENT STATE                 ERROR                              PORTS
4qrjsjqzdgiu   web-fe.1        nginx:stable   node2     Running         Running 1 second ago                                             
2u7oj8ere0jh    \_ web-fe.1    nginx:latest   node2     Shutdown        Shutdown 3 seconds ago                                           
5mzx1icqp7x7   web-fe.2        nginx:stable   node1     Running         Running 53 seconds ago                                           
450ic1m7ij0f    \_ web-fe.2    nginx:latest   node1     Shutdown        Shutdown 54 seconds ago                                          
d2pyk0zl77lu   web-fe.3        nginx:stable   node2     Running         Running 1 second ago                                             
1ggalbjr0g9e    \_ web-fe.3    nginx:latest   node2     Shutdown        Failed 3 seconds ago          "error while removing network:…"   
24ifzyhu96hd   web-fe.4        nginx:stable   node3     Running         Running about a minute ago                                       
7c6owd7kvcf3    \_ web-fe.4    nginx:latest   node1     Shutdown        Shutdown about a minute ago                                      
y0dxcybkmklq   web-fe.5        nginx:stable   node5     Running         Running 27 seconds ago                                           
pdujac0801kl    \_ web-fe.5    nginx:latest   node5     Shutdown        Shutdown 29 seconds ago                                          
8vv4hmm7iqfa   web-fe.6        nginx:stable   node3     Running         Running 27 seconds ago                                           
m6u7nu8qijkr    \_ web-fe.6    nginx:latest   node3     Shutdown        Shutdown 29 seconds ago                                          
z6op7153ufpj   web-fe.7        nginx:stable   node5     Running         Running 53 seconds ago                                           
fbcqknuqlcgn    \_ web-fe.7    nginx:latest   node5     Shutdown        Shutdown 54 seconds ago                                          
1gmx8zp4czde   web-fe.8        nginx:stable   node1     Running         Running about a minute ago                                       
aeh626vdw93r    \_ web-fe.8    nginx:latest   node3     Shutdown        Shutdown about a minute ago                                      
opi9vpc6lz8w   web-fe.9        nginx:stable   node4     Running         Running about a minute ago                                       
2pnqlhxpv3ov    \_ web-fe.9    nginx:latest   node4     Shutdown        Shutdown about a minute ago                                      
pp8ib1c5v0r9   web-fe.10       nginx:stable   node4     Running         Running about a minute ago                                       
1zx5hidgt9dj    \_ web-fe.10   nginx:latest   node4     Shutdown        Shutdown about a minute ago