2014-04-15 20:53:12 -04:00
|
|
|
page_title: Link via an Ambassador Container
|
|
|
|
page_description: Using the Ambassador pattern to abstract (network) services
|
|
|
|
page_keywords: Examples, Usage, links, docker, documentation, examples, names, name, container naming
|
|
|
|
|
|
|
|
# Link via an Ambassador Container
|
|
|
|
|
|
|
|
## Introduction
|
|
|
|
|
|
|
|
Rather than hardcoding network links between a service consumer and
|
2014-05-14 13:22:49 -04:00
|
|
|
provider, Docker encourages service portability, for example instead of:
|
2014-04-15 20:53:12 -04:00
|
|
|
|
|
|
|
(consumer) --> (redis)
|
|
|
|
|
2014-05-14 13:22:49 -04:00
|
|
|
Requiring you to restart the `consumer` to attach it to a different
|
|
|
|
`redis` service, you can add ambassadors:
|
2014-04-15 20:53:12 -04:00
|
|
|
|
|
|
|
(consumer) --> (redis-ambassador) --> (redis)
|
|
|
|
|
2014-05-14 13:22:49 -04:00
|
|
|
Or
|
2014-04-15 20:53:12 -04:00
|
|
|
|
|
|
|
(consumer) --> (redis-ambassador) ---network---> (redis-ambassador) --> (redis)
|
|
|
|
|
2014-05-14 13:22:49 -04:00
|
|
|
When you need to rewire your consumer to talk to a different Redis
|
|
|
|
server, you can just restart the `redis-ambassador` container that the
|
|
|
|
consumer is connected to.
|
2014-04-15 20:53:12 -04:00
|
|
|
|
2014-05-14 13:22:49 -04:00
|
|
|
This pattern also allows you to transparently move the Redis server to a
|
2014-04-15 20:53:12 -04:00
|
|
|
different docker host from the consumer.
|
|
|
|
|
2014-05-14 13:22:49 -04:00
|
|
|
Using the `svendowideit/ambassador` container, the link wiring is
|
|
|
|
controlled entirely from the `docker run` parameters.
|
2014-04-15 20:53:12 -04:00
|
|
|
|
|
|
|
## Two host Example
|
|
|
|
|
2014-05-14 13:22:49 -04:00
|
|
|
Start actual Redis server on one Docker host
|
2014-04-15 20:53:12 -04:00
|
|
|
|
2014-05-21 09:35:22 -04:00
|
|
|
big-server $ docker run -d --name redis crosbymichael/redis
|
2014-04-15 20:53:12 -04:00
|
|
|
|
2014-05-14 13:22:49 -04:00
|
|
|
Then add an ambassador linked to the Redis server, mapping a port to the
|
2014-04-15 20:53:12 -04:00
|
|
|
outside world
|
|
|
|
|
2014-05-21 09:35:22 -04:00
|
|
|
big-server $ docker run -d --link redis:redis --name redis_ambassador -p 6379:6379 svendowideit/ambassador
|
2014-04-15 20:53:12 -04:00
|
|
|
|
|
|
|
On the other host, you can set up another ambassador setting environment
|
2014-05-14 13:22:49 -04:00
|
|
|
variables for each remote port we want to proxy to the `big-server`
|
2014-04-15 20:53:12 -04:00
|
|
|
|
2014-05-21 09:35:22 -04:00
|
|
|
client-server $ docker run -d --name redis_ambassador --expose 6379 -e REDIS_PORT_6379_TCP=tcp://192.168.1.52:6379 svendowideit/ambassador
|
2014-04-15 20:53:12 -04:00
|
|
|
|
2014-05-14 13:22:49 -04:00
|
|
|
Then on the `client-server` host, you can use a Redis client container
|
|
|
|
to talk to the remote Redis server, just by linking to the local Redis
|
|
|
|
ambassador.
|
2014-04-15 20:53:12 -04:00
|
|
|
|
2014-05-21 09:35:22 -04:00
|
|
|
client-server $ docker run -i -t --rm --link redis_ambassador:redis relateiq/redis-cli
|
2014-04-15 20:53:12 -04:00
|
|
|
redis 172.17.0.160:6379> ping
|
|
|
|
PONG
|
|
|
|
|
|
|
|
## How it works
|
|
|
|
|
2014-05-14 13:22:49 -04:00
|
|
|
The following example shows what the `svendowideit/ambassador` container
|
|
|
|
does automatically (with a tiny amount of `sed`)
|
2014-04-15 20:53:12 -04:00
|
|
|
|
2014-05-14 13:22:49 -04:00
|
|
|
On the Docker host (192.168.1.52) that Redis will run on:
|
2014-04-15 20:53:12 -04:00
|
|
|
|
|
|
|
# start actual redis server
|
2014-05-21 09:35:22 -04:00
|
|
|
$ docker run -d --name redis crosbymichael/redis
|
2014-04-15 20:53:12 -04:00
|
|
|
|
|
|
|
# get a redis-cli container for connection testing
|
|
|
|
$ docker pull relateiq/redis-cli
|
|
|
|
|
|
|
|
# test the redis server by talking to it directly
|
2014-05-21 09:35:22 -04:00
|
|
|
$ docker run -t -i --rm --link redis:redis relateiq/redis-cli
|
2014-04-15 20:53:12 -04:00
|
|
|
redis 172.17.0.136:6379> ping
|
|
|
|
PONG
|
|
|
|
^D
|
|
|
|
|
|
|
|
# add redis ambassador
|
2014-05-21 09:35:22 -04:00
|
|
|
$ docker run -t -i --link redis:redis --name redis_ambassador -p 6379:6379 busybox sh
|
2014-04-15 20:53:12 -04:00
|
|
|
|
2014-05-14 13:22:49 -04:00
|
|
|
In the `redis_ambassador` container, you can see the linked Redis
|
|
|
|
containers `env`:
|
2014-04-15 20:53:12 -04:00
|
|
|
|
|
|
|
$ env
|
|
|
|
REDIS_PORT=tcp://172.17.0.136:6379
|
|
|
|
REDIS_PORT_6379_TCP_ADDR=172.17.0.136
|
|
|
|
REDIS_NAME=/redis_ambassador/redis
|
|
|
|
HOSTNAME=19d7adf4705e
|
|
|
|
REDIS_PORT_6379_TCP_PORT=6379
|
|
|
|
HOME=/
|
|
|
|
REDIS_PORT_6379_TCP_PROTO=tcp
|
|
|
|
container=lxc
|
|
|
|
REDIS_PORT_6379_TCP=tcp://172.17.0.136:6379
|
|
|
|
TERM=xterm
|
|
|
|
PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin
|
|
|
|
PWD=/
|
|
|
|
|
2014-05-14 13:22:49 -04:00
|
|
|
This environment is used by the ambassador `socat` script to expose Redis
|
|
|
|
to the world (via the `-p 6379:6379` port mapping):
|
2014-04-15 20:53:12 -04:00
|
|
|
|
|
|
|
$ docker rm redis_ambassador
|
|
|
|
$ sudo ./contrib/mkimage-unittest.sh
|
2014-05-21 09:35:22 -04:00
|
|
|
$ docker run -t -i --link redis:redis --name redis_ambassador -p 6379:6379 docker-ut sh
|
2014-04-15 20:53:12 -04:00
|
|
|
|
|
|
|
$ socat TCP4-LISTEN:6379,fork,reuseaddr TCP4:172.17.0.136:6379
|
|
|
|
|
2014-05-14 13:22:49 -04:00
|
|
|
Now ping the Redis server via the ambassador:
|
2014-04-15 20:53:12 -04:00
|
|
|
|
2014-05-14 13:22:49 -04:00
|
|
|
Now go to a different server:
|
2014-04-15 20:53:12 -04:00
|
|
|
|
|
|
|
$ sudo ./contrib/mkimage-unittest.sh
|
2014-05-21 09:35:22 -04:00
|
|
|
$ docker run -t -i --expose 6379 --name redis_ambassador docker-ut sh
|
2014-04-15 20:53:12 -04:00
|
|
|
|
|
|
|
$ socat TCP4-LISTEN:6379,fork,reuseaddr TCP4:192.168.1.52:6379
|
|
|
|
|
2014-05-14 13:22:49 -04:00
|
|
|
And get the `redis-cli` image so we can talk over the ambassador bridge.
|
2014-04-15 20:53:12 -04:00
|
|
|
|
|
|
|
$ docker pull relateiq/redis-cli
|
2014-05-21 09:35:22 -04:00
|
|
|
$ docker run -i -t --rm --link redis_ambassador:redis relateiq/redis-cli
|
2014-04-15 20:53:12 -04:00
|
|
|
redis 172.17.0.160:6379> ping
|
|
|
|
PONG
|
|
|
|
|
|
|
|
## The svendowideit/ambassador Dockerfile
|
|
|
|
|
2014-05-14 13:22:49 -04:00
|
|
|
The `svendowideit/ambassador` image is a small `busybox` image with
|
|
|
|
`socat` built in. When you start the container, it uses a small `sed`
|
|
|
|
script to parse out the (possibly multiple) link environment variables
|
|
|
|
to set up the port forwarding. On the remote host, you need to set the
|
|
|
|
variable using the `-e` command line option.
|
|
|
|
|
|
|
|
--expose 1234 -e REDIS_PORT_1234_TCP=tcp://192.168.1.52:6379
|
2014-04-15 20:53:12 -04:00
|
|
|
|
2014-05-14 13:22:49 -04:00
|
|
|
Will forward the local `1234` port to the remote IP and port, in this
|
|
|
|
case `192.168.1.52:6379`.
|
2014-04-15 20:53:12 -04:00
|
|
|
|
|
|
|
#
|
|
|
|
#
|
|
|
|
# first you need to build the docker-ut image
|
|
|
|
# using ./contrib/mkimage-unittest.sh
|
|
|
|
# then
|
|
|
|
# docker build -t SvenDowideit/ambassador .
|
|
|
|
# docker tag SvenDowideit/ambassador ambassador
|
|
|
|
# then to run it (on the host that has the real backend on it)
|
2014-05-21 09:35:22 -04:00
|
|
|
# docker run -t -i --link redis:redis --name redis_ambassador -p 6379:6379 ambassador
|
2014-04-15 20:53:12 -04:00
|
|
|
# on the remote host, you can set up another ambassador
|
2014-05-21 09:35:22 -04:00
|
|
|
# docker run -t -i --name redis_ambassador --expose 6379 sh
|
2014-04-15 20:53:12 -04:00
|
|
|
|
|
|
|
FROM docker-ut
|
|
|
|
MAINTAINER SvenDowideit@home.org.au
|
|
|
|
|
|
|
|
|
2014-04-17 18:55:24 -04:00
|
|
|
CMD env | grep _TCP= | sed 's/.*_PORT_\([0-9]*\)_TCP=tcp:\/\/\(.*\):\(.*\)/socat TCP4-LISTEN:\1,fork,reuseaddr TCP4:\2:\3 \&/' | sh && top
|