当前位置: 移动技术网 > IT编程>脚本编程>Ruby > 在Docker中自动化部署Ruby on Rails的教程

在Docker中自动化部署Ruby on Rails的教程

2017年12月08日  | 移动技术网IT编程  | 我要评论

基本的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的内容。

接下来,在项目的根目录下添加一个procfile,以便可以使用foreman启动应用,内容为下:

   

复制代码 代码如下:
web: bundle exec unicorn -p $port -c ./config/unicorn.rb

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

现在我们构建一个镜像来运行我们的应用。在这个rails项目的根目录下,创建一个名为dockerfile的文件,然后粘贴进以下内容: 

   

复制代码 代码如下:
# 基于镜像 ruby 2.2.0
    from ruby:2.2.0
    # 安装所需的库和依赖
    run apt-get update && apt-get install -qy nodejs postgresql-client sqlite3 --no-install-recommends && rm -rf /var/lib/apt/lists/*
    # 设置 rails 版本
    env rails_version 4.1.1
    # 安装 rails
    run gem install rails --version "$rails_version"
    # 创建代码所运行的目录
    run mkdir -p /usr/src/app 
    workdir /usr/src/app
    # 使 webserver 可以在容器外面访问
    expose 3000
    # 设置环境变量
    env port=3000
    # 启动 web 应用
    cmd ["foreman","start"]
    # 安装所需的 gems
    add gemfile /usr/src/app/gemfile 
    add gemfile.lock /usr/src/app/gemfile.lock 
    run bundle install --without development test
    # 将 rails 项目(和 dockerfile 同一个目录)添加到项目目录
    add ./ /usr/src/app
    # 运行 rake 任务
    run rails_env=production rake db:create db:migrate 

使用上述dockerfile,执行下列命令创建一个镜像(确保boot2docker已经启动并在运行当中):

   

 $ 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

通过你的boot2docker虚拟机的3000号端口(我的是http://192.168.59.103:3000),你可以观察你的rails应用。(如果不清楚你的boot2docker虚拟地址,输入$ boot2docker ip命令查看。)
使用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脚本,在里面填进如下内容:

复制代码 代码如下:
    # 打开 boot2docker 到私有注册库的 ssh 连接
    boot2docker ssh "ssh -o 'stricthostkeychecking no' -i /users/username/.ssh/id_boot2docker -n -l 5000:localhost:5000 root@your-registry.com &" &
    # 在推送前先确认该 ssh 通道是开放的。
    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 

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

给脚本加上执行权限。
使用rake将以上所有绑定

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

  1.     clean
  2.     build
  3.     deploy / push

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

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

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

  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)里声明了三个任务。

三个任务是:

  1.     rake docker:clean
  2.     rake docker:build
  3.     rake docker:deploy

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

 

  $ rake docker:deploy

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

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

  $ rake docker:deploy

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

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

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

如对本文有疑问, 点击进行留言回复!!

相关文章:

验证码:
移动技术网