El siguiente ejemplo es un cliente-servidor el cual corre en containers con la ayuda de docker-compose. El presente repositorio es un ejemplo práctico brindado por la cátedra para que los alumnos tengan un esqueleto básico de cómo armar un proyecto de cero en donde todas las dependencias del mismo se encuentren encapsuladas en containers.
El cliente (golang) y el servidor (python) fueron desarrollados en diferentes lenguajes simplemente para mostrar cómo dos lenguajes de programación pueden convivir en el mismo proyecto con la ayuda de containers.
El repositorio cuenta con un Makefile que posee encapsulado diferentes comandos utilizados recurrentemente en el proyecto en forma de targets. Los targets se ejecutan mediante la invocación de:
- make
El target principal a utilizar es docker-compose-up el cual permite inicializar el ambiente de desarrollo (buildear docker images del servidor y client, inicializar la red a utilizar por docker, etc.) y arrancar los containers de las aplicaciones que componen el proyecto.
Otros targets de utilizar son:
- docker-compose-down: Realiza un docker compose stop para detener los containers asociados al compose y luego realiza un docker compose down para destruir todos los recursos asociados al proyecto que fueron inicializados durante el docker-compose-up target. Se recomienda ejecutar este comando al finalizar cada ejecución para evitar que el disco de la máquina host se llene.
- docker-compose-logs: Permite ver los logs actuales del proyecto. Acompañar con grep para lograr ver mensajes de una aplicación específica dentro del compose
- docker-image: Buildea las imágenes a ser utilizadas tanto en el client como el server. Este target es utilizado por docker-compose-up, por lo cual se lo puede utilizar para testear nuevos cambios en las imágenes antes de arrancar el proyecto.
- build: Compila la aplicación cliente en la máquina en vez de docker. La compilación de esta forma es mucho más rápida pero requiere tener el entorno de golang instalado en la máquina.
El servidor del presente ejemplo es un Echo Server: los mensajes recibidos por el cliente son devueltos por el servidor. El servidor actual funciona de la siguiente forma:
- Servidor acepta una nueva conexión
- Servidor recibe mensaje del cliente y procede a responder el mismo
- Servidor desconecta al cliente
- Servidor procede a recibir una conexión nuevamente
Al ejecutar el comando make docker-compose-up para comenzar la ejecución del ejemplo y luego el comando make docker-compose-logs, se observan los siguientes logs:
efeyuk@Helena:~/Development/docker-compose-init$ make docker-compose-logs
docker-compose -f docker-compose-dev.yaml logs -f
Attaching to client1, server
server | 2020-04-10 23:10:54 INFO Proceed to accept new connections
server | 2020-04-10 23:10:55 INFO Got connection from ('172.24.125.3', 60392)
server | 2020-04-10 23:10:55 INFO Message received from connection ('172.24.125.3', 60392). Msg: b'[CLIENT 1] Message number 1 sent'
server | 2020-04-10 23:10:55 INFO Proceed to accept new connections
server | 2020-04-10 23:11:05 INFO Got connection from ('172.24.125.3', 60400)
server | 2020-04-10 23:11:05 INFO Message received from connection ('172.24.125.3', 60400). Msg: b'[CLIENT 1] Message number 2 sent'
server | 2020-04-10 23:11:05 INFO Proceed to accept new connections
server | 2020-04-10 23:11:15 INFO Got connection from ('172.24.125.3', 60406)
server | 2020-04-10 23:11:15 INFO Message received from connection ('172.24.125.3', 60406). Msg: b'[CLIENT 1] Message number 3 sent'
server | 2020-04-10 23:11:15 INFO Proceed to accept new connections
server | 2020-04-10 23:11:25 INFO Got connection from ('172.24.125.3', 60410)
server | 2020-04-10 23:11:25 INFO Message received from connection ('172.24.125.3', 60410). Msg: b'[CLIENT 1] Message number 4 sent'
server | 2020-04-10 23:11:25 INFO Proceed to accept new connections
client1 | time="2020-04-10T23:10:55Z" level=info msg="[CLIENT 1] Message from server: Your Message has been received: b'[CLIENT 1] Message number 1 sent'"
client1 | time="2020-04-10T23:11:05Z" level=info msg="[CLIENT 1] Message from server: Your Message has been received: b'[CLIENT 1] Message number 2 sent'"
server | 2020-04-10 23:11:35 INFO Got connection from ('172.24.125.3', 60412)
client1 | time="2020-04-10T23:11:15Z" level=info msg="[CLIENT 1] Message from server: Your Message has been received: b'[CLIENT 1] Message number 3 sent'"
server | 2020-04-10 23:11:35 INFO Message received from connection ('172.24.125.3', 60412). Msg: b'[CLIENT 1] Message number 5 sent'
client1 | time="2020-04-10T23:11:25Z" level=info msg="[CLIENT 1] Message from server: Your Message has been received: b'[CLIENT 1] Message number 4 sent'"
server | 2020-04-10 23:11:35 INFO Proceed to accept new connections
client1 | time="2020-04-10T23:11:35Z" level=info msg="[CLIENT 1] Message from server: Your Message has been received: b'[CLIENT 1] Message number 5 sent'"
server | 2020-04-10 23:11:45 INFO Got connection from ('172.24.125.3', 60418)
client1 | time="2020-04-10T23:11:45Z" level=info msg="[CLIENT 1] Message from server: Your Message has been received: b'[CLIENT 1] Message number 6 sent'"
server | 2020-04-10 23:11:45 INFO Message received from connection ('172.24.125.3', 60418). Msg: b'[CLIENT 1] Message number 6 sent'
server | 2020-04-10 23:11:45 INFO Proceed to accept new connections
server | 2020-04-10 23:11:55 INFO Got connection from ('172.24.125.3', 60420)
server | 2020-04-10 23:11:55 INFO Message received from connection ('172.24.125.3', 60420). Msg: b'[CLIENT 1] Message number 7 sent'
client1 | time="2020-04-10T23:11:55Z" level=info msg="[CLIENT 1] Message from server: Your Message has been received: b'[CLIENT 1] Message number 7 sent'"
server | 2020-04-10 23:11:55 INFO Proceed to accept new connections
client1 | time="2020-04-10T23:12:05Z" level=info msg="[CLIENT 1] Main loop finished"
client1 exited with code 0
La idea de los siguientes ejercicios prácticos consiste en que los alumnos se familiaricen con ambientes de desarrollo implementados en docker/docker-compose, poniendo énfasis en la automatización de cualquier parte del proyecto y la portabilidad del mismo (evitar que el usuario que emplea el proyecto tenga que instalar otra dependencia más que docker/docker-compose para correr las aplicaciones).
El alumno debe forkear el presente repositorio en su cuenta de Github personal y crear un branch por cada uno de los ejercicios que se encuentran a continuación con su correspondiente implementación.
Se recomienda que el alumno realize un fork privado del repositorio. Para realizar un fork privado, se debe duplicar el repositorio sobre un repositorio privado ya creado. El siguiente documento explica como realizar esto.
Modificar la definición del docker-compose para agregar un nuevo cliente al proyecto.
Definir un script (en el lenguaje deseado) que permita crear una definición de docker-compose con cantidad de clientes N.
Modificar el cliente y el servidor para lograr que la configuración de ambas aplicaciones sea leída tanto de variables de ambiente como de un archivo de configuración. La configuración a través de archivo debe ser injectada al ejemplo y persistida afuera del mismo. (Hint: docker volumes)
Crear un script que permita testear el correcto funcionamiento del servidor utilizando el comando netcat. Dado que el servidor es un EchoServer, se debe enviar un mensaje el servidor y esperar recibir el mismo mensaje enviado.
Netcat no debe ser instalado en la maquina y no se puede exponer puertos del servidor para realizar la comunicación. (Hint: docker network)
Modificar el servidor actual para que el mismo permita procesar mensajes y aceptar nuevas conexiones en paralelo.
El alumno puede elegir el lenguaje en el cual desarrollar el nuevo código del servidor. Si el alumno desea realizar las modificaciones en Python, tener en cuenta las limitaciones del lenguaje.