TranslateProject/translated/tech/20150128 Docker-3 Automated Docker-based Rails deployments.md
2015-05-01 10:28:30 +08:00

9.5 KiB
Raw Blame History

自动化部署基于Docker的Rails应用

[TL;DR] 这是系列文章的第三篇讲述了我的公司是如何将基础设施从PaaS移植到Docker上的。

  • 第一部分:谈论了我接触Docker之前的经历
  • 第二部分:一步步搭建一个安全而又私有的registry。

在系列文章的最后一篇里,我们将用一个实例来学习如何自动化整个部署过程。

基本的Rails应用程序###

现在让我们启动一个基本的Rails应用。为了更好的展示我使用Ruby 2.2.0和Rails 4.1.1

在终端运行:

$ rvm use 2.2.0
$ rails new  && cd docker-test

创建一个基础控制器:

$ rails g controller welcome index

……然后编辑 routes.rb 以便让工程的根指向我们新创建的welcome#index方法这句话理解不太理解

root 'welcome#index'  

在终端运行 rails s ,然后打开浏览器,登录http://localhost:3000,你会进入到索引界面当中。我们不准备给应用加上多么神奇的东西,这只是一个基础实例,用来验证当我们将要创建并部署容器的时候,一切运行正常。

安装webserver

我们打算使用Unicorn当做我们的webserver。在Gemfile中添加 gem 'unicorn'gem 'foreman'然后将它bundle起来(运行 bundle install命令)。

在Rails应用启动的伺候需要配置Unicorn所以我们将一个unicorn.rb文件放在config目录下。这里有一个Unicorn配置文件的例子你可以直接复制粘贴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: 接下来在工程的根目录下添加一个Procfile以便可以使用foreman启动应用内容为下

web: bundle exec unicorn -p $PORT -c ./config/unicorn.rb  

现在运行foreman start命令启动应用,一切都将正常运行,并且你将能够在http://localhost:5000上看到一个正在运行的应用。

创建一个Docker映像

现在我们创建一个映像来运行我们的应用。在Rails工程的跟目录下创建一个名为Dockerfile的文件,然后粘贴进以下内容:

# 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  

使用提供的Dockerfile执行下列命令创建一个映像1:

$ docker build -t localhost:5000/your_username/docker-test .

然后,如果一切正常,长日志输出的最后一行应该类似于:

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  

来运行容器吧!

$ 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 VM2 (in my case http://192.168.59.103:3000). 通过你的boot2docker虚拟机2的3000号端口我的是http://192.168.59.103:3000你可以观察你的Rails应用。

使用shell脚本进行自动化部署

前面的文章指文章1和文章2已经告诉了你如何将新创建的映像推送到私有registry中并将其部署在服务器上所以我们跳过这一部分直接开始自动化进程。

我们将要定义3个shell脚本然后最后使用rake将它们捆绑在一起。

清除

每当我们创建映像的时候,

  • 停止并重启boot2docker
  • 去除Docker孤儿映像那些没有标签并且不再被容器所使用的映像们

在你的工程根目录下的clean.sh文件中输入下列命令。

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  

给脚本加上执行权限:

$ chmod +x clean.sh

创建

创建的过程基本上和之前我们所做的docker build内容相似。在工程的根目录下创建一个build.sh脚本,填写如下内容:

docker build -t localhost:5000/your_username/docker-test .  

给脚本执行权限。

部署

最后,创建一个deploy.sh脚本,在里面填进如下内容:

# 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  

如果你不理解这其中的含义,请先仔细阅读这部分part 2

给脚本加上执行权限。

使用rake将以上所有绑定

现在的情况是,每次你想要部署你的应用时,你都需要单独运行这三个脚本。

  1. clean
  2. build
  3. deploy / push

这一点都不费工夫,可是事实上开发者比你想象的要懒得多!那么咱们就索性再懒一点!

我们最后再把工作好好整理一番我们现在要将三个脚本捆绑在一起通过rake。

为了更简单一点你可以在工程根目录下已经存在的Rakefile中添加几行代码打开Rakefile文件——pun intended——把下列内容粘贴进去。

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  

即使你不清楚rake的语法其实你真应该去了解一下这玩意太酷了上面的内容也是很显然的吧。我们在一个命名空间docker里声明了三个任务。

三个任务是:

  • rake docker:clean
  • rake docker:build
  • rake docker:deploy

Deploy独立于buildbuild独立于clean。所以每次我们输入命令运行的时候。

$ rake docker:deploy

所有的脚本都会按照顺序执行。

测试

现在我们来看看是否一切正常你只需要在app的代码里做一个小改动

$ rake docker:deploy

接下来就是见证奇迹的时刻了。一旦映像文件被上传第一次可能花费较长的时间你就可以ssh登录产品服务器并且通过SSH管道把docker映像拉取到服务器并运行了。多么简单

也许你需要一段时间来习惯但是一旦成功它几乎与用Heroku部署一样简单。

备注像往常一样请让我了解到你的意见。我不敢保证这种方法是最好最快或者最安全的Docker开发的方法但是这东西对我们确实奏效。

  • 确保boot2docker已经启动并在运行当中。
  • 如果你不了解你的boot2docker虚拟地址输入 $ boot2docker ip命令查看。
  • 点击here教你怎样搭建私有的registry。

via: http://cocoahunter.com/2015/01/23/docker-3/

作者:Michelangelo Chasseur 译者:DongShuaike 校对:校对者ID

本文由 LCTT 原创翻译,Linux中国 荣誉推出