Hello dear coder, welcome to my tech articles series dedicated to Node.js and Docker. Hope you enjoy!
Problem:
You have never used Docker before and you are a JavaScript developer. You want to understand how to use containers together with Node. Lets’ go!
1. Setup
In this tutorial I will be using my favorite editor Visual Studio Code. You will also need Nodejs and Docker installed. In my case I’m using Windows, so I got Docker for Windows from here.
Next command creates a new folder with name test-node-app, then creates npm package file and opens folder in VS Code.
mkdir test-node-app && cd test-node-app && npm init -y && code .
Also please install express to create a server.
npm i express
2. Create basic Node app
Open package.json file and modify it like this:
{
"name": "test-node-app",
"version": "1.0.0",
"description": "",
"main": "server.js",
"scripts": {
"start": "node server.js",
"test": "echo \"Error: no test specified\" && exit 1"
},
"keywords": [],
"author": "",
"license": "ISC",
"dependencies": {
"express": "^4.17.1"
}
}
Now add new file with name server.js:
Super basic basic basic app is ready. Now let’s test it locally. Run this command inside app folder:
npm start
You should see following in the console and will be able to access the server on the link http://localhost:8080/
3. Prepare Docker file
To build our image we need a docker file. Create a file with name Dockerfile
touch Dockerfile
And paste this inside:
FROM node:8
# Create app directory
WORKDIR /usr/src/app
# Install app dependencies
# A wildcard is used to ensure both package.json AND package-lock.json are copied
COPY package*.json ./
RUN npm install
# Bundle app source
COPY . .
EXPOSE 8080
CMD [ "npm", "start" ]
This file tells docker to download Node.js of version 8, create directory for the app, copy there package.json files, run npm install for node_modules, tells which port this app wants to use and finally runs it with npm start.
Uhhh.. a lot yeah? 😮
We will also need a file called .dockerignore. Create and paste following inside:
node_modules
npm-debug.log
4. Build an Docker image
First of all, build our image with this command.
docker build -t test-node-app .
Wait until it finishes and now we can see new image created.
docker images
If you are using VS Code, you can install a very nice Docker extension. It adds a button to side panel which opens Docker explorer. Here you can see all available images and containers. From there you can run, stop and remove containers which is pretty useful.
If you want to remove an image run this:
docker rmi <image id>
5. Run Docker container (finally)
Time to run a container based on our image (our application in isolated environment). The important thing is that we need to tell docker how to map the port. Now we run our container and map 8080 port to 49165.
docker run -p 49165:8080 -d test-node-app
Which means we can now access the app by this url http://localhost:49165/
There is another very useful command to see all running containers:
docker ps
You can stop container by right click in VS Code Docker Extension or by this command:
docker stop <container id>
Now if you run docker ps
again you will not see your container running. If you want to see stopped containers you need to add —all
flag to the command like this:
docker ps -a
If you still have some questions you can easy refer to official documentation here.
🚀 If you read something interesting from that article, please like and follow me for more posts. Thank you dear coder! 😏
Top comments (21)
Great article 👏, I remember wanting to dockerize a NodeJS server before. I managed to do it, but I struggled with the mysql database.
Ummm, Interesting topic for a future article maybe? Just saying 🖐️😂
I plan to do the same but with MongoDb :) thank you for the feedback
What exactly are you having problems with, @Bassem?
I read about it online for a while first. Everyone was saying that it is better to have the database as a standalone image. So I got myself an Ubuntu MySQL dockerfile off the internet to create that DB image.
Now I have 2 images, one for the NodeJS server and another for MySql DB.
But, I can't connect them both. I even tried accessing the database using phpmyadmin from outside the docker container and failed.
In my docker-compose, I created a network, set both images on the same network. and added the links: - db:db to make the backend server depend on the database.
I know that the network is fine. because I have a third frontend server image and it can communicate with the backend successfully.
For the PHPMyAdmin bit, are you exposing a port on the DB image? If you put two images on a Docker network like that, they just talk to everything on that network. You have to explicitly expose a port to have that container talk to anything "local" not on that network.
For the Node -> DB bit, do you mind sharing your docker-compose.yaml and the connection string you're using in your Node container?
Yeah sure, No problem!
I think that I am exposing ports using the "3306:3306". and for the backend config, I have a .env file
Actually, I have no idea what the username/password for the DB should be. But that's not the issue anyway according to the logs.
You're correct that you're exposing port 3306.
Your ENV file looks good, but what about the line of code that's using those ENV variables?
I am using a package called Sequelize for connecting to the DB and here is how :
Running
docker-compose up
produces this error :Unhandled rejection SequelizeConnectionRefusedError: connect ECONNREFUSED 127.0.0.1:3306
Could it be that the DB is not yet up and running?
I mean, could be. Have you tried using Kitematic or
docker exec -it <container name> /bin/bash
?Also, what is your output for
docker ps -a
when you have your compose file spun up?Edit: I see a couple things that could be at fault here, but I want to see the answer to those two questions before I take a stab at it. ^
I'd consider updating your docker-compose.yaml to version 3.x and use the container_name item to force a name on your container.
Then, in your ENV file, use whatever you set that container_name to DB_HOST. I understand why you're trying to use localhost, but if all of these containers are on one Docker network, they'll want to refer to each other by their container names.
I feel like i need to have a better understanding of docker.
Definitely will try doing that and will update you.
We all start somewhere, my dude. :)
Hello man, I created next part of series, you should see it 🖐️
Will do, definitely 👍
I failed at the second hurdle. Some pre-requisite knowledge that you have not explained. For a topic that appeared to be aimed at new docker user and using Windows system with VS Code as I have, I did not expect to be stumped so soon. I have reviewed a lot of tutorials and I have found ever one of them to be missing vital information (some are outright wrong). The effort you have gone to here looked promising so I have signed up to this site just to see if you still monitor this old topic and might have a response.
Here is the issue:
In the opening paragraph you write:
"Next command creates a new folder with name test-node-app, then creates npm package file and opens folder in VS Code.
mkdir test-node-app && cd test-node-app && npm init -y && code ."
Where do you type that? the syntax is wrong for powershell or command prompt and WSL doesn't know about npm so there must be some other pre-requsite knowledge or something missing in the Setup section.
Using powershell and appropriate commands to achieve the same as the linux style command line intends I have been able to continue. Perhaps though I am misisng something really important. Do you assume for example that the reader has already installed a unix VM and that they are running from that shell?
Was there anything specific to VSCode in here?
Only Docker Extension and command line
Word, I was gonna say I love VSCode but nothing about the editor really does any Docker stuff for me besides the Quake terminal. I have the Docker extension but it doesn't really add anything for me.
Thank you so much for the article!