基本的Rails應(yīng)用程序
現(xiàn)在讓我們啟動(dòng)一個(gè)基本的Rails應(yīng)用。為了更好的展示,我使用Ruby 2.2.0和Rails 4.1.1
在終端運(yùn)行:
$ rvm use 2.2.0
$ rails new cd docker-test
創(chuàng)建一個(gè)基本的控制器:
$ rails g controller welcome index
……,然后編輯 routes.rb ,以便讓該項(xiàng)目的根指向我們新創(chuàng)建的welcome#index方法:
在終端運(yùn)行 rails s ,然后打開瀏覽器,登錄http://localhost:3000,你會(huì)進(jìn)入到索引界面當(dāng)中。我們不準(zhǔn)備給應(yīng)用加上多么神奇的東西,這只是一個(gè)基礎(chǔ)的實(shí)例,當(dāng)我們將要?jiǎng)?chuàng)建并部署容器的時(shí)候,用它來(lái)驗(yàn)證一切是否運(yùn)行正常。
安裝webserver
我們打算使用Unicorn當(dāng)做我們的webserver。在Gemfile中添加 gem 'unicorn'和 gem 'foreman'然后將它bundle起來(lái)(運(yùn)行 bundle install命令)。
啟動(dòng)Rails應(yīng)用時(shí),需要先配置好Unicorn,所以我們將一個(gè)unicorn.rb文件放在config目錄下。這里有一個(gè)Unicorn配置文件的例子,你可以直接復(fù)制粘貼Gist的內(nèi)容。
接下來(lái),在項(xiàng)目的根目錄下添加一個(gè)Procfile,以便可以使用foreman啟動(dòng)應(yīng)用,內(nèi)容為下:
復(fù)制代碼 代碼如下:
web: bundle exec unicorn -p $PORT -c ./config/unicorn.rb
現(xiàn)在運(yùn)行foreman start命令啟動(dòng)應(yīng)用,一切都將正常運(yùn)行,并且你將能夠在http://localhost:5000上看到一個(gè)正在運(yùn)行的應(yīng)用。
構(gòu)建一個(gè)Docker鏡像
現(xiàn)在我們構(gòu)建一個(gè)鏡像來(lái)運(yùn)行我們的應(yīng)用。在這個(gè)Rails項(xiàng)目的根目錄下,創(chuàng)建一個(gè)名為Dockerfile的文件,然后粘貼進(jìn)以下內(nèi)容:
復(fù)制代碼 代碼如下:
# 基于鏡像 ruby 2.2.0
FROM ruby:2.2.0
# 安裝所需的庫(kù)和依賴
RUN apt-get update apt-get install -qy nodejs postgresql-client sqlite3 --no-install-recommends rm -rf /var/lib/apt/lists/*
# 設(shè)置 Rails 版本
ENV RAILS_VERSION 4.1.1
# 安裝 Rails
RUN gem install rails --version "$RAILS_VERSION"
# 創(chuàng)建代碼所運(yùn)行的目錄
RUN mkdir -p /usr/src/app
WORKDIR /usr/src/app
# 使 webserver 可以在容器外面訪問
EXPOSE 3000
# 設(shè)置環(huán)境變量
ENV PORT=3000
# 啟動(dòng) web 應(yīng)用
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 項(xiàng)目(和 Dockerfile 同一個(gè)目錄)添加到項(xiàng)目目錄
ADD ./ /usr/src/app
# 運(yùn)行 rake 任務(wù)
RUN RAILS_ENV=production rake db:create db:migrate
使用上述Dockerfile,執(zhí)行下列命令創(chuàng)建一個(gè)鏡像(確保boot2docker已經(jīng)啟動(dòng)并在運(yùn)行當(dāng)中):
$ docker build -t localhost:5000/your_username/docker-test .
然后,如果一切正常,長(zhǎng)長(zhǎng)的日志輸出的最后一行應(yīng)該類似于:
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
讓我們運(yùn)行一下容器試試!
$ docker run -d -p 3000:3000 --name docker-test localhost:5000/your_username/docker-test
通過你的boot2docker虛擬機(jī)的3000號(hào)端口(我的是http://192.168.59.103:3000),你可以觀察你的Rails應(yīng)用。(如果不清楚你的boot2docker虛擬地址,輸入$ boot2docker ip命令查看。)
使用shell腳本進(jìn)行自動(dòng)化部署
前面的文章(指文章1和文章2)已經(jīng)告訴了你如何將新創(chuàng)建的鏡像推送到私有registry中,并將其部署在服務(wù)器上,所以我們跳過這一部分直接開始自動(dòng)化進(jìn)程。
我們將要定義3個(gè)shell腳本,然后最后使用rake將它們捆綁在一起。
清除
每當(dāng)我們創(chuàng)建鏡像的時(shí)候,
- 停止并重啟boot2docker;
- 去除Docker孤兒鏡像(那些沒有標(biāo)簽,并且不再被容器所使用的鏡像們)。
在你的工程根目錄下的clean.sh文件中輸入下列命令。
復(fù)制代碼 代碼如下:
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
給腳本加上執(zhí)行權(quán)限:
構(gòu)建
構(gòu)建的過程基本上和之前我們所做的(docker build)內(nèi)容相似。在工程的根目錄下創(chuàng)建一個(gè)build.sh腳本,填寫如下內(nèi)容:
復(fù)制代碼 代碼如下:
docker build -t localhost:5000/your_username/docker-test .
記得給腳本執(zhí)行權(quán)限。
部署
最后,創(chuàng)建一個(gè)deploy.sh腳本,在里面填進(jìn)如下內(nèi)容:
復(fù)制代碼 代碼如下:
# 打開 boot2docker 到私有注冊(cè)庫(kù)的 SSH 連接
boot2docker ssh "ssh -o 'StrictHostKeyChecking no' -i /Users/username/.ssh/id_boot2docker -N -L 5000:localhost:5000 root@your-registry.com "
# 在推送前先確認(rèn)該 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
如果你不理解這其中的含義,請(qǐng)先仔細(xì)閱讀這部分第二部分。
給腳本加上執(zhí)行權(quán)限。
使用rake將以上所有綁定
現(xiàn)在的情況是,每次你想要部署你的應(yīng)用時(shí),你都需要單獨(dú)運(yùn)行這三個(gè)腳本。
- clean
- build
- deploy / push
這一點(diǎn)都不費(fèi)工夫,可是事實(shí)上開發(fā)者比你想象的要懶得多!那么咱們就索性再懶一點(diǎn)!
我們最后再把工作好好整理一番,我們現(xiàn)在要將三個(gè)腳本通過rake捆綁在一起。
為了更簡(jiǎn)單一點(diǎn),你可以在工程根目錄下已經(jīng)存在的Rakefile中添加幾行代碼,打開Rakefile文件,把下列內(nèi)容粘貼進(jìn)去。
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的語(yǔ)法(其實(shí)你真應(yīng)該去了解一下,這玩意太酷了!),上面的內(nèi)容也是很顯然的吧。我們?cè)谝粋€(gè)命名空間(docker)里聲明了三個(gè)任務(wù)。
三個(gè)任務(wù)是:
- rake docker:clean
- rake docker:build
- rake docker:deploy
Deploy獨(dú)立于build,build獨(dú)立于clean。所以每次我們輸入命令運(yùn)行的時(shí)候。
所有的腳本都會(huì)按照順序執(zhí)行。
測(cè)試
現(xiàn)在我們來(lái)看看是否一切正常,你只需要在app的代碼里做一個(gè)小改動(dòng):
接下來(lái)就是見證奇跡的時(shí)刻了。一旦鏡像文件被上傳(第一次可能花費(fèi)較長(zhǎng)的時(shí)間),你就可以ssh登錄產(chǎn)品服務(wù)器,并且(通過SSH管道)把docker鏡像拉取到服務(wù)器并運(yùn)行了。多么簡(jiǎn)單!
也許你需要一段時(shí)間來(lái)習(xí)慣,但是一旦成功,它幾乎與用Heroku部署一樣簡(jiǎn)單。
備注:像往常一樣,請(qǐng)讓我了解到你的意見。我不敢保證這種方法是最好,最快,或者最安全的Docker開發(fā)的方法,但是這東西對(duì)我們確實(shí)奏效。
您可能感興趣的文章:- Maven一鍵部署Springboot到Docker倉(cāng)庫(kù)為自動(dòng)化做準(zhǔn)備(推薦)
- Linux+Docker+SpringBoot+IDEA一鍵自動(dòng)化部署的詳細(xì)步驟
- springboot docker jenkins 自動(dòng)化部署并上傳鏡像的步驟詳解
- 從零搭建docker+jenkins+node.js自動(dòng)化部署環(huán)境的方法
- docker利用selenium+testng實(shí)現(xiàn)web自動(dòng)化的方法
- .Net Core自動(dòng)化部署之利用docker版jenkins部署dotnetcore應(yīng)用的方法
- Centos7+Docker+Jenkins+ASP.NET Core 2.0自動(dòng)化發(fā)布與部署的實(shí)現(xiàn)
- Docker自動(dòng)化構(gòu)建Automated Build實(shí)現(xiàn)過程圖解