mirror of
https://github.com/LCTT/TranslateProject.git
synced 2025-01-13 22:30:37 +08:00
253 lines
9.6 KiB
Markdown
253 lines
9.6 KiB
Markdown
|
Automated Docker-based Rails deployments
|
||
|
================================================================================
|
||
|
![](http://cocoahunter.com/content/images/2015/01/docker3.jpeg)
|
||
|
|
||
|
[TL;DR] This is the third post in a series of 3 on how my company moved its infrastructure from PaaS to Docker based deployment.
|
||
|
|
||
|
- [First part][1]: where I talk about the process we went thru before approaching Docker;
|
||
|
- [Second part][2]: where I explain how setting up a private registry for in house secure deployments.
|
||
|
|
||
|
----------
|
||
|
|
||
|
In this final part we will see how to automate the whole deployment process with a real world (though very basic) example.
|
||
|
|
||
|
### Basic Rails app ###
|
||
|
|
||
|
Let's dive into the topic right away and bootstrap a basic Rails app. For the purpose of this demonstration I'm going to use Ruby 2.2.0 and Rails 4.1.1
|
||
|
|
||
|
From the terminal run:
|
||
|
|
||
|
$ rvm use 2.2.0
|
||
|
$ rails new && cd docker-test
|
||
|
|
||
|
Let's create a basic controller:
|
||
|
|
||
|
$ rails g controller welcome index
|
||
|
|
||
|
...and edit `routes.rb` so that the root of the project will point to our newly created welcome#index method:
|
||
|
|
||
|
root 'welcome#index'
|
||
|
|
||
|
Running `rails s` from the terminal and browsing to [http://localhost:3000][3] should bring you to the index page. We're not going to make anything fancier to the app, it's just a basic example to prove that when we'll build and deploy the container everything is working.
|
||
|
|
||
|
### Setup the webserver ###
|
||
|
|
||
|
We are going to use Unicorn as our webserver. Add `gem 'unicorn'` and `gem 'foreman'` to the Gemfile and bundle it up (run `bundle install` from the command line).
|
||
|
|
||
|
Unicorn needs to be configured when the Rails app launches, so let's put a **unicorn.rb** file inside the **config** directory. [Here is an example][4] of a Unicorn configuration file. You can just copy & paste the content of the Gist.
|
||
|
|
||
|
Let's also add a Procfile with the following content inside the root of the project so that we will be able to start the app with foreman:
|
||
|
|
||
|
web: bundle exec unicorn -p $PORT -c ./config/unicorn.rb
|
||
|
|
||
|
If you now try to run the app with **foreman start** everything should work as expected and you should have a running app on [http://localhost:5000][5]
|
||
|
|
||
|
### Building a Docker image ###
|
||
|
|
||
|
Now let's build the image inside which our app is going to live. In the root of our Rails project, create a file named **Dockerfile** and paste in it the following:
|
||
|
|
||
|
# Base image with ruby 2.2.0
|
||
|
FROM ruby:2.2.0
|
||
|
|
||
|
# Install required libraries and dependencies
|
||
|
RUN apt-get update && apt-get install -qy nodejs postgresql-client sqlite3 --no-install-recommends && rm -rf /var/lib/apt/lists/*
|
||
|
|
||
|
# Set Rails version
|
||
|
ENV RAILS_VERSION 4.1.1
|
||
|
|
||
|
# Install Rails
|
||
|
RUN gem install rails --version "$RAILS_VERSION"
|
||
|
|
||
|
# Create directory from where the code will run
|
||
|
RUN mkdir -p /usr/src/app
|
||
|
WORKDIR /usr/src/app
|
||
|
|
||
|
# Make webserver reachable to the outside world
|
||
|
EXPOSE 3000
|
||
|
|
||
|
# Set ENV variables
|
||
|
ENV PORT=3000
|
||
|
|
||
|
# Start the web app
|
||
|
CMD ["foreman","start"]
|
||
|
|
||
|
# Install the necessary gems
|
||
|
ADD Gemfile /usr/src/app/Gemfile
|
||
|
ADD Gemfile.lock /usr/src/app/Gemfile.lock
|
||
|
RUN bundle install --without development test
|
||
|
|
||
|
# Add rails project (from same dir as Dockerfile) to project directory
|
||
|
ADD ./ /usr/src/app
|
||
|
|
||
|
# Run rake tasks
|
||
|
RUN RAILS_ENV=production rake db:create db:migrate
|
||
|
|
||
|
Using the provided Dockerfile, let's try and build an image with the following command[1][7]:
|
||
|
|
||
|
$ docker build -t localhost:5000/your_username/docker-test .
|
||
|
|
||
|
And again, if everything worked out correctly, the last line of the long log output should read something like:
|
||
|
|
||
|
Successfully built 82e48769506c
|
||
|
$ docker images
|
||
|
REPOSITORY TAG IMAGE ID CREATED VIRTUAL SIZE
|
||
|
localhost:5000/your_username/docker-test latest 82e48769506c About a minute ago 884.2 MB
|
||
|
|
||
|
Let's try and run the container!
|
||
|
|
||
|
$ docker run -d -p 3000:3000 --name docker-test localhost:5000/your_username/docker-test
|
||
|
|
||
|
You should be able to reach your Rails app running inside the Docker container at port 3000 of your boot2docker VM[2][8] (in my case [http://192.168.59.103:3000][6]).
|
||
|
|
||
|
### Automating with shell scripts ###
|
||
|
|
||
|
Since you should already know from the previous post3 how to push your newly created image to a private regisitry and deploy it on a server, let's skip this part and go straight to automating the process.
|
||
|
|
||
|
We are going to define 3 shell scripts and finally tie it all together with rake.
|
||
|
|
||
|
### Clean ###
|
||
|
|
||
|
Every time we build our image and deploy we are better off always clean everything. That means the following:
|
||
|
|
||
|
- stop (if running) and restart boot2docker;
|
||
|
- remove orphaned Docker images (images that are without tags and that are no longer used by your containers).
|
||
|
|
||
|
Put the following into a **clean.sh** file in the root of your project.
|
||
|
|
||
|
echo Restarting boot2docker...
|
||
|
boot2docker down
|
||
|
boot2docker up
|
||
|
|
||
|
echo Exporting Docker variables...
|
||
|
sleep 1
|
||
|
export DOCKER_HOST=tcp://192.168.59.103:2376
|
||
|
export DOCKER_CERT_PATH=/Users/user/.boot2docker/certs/boot2docker-vm
|
||
|
export DOCKER_TLS_VERIFY=1
|
||
|
|
||
|
sleep 1
|
||
|
echo Removing orphaned images without tags...
|
||
|
docker images | grep "<none>" | awk '{print $3}' | xargs docker rmi
|
||
|
|
||
|
Also make sure to make the script executable:
|
||
|
|
||
|
$ chmod +x clean.sh
|
||
|
|
||
|
### Build ###
|
||
|
|
||
|
The build process basically consists in reproducing what we just did before (docker build). Create a **build.sh** script at the root of your project with the following content:
|
||
|
|
||
|
docker build -t localhost:5000/your_username/docker-test .
|
||
|
|
||
|
Make the script executable.
|
||
|
|
||
|
### Deploy ###
|
||
|
|
||
|
Finally, create a **deploy.sh** script with this content:
|
||
|
|
||
|
# Open SSH connection from boot2docker to private registry
|
||
|
boot2docker ssh "ssh -o 'StrictHostKeyChecking no' -i /Users/username/.ssh/id_boot2docker -N -L 5000:localhost:5000 root@your-registry.com &" &
|
||
|
|
||
|
# Wait to make sure the SSH tunnel is open before pushing...
|
||
|
echo Waiting 5 seconds before pushing image.
|
||
|
|
||
|
echo 5...
|
||
|
sleep 1
|
||
|
echo 4...
|
||
|
sleep 1
|
||
|
echo 3...
|
||
|
sleep 1
|
||
|
echo 2...
|
||
|
sleep 1
|
||
|
echo 1...
|
||
|
sleep 1
|
||
|
|
||
|
# Push image onto remote registry / repo
|
||
|
echo Starting push!
|
||
|
docker push localhost:5000/username/docker-test
|
||
|
|
||
|
If you don't understand what's going on here, please make sure you've read thoroughfully [part 2][9] of this series of posts.
|
||
|
|
||
|
Make the script executable.
|
||
|
|
||
|
### Tying it all together with rake ###
|
||
|
|
||
|
Having 3 scripts would now require you to run them individually each time you decide to deploy your app:
|
||
|
|
||
|
1. clean
|
||
|
1. build
|
||
|
1. deploy / push
|
||
|
|
||
|
That wouldn't be much of an effort, if it weren't for the fact that developers are lazy! And lazy be it, then!
|
||
|
|
||
|
The final step to wrap things up, is tying the 3 parts together with rake.
|
||
|
|
||
|
To make things even simpler you can just append a bunch of lines of code to the end of the already present Rakefile in the root of your project. Open the Rakefile file - pun intended :) - and paste the following:
|
||
|
|
||
|
namespace :docker do
|
||
|
desc "Remove docker container"
|
||
|
task :clean do
|
||
|
sh './clean.sh'
|
||
|
end
|
||
|
|
||
|
desc "Build Docker image"
|
||
|
task :build => [:clean] do
|
||
|
sh './build.sh'
|
||
|
end
|
||
|
|
||
|
desc "Deploy Docker image"
|
||
|
task :deploy => [:build] do
|
||
|
sh './deploy.sh'
|
||
|
end
|
||
|
end
|
||
|
|
||
|
Even if you don't know rake syntax (which you should, because it's pretty awesome!), it's pretty obvious what we are doing. We have declared 3 tasks inside a namespace (docker).
|
||
|
|
||
|
This will create the following 3 tasks:
|
||
|
|
||
|
- rake docker:clean
|
||
|
- rake docker:build
|
||
|
- rake docker:deploy
|
||
|
|
||
|
Deploy is dependent on build, build is dependent on clean. So every time we run from the command line
|
||
|
|
||
|
$ rake docker:deploy
|
||
|
|
||
|
All the script will be executed in the required order.
|
||
|
|
||
|
### Test it ###
|
||
|
|
||
|
To see if everything is working, you just need to make a small change in the code of your app and run
|
||
|
|
||
|
$ rake docker:deploy
|
||
|
|
||
|
and see the magic happening. Once the image has been uploaded (and the first time it could take quite a while), you can ssh into your production server and pull (thru an SSH tunnel) the docker image onto the server and run. It's that easy!
|
||
|
|
||
|
Well, maybe it takes a while to get accustomed to how everything works, but once it does, it's almost (almost) as easy as deploying with Heroku.
|
||
|
|
||
|
P.S. As always, please let me have your ideas. I'm not sure this is the best, or the fastest, or the safest way of doing devops with Docker, but it certainly worked out for us.
|
||
|
|
||
|
- make sure to have **boot2docker** up and running.
|
||
|
- If you don't know your boot2docker VM address, just run `$ boot2docker ip`
|
||
|
- if you don't, you can read it [here][10]
|
||
|
|
||
|
--------------------------------------------------------------------------------
|
||
|
|
||
|
via: http://cocoahunter.com/2015/01/23/docker-3/
|
||
|
|
||
|
作者:[Michelangelo Chasseur][a]
|
||
|
译者:[译者ID](https://github.com/译者ID)
|
||
|
校对:[校对者ID](https://github.com/校对者ID)
|
||
|
|
||
|
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创翻译,[Linux中国](http://linux.cn/) 荣誉推出
|
||
|
|
||
|
[a]:http://cocoahunter.com/author/michelangelo/
|
||
|
[1]:http://cocoahunter.com/docker-1
|
||
|
[2]:http://cocoahunter.com/2015/01/23/docker-2/
|
||
|
[3]:http://localhost:3000/
|
||
|
[4]:https://gist.github.com/chasseurmic/0dad4d692ff499761b20
|
||
|
[5]:http://localhost:5000/
|
||
|
[6]:http://192.168.59.103:3000/
|
||
|
[7]:http://cocoahunter.com/2015/01/23/docker-3/#fn:1
|
||
|
[8]:http://cocoahunter.com/2015/01/23/docker-3/#fn:2
|
||
|
[9]:http://cocoahunter.com/2015/01/23/docker-2/
|
||
|
[10]:http://cocoahunter.com/2015/01/23/docker-2/
|