A project that runs a Node server and a create-react-app app via two separate containers, using Docker Compose.
docker-compose up
For development, the server/
and client/
directories have their own docker containers, which are configured via the docker-compose.yml
file.
The client server is spun up at localhost:3000
and it proxies internally to the server using the linked name as server:8080
.
The local directories are mounted into the containers, so changes will reflect immediately. However, changes to package.json will likely need to a rebuild: docker-compose down && docker-compose build && docker-compose up
.
The node-sass
watch feature does not notice new files. In order to get new files working, restart the client container:
docker-compose restart client
All changes to node_modules
should happen inside the containers. Install any new dependencies by inside the container. You can do this via docker-compose run
, but it’s easier to just upadte a running container and avoid having to rebuild everything:
docker-compose exec client
Then inside:
npm install --save <new_dependency>
docker-compose -f docker-compose.prod.yml up
For production, this uses the Dockerfile at the root of the repo. It creates a static build of the client React app and runs Express inside server, which handles both the API and serving of React files.
As a result, different code is executing to serve the React files, but all of the API calls should remain the same. The difference between development and production isn’t ideal, but it does offer the simplicity of having the entire app run in one server on one machine.
This is one of multiple ways a Node + React app could be setup, as suggested here:
- Keep them together - have Express serve both the API and React files
- Split them apart - have Express API on one machine and the React files on another (e.g., on S3 and use CORS to access the API)
- Put the API behind a proxy - use something like NGINX to proxy the Express API server and React static files separately
This project uses the “keep them together” approach. For better performance, you can set up a proxy (like Cloudflare) in between your server and the Internet to cache the static files. Or with some extra work you can fashion it to do either of the other two options.
I have comp
aliased to docker-compose
on my computer.
Start via:
comp up
# or detached
comp up -d
Run a container of the server image via:
comp run server /bin/bash
Check status:
comp ps
Stop:
comp down
Run the production image:
comp -f docker-compose.prod.yml up
NOTE: if any dependencies change in package.json files, you probably will need to rebuild the container for the changes to appear, e.g.,
comp down
comp build
comp up
References for setting up a Node project with Docker and docker-compose:
- https://nodejs.org/en/docs/guides/nodejs-docker-webapp/
- https://blog.codeship.com/using-docker-compose-for-nodejs-development/
- http://jdlm.info/articles/2016/03/06/lessons-building-node-app-docker.html
Express + React:
- https://daveceddia.com/create-react-app-express-production/
- http://ericsowell.com/blog/2017/5/16/create-react-app-and-express
- https://medium.freecodecamp.org/how-to-make-create-react-app-work-with-a-node-backend-api-7c5c48acb1b0
- https://medium.freecodecamp.org/how-to-host-a-website-on-s3-without-getting-lost-in-the-sea-e2b82aa6cd38