This post describes steps to set up expendable full stack denvironment. What's a denvironment, you may ask? It's development environment. That is just tooooo long to say and write:)
Take time and prepare your dev machine if you want to play along right away.
Description of the project
This project with made-up name "World's largest bass players database" consists of:
- ReactJS frontend
- SailsJS JSON API
- MongoDB for database
- RabbitMQ for queue and async processing
- Redis for cache
- Nginx for reverse proxy that fronts the API.
Let's call it "players", for short.
Let this project have it's main git repository be at https://github.com/svenvarkel/players
(it's time to create yours, now).
Pre-requisites
-
Create 2 names in your /etc/hosts file.
# /etc/hosts 127.0.0.1 api.players.local #for the API 127.0.0.1 app.players.local #for the web APP
Install Docker Desktop
Get it from here and follow the instructions.
Directory layout
The directory layout reflects the stack. On top level there are all familiar names that help the developer to navigate to a component quickly and not waste time on searching for things in obscurely named subfolders or elsewhere. Also - each component is a real component, self-containing and complete. All output or config files or anything that a component would need are placed into the component's directory.
The folder of your development projects is the /.
So here is the layout:
/ /api /sails bits and pieces /.dockerignore /Dockerfile /mongodb /nginx /Dockerfile /conf.d/ /api.conf /app.conf /rabbitmq /redis /web /react bits and pieces /.dockerignore /Dockerfile /docker-compose.yml
It is all set up as an umbrella git repository with api and web as git submodules. Nginx, MongoDB, Redis and RabbitMQ don't need to have their own repositories.
From now on you have choice either to clone my demo repository or create your own.
If you decide to use my example repository, then run commands:
git clone git@github.com:svenvarkel/players.git cd players git submodule init git submodule update
Steps
First step - create docker-compose.yml
In docker-compose.yml you define your stack in full.
version: "3.7" services: rabbitmq: image: rabbitmq:3-management environment: RABBITMQ_DEFAULT_VHOST: "/players" RABBITMQ_DEFAULT_USER: "dev" RABBITMQ_DEFAULT_PASS: "dev" volumes: - type: volume source: rabbitmq target: /var/lib/rabbitmq/mnesia ports: - "5672:5672" - "15672:15672" networks: - local redis: image: redis:5.0.5 volumes: - type: volume source: redis target: /data ports: - "6379:6379" command: redis-server --appendonly yes networks: - local mongodb: image: mongo:4.2 ports: - "27017:27017" environment: MONGO_INITDB_DATABASE: "admin" MONGO_INITDB_ROOT_USERNAME: "root" MONGO_INITDB_ROOT_PASSWORD: "root" volumes: - type: bind source: ./mongodb/docker-entrypoint-initdb.d target: /docker-entrypoint-initdb.d - type: volume source: mongodb target: /data networks: - local api: build: ./api image: players-api:latest ports: - 1337:1337 - 9337:9337 environment: PORT: 1337 DEBUG_PORT: 9337 WAIT_HOSTS: rabbitmq:5672,mongodb:27017,redis:6379 NODE_ENV: development MONGODB_URL: mongodb://dev:dev@mongodb:27017/players?authSource=admin volumes: - type: bind source: ./api/api target: /var/app/current/api - type: bind source: ./api/config target: /var/app/current/config networks: - local depends_on: - "rabbitmq" - "mongodb" - "redis" web: build: ./web image: players-web:latest ports: - 3000:3000 environment: REACT_APP_API_URL: http://api.players.local volumes: - type: bind source: ./web/src target: /var/app/current/src - type: bind source: ./web/public target: /var/app/current/public networks: - local depends_on: - "api" nginx: build: nginx image: nginx-wait:latest restart: on-failure environment: WAIT_HOSTS: api:1337,web:3000 volumes: - type: bind source: ./nginx/conf.d target: /etc/nginx/conf.d - type: bind source: ./nginx/log target: /var/log/nginx ports: - 80:80 networks: - local depends_on: - "api" - "web" networks: local: driver: overlay volumes: rabbitmq: redis: mongodb:
A few comments about features and tricks used here.
My favorite docker trick that I learnt just a few days ago is the use of wait. You will see it in api and nginx Dockerfiles. It's a special app that let's the docker container wait for dependencies until a service actually comes available at a port. The Docker's own "depends_on" is good but it just waits until a dependence container becomes available, not when the actual service is started inside a container. For example - rabbitmq is quite slow to start and it may cause the API behave erratically if it starts up before rabbitmq or mongodb have been fully started.
The second trick you'll see in docker-compose.yml is the use of bind mounts. The code from the dev machine is mounted as a folder inside docker container. It's good for rapid development. Whenever the source code is changed in the editor on developer machine the SailsJS application (or actually - nodemon) in container can detect the changes and restart the application. More details about setting up SailsJS app will follow in future posts, I hope.
Second step - create API and add it as git submodule
sails new api --fast cd api git init git remote add origin <your api repo origin> git add . git push -u origin master
Then create Dockerfile for API project:
FROM node:10 ADD https://github.com/ufoscout/docker-compose-wait/releases/download/2.6.0/wait /wait RUN chmod +x /wait RUN mkdir -p /var/app/current # Copy application sources COPY . /var/app/current WORKDIR /var/app/current RUN npm i RUN chown -R node:node /var/app/current USER node # Set the workdir /var/app/current EXPOSE 1337 # Start the application CMD /wait && npm run start
Then move up and add it as your main project's submodule
cd .. git submodule add <your api repo origin> api
Third step - create web app and add it as git submodule
This step is almost a copy of step 2, but it's necessary.
npx create-react-app my-app cd web git init git remote add origin <your web repo origin> git add . git push -u origin master
Then create Dockerfile for WEB project:
FROM node:10 ADD https://github.com/ufoscout/docker-compose-wait/releases/download/2.6.0/wait /wait RUN chmod +x /wait RUN mkdir -p /var/app/current # Copy application sources COPY . /var/app/current WORKDIR /var/app/current RUN npm i RUN chown -R node:node /var/app/current USER node # Set the workdir /var/app/current EXPOSE 3000 # Start the application CMD /wait && npm run start
As you can see the Dockerfiles for api and web are almost identical. Only the port number is different.
Then move up and add it as your main project's submodule
cd .. git submodule add <your web repo origin> web
For both projects, api and web, it's also advisable to create .dockerignore file with just two lines:
node_modules package-lock.json
We want the npm modules inside the container being built fresh every time we rebuild the docker container.
It's time for our first smoke test!
Run docker-compose:
docker-compose up
After Docker grinding a while you should have a working stack! It doesn't do much yet but it's there.
Check with docker-compose:
$ docker-compose ps Name Command State Ports ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- players_api_1 docker-entrypoint.sh /bin/ ... Up 0.0.0.0:1337->1337/tcp, 0.0.0.0:9337->9337/tcp players_mongodb_1 docker-entrypoint.sh mongod Up 0.0.0.0:27017->27017/tcp players_nginx_1 /bin/sh -c /wait && exec n ... Up 0.0.0.0:80->80/tcp players_rabbitmq_1 docker-entrypoint.sh rabbi ... Up 0.0.0.0:15671->15671/tcp, 0.0.0.0:15672->15672/tcp, 0.0.0.0:25672->25672/tcp, 4369/tcp, 0.0.0.0:5671->5671/tcp, 0.0.0.0:5672->5672/tcp players_redis_1 docker-entrypoint.sh redis ... Up 0.0.0.0:6379->6379/tcp players_web_1 docker-entrypoint.sh /bin/ ... Up 0.0.0.0:3000->3000/tcp
As you can see you have:
- API running on port 1337 (9337 also exposed for debugging)
- MongoDB running on port 27017
- RabbitMQ running on many ports, where AMQP port 5672 is of our interest. 15672 is for management - check it out in your browser (use dev as username and password)!
- Redis running on port 6379
- Web app running on port 3000
- Nginx running on port 80.
Nginx proxies both API and web app. So now it's time to give it a look in your browser.
There it is!
And there is the ReactJS app.
With this post we won't go into depths of the applications but we focus rather on stack and integration.
So how can services access each other in this Docker setup, you may ask.
Right - it's very straightforward - the services can access each other on a common shared network by calling each other with exactly the same names that are defined in docker-compose.yml.
Redis is at "redis:6379", MongoDB is at "mongodb:27017" etc.
See docker-compose.yml for a tip on how to connect your SailsJS API to MongoDB.
A note about storage
You may have a question like "where is mongodb data stored?". There are 3 volumes defined in docker-compose.yml:
mongodb redis rabbitmq
These are special docker volumes that hold the data for each component. It's convenient way of storing data outside of application container but still under control and management of Docker.
A word of warning
There's something I learnt the hard way (not that hard, though) during my endeavour towards full stack dev env. I used command
docker-compose up
lightly and it created temptation to use command
docker-compose down
as lightly because "what goes up must come down", right? Not so fast! Beware that if you run docker-compose down it will destroy your stack including data volumes. So - be careful and better read docker-compose manuals first. Use docker-compose start, stop and restart.
Wrapping it up
More details could follow in similar posts in the future if there's interest for such guides. Shall I continue to add more examples on how to integrate RabbitMQ and Redis within such stack, perhaps? Let me know.
Conclusion
In this post there is a step by step guide on how to set up full stack SailsJS/ReactJS application denvironment (development environment) by using Docker. The denvironment consists of multiple components that are integrated with the API - database, cache and queue. User-facing applications are fronted by the Nginx reverse proxy.
Top comments (4)
Hi Sven, cool article! I cloned your repo and tried to run it and I got a permission error running git submodule update:
Also, I had to run "docker swarm init" before "docker-compose up", which throwed an error I guess because of the failed "git submodule update" from before:
I would appreciate if you could guide me please!
Thanks
Awesome article Sven!
Thanks a lot for your article.
Can you give me some instruction for production build
Specially because react create a static files.
Thanks
Hi, Nguyen. Thank you that you routed your question from GitHub to here :)
About production - yes, your React app would build static files, JS, HTML and CSS, right. So how I would do it?
Here's a guide that basically builds the react app and installs Nginx into the same container. So Nginx would serve the static files from the build results folder.
However - in my article I proposed a stack where Nginx is in a separate container and let it be that way, for fun :)
In this case I would probably create a new data volume called "common" and mount it to both containers - Nginx and web.
Add into web/Dockerfile:
In Nginx conf I'd set the document root to /var/app/current/public
I haven't exactly tested it in production but it could work, in theory at least. Worth to try?
Please mind - right now I don't have a ReactJS app handy, I'm playing with Svelte. So the build folders etc may differ. But I hope you get the idea.
If this doesn't work then try with shared folder and mount it with bind option.