- You can work in groups of 2 students.
- It is up to you if you want to fork this repo, or if you prefer to work in a private repo. However, you have to use exactly the same directory structure for the validation procedure to work.
- We expect that you will have more issues and questions than with other labs (because we have a left some questions open on purpose). Please ask your questions on Teams, so that everyone in the class can benefit from the discussion.
⚠️ You will have to send your GitHub URL, answer the questions and send the output log of thevalidate.sh
script, which prove that your project is working in this Google Form.
This lab has 4 objectives:
-
The first objective is to design and implement a simple application protocol on top of UDP. It will be very similar to the protocol presented during the lecture (where thermometers were publishing temperature events in a multicast group and where a station was listening for these events).
-
The second objective is to get familiar with several tools from the JavaScript ecosystem. You will implement two simple Node.js applications. You will also have to search for and use a couple of npm modules (i.e. third-party libraries).
-
The third objective is to continue practicing with Docker. You will have to create 2 Docker images (they will be very similar to the images presented in class). You will then have to run multiple containers based on these images.
-
Last but not least, the fourth objective is to work with a bit less upfront guidance, as compared with previous labs. This time, we do not provide a complete webcast to get you started, because we want you to search for information (this is a very important skill that we will increasingly train). Don't worry, we have prepared a fairly detailed list of tasks that will put you on the right track. If you feel a bit overwhelmed at the beginning, make sure to read this document carefully and to find answers to the questions asked in the tables. You will see that the whole thing will become more and more approachable.
In this lab, you will write 2 small NodeJS applications and package them in Docker images:
-
the first app, Musician, simulates someone who plays an instrument in an orchestra. When the app is started, it is assigned an instrument (piano, flute, etc.). As long as it is running, every second it will emit a sound (well... simulate the emission of a sound: we are talking about a communication protocol). Of course, the sound depends on the instrument.
-
the second app, Auditor, simulates someone who listens to the orchestra. This application has two responsibilities. Firstly, it must listen to Musicians and keep track of active musicians. A musician is active if it has played a sound during the last 5 seconds. Secondly, it must make this information available to you. Concretely, this means that it should implement a very simple TCP-based protocol.
The following table gives you the mapping between instruments and sounds. Please use exactly the same string values in your code, so that validation procedures can work.
Instrument | Sound |
---|---|
piano |
ti-ta-ti |
trumpet |
pouet |
flute |
trulu |
violin |
gzi-gzi |
drum |
boum-boum |
- The auditor should include a TCP server and accept connection requests on port 2205.
- After accepting a connection request, the auditor must send a JSON payload containing the list of active musicians, with the following format (it can be a single line, without indentation):
[
{
"uuid" : "aa7d8cb3-a15f-4f06-a0eb-b8feb6244a60",
"instrument" : "piano",
"activeSince" : "2016-04-27T05:20:50.731Z"
},
{
"uuid" : "06dbcbeb-c4c8-49ed-ac2a-cd8716cbf2d3",
"instrument" : "flute",
"activeSince" : "2016-04-27T05:39:03.211Z"
}
]
You should be able to start an Auditor container with the following command:
$ docker run -d -p 2205:2205 api/auditor
You should be able to connect to your Auditor container over TCP and see that there is no active musician.
$ telnet IP_ADDRESS_THAT_DEPENDS_ON_YOUR_SETUP 2205
[]
You should then be able to start a first Musician container with the following command:
$ docker run -d api/musician piano
After this, you should be able to verify two points. Firstly, if you connect to the TCP interface of your Auditor container, you should see that there is now one active musician (you should receive a JSON array with a single element). Secondly, you should be able to use tcpdump
to monitor the UDP datagrams generated by the Musician container.
You should then be able to kill the Musician container, wait 5 seconds and connect to the TCP interface of the Auditor container. You should see that there is now no active musician (empty array).
You should then be able to start several Musician containers with the following commands:
$ docker run -d api/musician piano
$ docker run -d api/musician flute
$ docker run -d api/musician flute
$ docker run -d api/musician drum
When you connect to the TCP interface of the Auditor, you should receive an array of musicians that corresponds to your commands. You should also use tcpdump
to monitor the UDP trafic in your system.
Reminder: answer the following questions here.
# | Topic |
---|---|
Question | In a JavaScript program, if we have an object, how can we serialize it in JSON? |
We can use the JSON.stringify() function. |
|
Question | What is npm? |
npm is the official package manager for the Node.js ecosystem. | |
Question | What is the npm install command and what is the purpose of the --save flag? |
npm install will install a specific package and any packages that it depends on. --save is a flag to specifiy that we want to save the package as a dependency in our package.json file. It's now the default behavior when we install a package so we can omit it. |
|
Question | How can we use the https://www.npmjs.com/ web site? |
We can search for a package name and look at the availables versions for example. We can also check the weekly downloads number to verify if it's a well known package or not. | |
Question | In JavaScript, how can we generate a UUID compliant with RFC4122? |
From Node.js 14.17.0, it's built-in: const { randomUUID } = require('crypto'); console.log(randomUUID()); |
|
Question | In Node.js, how can we execute a function on a periodic basis? |
We can use the setInterval() function. It takes two parameters. The first one is the function to execute and the second one is the interval in milliseconds. |
|
Question | In Node.js, how can we emit UDP datagrams? |
const s = dgram.createSocket('udp4'); s.send(message, start, end, port, address, callback); |
|
Question | In Node.js, how can we access the command line arguments? |
We can access it from process.argv . |
# | Topic |
---|---|
Question | How do we define and build our own Docker image? |
We can define an image with a Dockerfile . To build it, we can use the command Docker build -t api/name-of-the-image . in the directory of the Dockerfile. |
|
Question | How can we use the ENTRYPOINT statement in our Dockerfile? |
We can use it to pass arguments to the executable in the container directly from the initial docker run command. |
|
Question | After building our Docker image, how do we use it to run containers? |
docker run api/musician name-of-the-instrument |
|
Question | How do we get the list of all running containers? |
docker ps |
|
Question | How do we stop/kill one running container? |
To stop: docker stop container-id or container-name To kill: docker kill container-id or container-name |
|
Question | How can we check that our running containers are effectively sending UDP datagrams? |
We can run a container (in the same network as the running containers) executing tcpdump with a filter for UDP requests. We created a specific Dockerfile for this task in image-tcpdump . It also filters UDP request by port. It it recommended to run this container with the -it flag to have a better output. |
# | Topic |
---|---|
Question | With Node.js, how can we listen for UDP datagrams in a multicast group? |
We can use the socket.addMembership() function with the multicast address as a paramter when we are binding the socket. |
|
Question | How can we use the Map built-in object introduced in ECMAScript 6 to implement a dictionary? |
We can use the uuid of the musicians as key and store the entire data of the musicians as value. After that, it's easy to access the data of a musician by its uuid. We can also loop through the musicians by transforming Map.entries() to an Array when we need to retrieve all the musicians. | |
Question | How can we use the Moment.js npm module to help us with date manipulations and formatting? |
We don't really need date manipulation in this lab. We can just make a new Date JavaScript Object with the timestamp we have from the musician. In addition, Moment.js is deprecated and the authors of the library recommend to use Luxon. With Luxon, it's easy to manipulate and format Date, for example to a readable string for the final user. | |
Question | When and how do we get rid of inactive players? |
The inactive musicians will be removed only when the list is requested (by a TCP connection). We loop through the Map and remove all entries with a lastActive timestamp older than 5 seconds. | |
Question | How do I implement a simple TCP server in Node.js? |
We can use the net Node.js module. With this, we can create a new TCP server with the net.createServer() function. Finally, we just have to bind the server to a selected port and to listen to the connection event. The callback function will be called every time a new TCP connection is made. |
# | Topic |
---|---|
Question | How do we validate that the whole system works, once we have built our Docker image? |
We can run the validate.sh script file. We first have to make sure it has been successfully executed until the end and that we don't have any error. We can also check that all the tests have passed, it's easy to check that in the check.log file the script has created. |
Please be careful to adhere to the specifications in this document, and in particular
- the Docker image names
- the names of instruments and their sounds
- the TCP PORT number
Also, we have prepared two directories, where you should place your two Dockerfile
with their dependent files.
Have a look at the validate.sh
script located in the top-level directory. This script automates part of the validation process for your implementation (it will gradually be expanded with additional operations and assertions). As soon as you start creating your Docker images (i.e. creating your Dockerfiles), you should try to run it to see if your implementation is correct. When you submit your project in the Google Form, the script will be used for grading, together with other criteria.