I\'m using boot2docker on OS X and cloned the following repo:
https://github.com/enokd/docker-node-hello
It basically has a Dockerfile and a very simple express
There's several scenarios that docker can help on web development:
lighting fast provision all different kinds of services locally. you want a clean couchdb, run docker run -d -p 5984:5984 tutum/couchdb
; mysql, no problem: docker run -d -p 3306:3306 tutum/mysql
, a selenium server with firefox? easy: sudo docker run -p 4444:4444 -d lzhang/selenium
, you will get them immediately, and they can be destroyed by docker kill
the next seconds you don't want them. fit the local testing scenario very well, no need to worry how to configure them.
manage software dependencies. for example, you want to run your node app under different node versions (0.8, 0.10, 0.10.25, etc), just find the nodejs docker images by docker search nodejs
, and create containers by mounting your application directory as volume, you can run multiple containers which is isolated to each other. Take a look at my keystonejs-example project on how to run a complicated node app with mongodb in seconds with 0 configuration. thinking a more complicated model: load balancer + app + database + cache, in old school way, there're tons of configuration options to fit the setup procedure, but, if you provision them as separate containers and link them by names, every components can discover each other by local environment variables, just like a local PaaS.
an easy way to use application by searching docker index. for example, there is a neat tool called Heartbleed checker, you can quickly download/use it in a container with ready configuration by a single command, even don't need to think about config, download language runtime, configure, uninstall, etc.
Per boot2docker, I assume you're on OSX, so you'd better pickup version v0.9.2+ which supports hostonly network configuration, then you can access the containers run inside the VM over the hostonly network.
This is what I ended up doing, so far seems to work but I'm still digging into it:
# script located in bin/run
NS=mycompany
PROJECT=myproject
# kill and remove old container if it exists
docker kill $PROJECT
docker rm $PROJECT
# tag the previously built image
docker tag $NS/$PROJECT $NS/$PROJECT:old
# build the new image
docker build -t $NS/$PROJECT .
# remove the old image
docker rmi $NS/$PROJECT:old
docker run -dP --name=$PROJECT $NS/$PROJECT /sbin/my_init
In my project root, I simply run:
nodemon -x bin/run
Credit goes to this source.
Update for docker 1.3 and fig
Fig is great, it really took a lot of the complexity out of the script I had before. In addition, boot2docker now natively supports mounting volumes on Mac OS X using Virtual Box's shared folders. This is what I find works really well for me now:
First, the Dockerfile
:
FROM ubuntu:14.04
# Replace shell with bash so we can source files
RUN rm /bin/sh && ln -s /bin/bash /bin/sh
# Set debconf to run non-interactively
RUN echo 'debconf debconf/frontend select Noninteractive' | debconf-set-selections
# Install base dependencies
RUN apt-get update && apt-get install -y -q --no-install-recommends \
build-essential \
ca-certificates \
curl \
git \
libssl-dev \
python \
rsync \
software-properties-common \
wget \
&& rm -rf /var/lib/apt/lists/*
ENV NVM_DIR /usr/local/nvm
ENV NODE_VERSION 0.10.33
# Install nvm with node and npm
RUN curl https://raw.githubusercontent.com/creationix/nvm/v0.20.0/install.sh | bash \
&& source $NVM_DIR/nvm.sh \
&& nvm install $NODE_VERSION \
&& nvm alias default $NODE_VERSION \
&& nvm use default
ENV NODE_PATH $NVM_DIR/v$NODE_VERSION/lib/node_modules
ENV PATH $NVM_DIR/v$NODE_VERSION/bin:$PATH
CMD ["npm", "start"]
The fig.yml
:
app:
image: myNodeImage
working_dir: /home/myProject
volumes_from:
- myvols
Here's the new bin/run
:
#!/usr/bin/env bash
# This is the the bin/run script
docker run --rm --volumes-from myvols myNodeImage \
rsync \
--delete \
--recursive \
--safe-links \
--exclude .git --exclude node_modules \
/data/myProject/ /home/myProject
fig up
I also have a bin/install
script that does the node_modules
dependency installs. This assumes I've already done an npm install on my host so that any private packages will work. Also, this works great with npm links, you just need to make a symlink from your /home/linkedProject
into $NODE_PATH/linkedProject
in your container.
#!/usr/bin/env bash
# This is the the bin/install script
docker run --rm --volumes-from myvols myNodeImage \
rm -rf /home/myProject && \
rsync \
--delete \
--recursive \
--safe-links \
--exclude .git \
/data/myProject/ /home/myProject && \
cd /home/myProject && \
npm rebuild
So, to put this all together, here's the steps in order:
Create my data volume container:
docker run -v $HOME/data:/data:ro \
-v /home \
-v /path/to/NODE_PATH \
--name myvols myNodeImage echo Creating my volumes
Run my install script: cd ~/data/myProject && ./bin/install
Run my run script: nodemon -x bin/run
Concerning shared folders with VirtualBox, please check out:
boot2docker together with VirtualBox Guest Additions
How to mount /Users into boot2docker
https://medium.com/boot2docker-lightweight-linux-for-docker/boot2docker-together-with-virtualbox-guest-additions-da1e3ab2465c
tl;dr Build your own custom boot2docker.iso with VirtualBox Guest Additions (see link) or download http://static.dockerfiles.io/boot2docker-v1.1.0-virtualbox-guest-additions-v4.3.12.iso and save it to ~/.boot2docker/boot2docker.iso.
EDIT: updated link to boot2docker v1.1.0
Your question is really interesting, and (partially) not specifically related to Docker. Let me say that the main and final problem here is the use of a virtual machine.
Tracking code changes on host computer can be a challenge when you use a VM (bare Virtualbox VM, Vagrant VM, Docker boot2docker VM...)
The VM is a complete layer of abstraction between your Mac (where your code changes happen) and your application stack (here, the Docker container running on a tiny Linux VM).
You have to explicitly define a shared folder (NFS, ...) in Virtualbox, between your Mac and boot2docker. This shared folder would be the root of your code repository. That's the first step.
Mac: /Users/You/stuff/approot ===> boot2docker: /something
docker run -v /something:/app/path/inside/docker -p 49160:8080 \ -d gasi/centos-node-hello /usr/bin/node /app/path/inside/docker/index.js
Having NodeJs detecting files change is another challenge. You need an additional wrapper watching the filesystem and restarting Node upon code changes (Forever, Nodemon...).
And, after that since the changes don't occur on a local filesystem, but on a shared folder, you'll probably have to tell the Nodejs watcher (Forever/Nodemon/...) to use polling mode. It works (more or less) but will burn a lot of CPU.