QCon 演讲火热征集中,快来分享技术实践与洞见! 了解详情
写点什么

深入浅出 Docker(五):基于 Fig 搭建开发环境

  • 2014-10-23
  • 本文字数:5880 字

    阅读完需:约 19 分钟

【编者按】Docker 是 PaaS 供应商 dotCloud 开源的一个基于 LXC 的高级容器引擎,源代码托管在 GitHub 上, 基于 Go 语言开发并遵从 Apache 2.0 协议开源。Docker 提供了一种在安全、可重复的环境中自动部署软件的方式,它的出现拉开了基于云计算平台发布产品方式的变革序幕。为了更好的促进 Docker 在国内的发展以及传播,我们决定开设《深入浅出Docker 》专栏,邀请Docker 相关的布道师、开发人员、技术专家来讲述Docker 的各方面内容,让读者对Docker 有更深入的了解,并且能够积极投入到新技术的讨论和实践中。另外,欢迎加入InfoQ Docker 技术交流群交流Docker 的最佳实践,QQ 群号:124378115。

1. 概述

在搭建开发环境时,我们都希望搭建过程能够简单,并且一劳永逸,其他的同事可以复用已经搭建好的开发环境以节省开发时间。而在搭建开发环境时,我们经常会被复杂的配置以及重复的下载安装所困扰。在 Docker 技术未出现之前,我们可以使用 Pupet、Chef、Ansible 等配置管理工具把复杂的配置管理起来,这样的管理配置技术仍然是目前比较流行的方式之一。配置管理工具使用的都是自己的 DSL 语法定义,考虑到环境的复杂性,配置一套通用的开发环境需要针对各个系统定制,对于大部分开发环境这种维护成本仍然是很高的。Docker 技术出现之后,系统的依赖问题得到了彻底的解决,我们可以通过镜像的方式简化环境的安装。结合 Docker 的开发部署工具 Fig,我们可以使用 fig.yml 文件来定义所有的环境,一次定义,多处使用,简单而且高效。

1.1 打包的方式

Docker 本省并不能创建一个真实的虚拟机,它只是基于 Linux Kernel 把额外的系统文件做了封装,并利用 Linux Kernel 相关的技术如 Cgroup、Namespace 隔离用户应用。应用 Docker 技术,团队之间通过共享 Image 或者 Dockefile 来复用开发环境。为了简化写 Dockerfile 的方式 ,Fig 提供更加精简的 DSL 定义文件 fig.yml,可以让新成员快速搭建开发环境并将精力投入到开发过程中去,而不是研究如何正确安装并配置诸如 PostgreSQL 之类的数据库。目前,软件开发需要的环境 Image,大部分都可以在 Docker Hub 中搜索到,需要使用时直接下载就可以使用。

1.2 应用组合的方式

使用 Docker 之后,我们不再需要在本地机器安装所有的软件包。我们可以根据项目需要在 Docker Hub 上搜索相关软件的 Image,然后使用 Fig pull 从 Docker Hub 上直接下载并由 Fig 调用 Docker 的 Link 命令把 Image 关联起来,这样所有应用都可以直接调用指定端口的服务,比如 Mysql 的 3306 端口提供数据库服务。开发者并不需要对 Docker 有太多的了解,只需要掌握常用的几条 Fig 命令就可以随时调试自己的环境。

1.3 环境共享的方式

Fig 直接定义好了 Image,我们不需要过多的关心容器或者镜像。在分享环境时,只需要把对应的 Dockerfile 和 fig.yml 文件分享给同事,他们就可以在自己的机器上运行并搭建出需要的环境,且不用再担心环境依赖带来的意外调试烦恼。团队成员在 git clone 项目代码后,就可以如下图一样使用一条命令启动自己的开发环境:

2. Fig 安装指南

首先,我们需要安装 Docker Engine,官方针对主流的系统提供了对应的安装手册。这里,我的开发机系统是一台 MacBook Pro,所以我需要安装 boot2docker 来支持 Docker。

接下来,我们就可以安装对应系统版本的 Fig 运行文件。比如在 Mac OS 或者在 64 位的 Linux 上安装 Fig:

复制代码
$ curl -L https://github.com/docker/fig/releases/download/1.0.0/fig-`uname
-s`-`uname -m` > /usr/local/bin/fig; chmod +x /usr/local/bin/fig

当以上的安装方式不能成功的话,我们可以选择使用 Python Package 的安装方式:

复制代码
$ sudo pip install -U fig

最后,不管使用什么 Linux 系统,安装完 Fig 之后通过运行以下命令来确保环境的一致性。

复制代码
$ fig --version

如果一切顺利的话,恭喜你,Fig 安装成功了。下面请随我一起学习如何使用 Fig 配置开发环境。

3. Rails 开发环境配置详解

我们来配置一套最常用的 Rails+PostgreSQL 项目。

第一步,确保 Fig 已经正确的安装到主机系统,如果还没有,请参考上一章节的安装指南。

第二步,在项目根目录下存放一个名为 Dockerfile 文件:

复制代码
FROM ruby
RUN apt-get update -qq && apt-get install -y build-essential libpq-dev
RUN mkdir /myapp
WORKDIR /myapp
ADD Gemfile /myapp/Gemfile
RUN bundle install
ADD . /myapp

第三步,创建一个 Gemfile 文件用来定义 Rails 软件包。 内容如下所示:

复制代码
source 'https://rubygems.org'
gem 'rails', '4.0.2'

第四步,创建一个 fig.yml 文件,并使用以下配置文件来做最后的环境初始化。

复制代码
db:
image: postgres
ports:
- "5432"
web:
build: .
command: bundle exec rackup -p 3000
volumes:
- .:/myapp
ports:
- "3000:3000"
links:
- db

第五步,当前你目录下空空如也,使用如下命令可以生成一套 Rails 项目骨架:

复制代码
$ fig run web rails new . --force --database=postgresql --skip-bundle

当你跑完以上命令,你就会得到一个崭新的 Rails 项目。

复制代码
$ ls
Dockerfile Rakefile config fig.yml public vendor
Gemfile app config.ru lib test
README.rdoc bin db log tmp

编辑一下 Gemfile 文件,去掉 therubyracer 包的注释, 让 Rails 依赖的 Javascript 的运行时环境。

复制代码
gem 'therubyracer', platforms: :ruby

所有的文件编辑都做完之后,运行命令创建开发环境 image。

复制代码
$ fig build

运行完 build 命令后,我们就有了可以立即使用的 Image。这两个 Image 的名字分别是 web 和 db。为了让 db 能连上 web,我们通常还需要修改 database.yml 来支持数据库连接。

development: &default adapter: postgresql encoding: unicode database: postgres pool: 5 username: postgres password: host: db test: <<: *default database: myapp_test

好了,让我们运行一下:

复制代码
$ fig up

命令行将显示如下日志:

复制代码
Recreating figtest_db_1...
Creating figtest_web_1...
Attaching to figtest_db_1, figtest_web_1
db_1 | LOG: database system was shut down at 2014-10-01 23:53:11 UTC
db_1 | LOG: autovacuum launcher started
db_1 | LOG: database system is ready to accept connections
web_1 | [2014-10-01 23:53:16] INFO WEBrick 1.3.1
web_1 | [2014-10-01 23:53:16] INFO ruby 2.1.2 (2014-05-08) [x86_64-linux]
web_1 | [2014-10-01 23:53:16] INFO WEBrick::HTTPServer#start: pid=1 port=3000
db_1 | FATAL: database "myapp_development" does not exist
db_1 | FATAL: database "myapp_development" does not exist
web_1 | 192.168.59.3 - - [01/Oct/2014 23:53:40] "GET / HTTP/1.1" 500 13476 0.5112
web_1 | 192.168.59.3 - - [01/Oct/2014 23:53:40] "GET %2Ffavicon.ico HTTP/1.1" 200 - 0.0067

通过以上日志可以知道开发数据库还没有创建。这时我们可以开启另外一个 terminal,创建开发数据库:

复制代码
$ fig run web rake db:create

当以上所有步骤都成功运行后,就需要来验证开发环境的正确性了。通过访问 http://localhost:3000/ 我们应该可以看到熟悉的 Rails 欢迎页面:

4. Django 开发环境配置详解

接下来让我们使用 Fig 来配置一套运行 Django/PostgreSQL 的应用程序吧。

首先我们新建一个项目目录,并在目录里创建 3 个文件。第一个文件是 Docker 镜像的定义文件: Dockerfile,用来描述安装在 Docker 容器里软件依赖关系。文件如下:

复制代码
FROM python:2.7
ENV PYTHONUNBUFFERED 1
RUN mkdir /code
WORKDIR /code
ADD requirements.txt /code/
RUN pip install -r requirements.txt
ADD . /code/

以上文件描述这个 Image 将安装 requirements.txt 指定的 python 依赖包。

第二个文件是requirements.txt,它是 python 依赖包定义描述文件,内容如下:

复制代码
Django
psycopg2

最后,Fig 需要把所有的环境都连接起来运行。这个文件名为 fig.yml 。它描述项目需要的服务组件、指定镜像的版本、如何连接服务、什么卷可以被载入容器内部、什么端口可以暴露出来等。内容形如:

复制代码
db:
image: postgres
web:
build: .
command: python manage.py runserver 0.0.0.0:8000
volumes:
- .:/code
ports:
- "8000:8000"
links:
- db

到此处,我们就可以使用 fig run 来创建一个 Django 项目了:

复制代码
$ fig run web django-admin.py startproject figexample .

当你运行完之后,可以在当前目录下看到创建的新项目文件:

复制代码
$ ls
Dockerfile fig.yml figexample manage.py requirements.txt

接下来的事情就是创建数据库链接,修改 figexample/settings.py 的 DATABASES = …部分。

复制代码
DATABASES = {
'default': {
'ENGINE': 'django.db.backends.postgresql_psycopg2',
'NAME': 'postgres',
'USER': 'postgres',
'HOST': 'db',
'PORT': 5432,
}
}

这个配置信息是用来连接 postgres 镜像服务。

然后,运行命令 fig up 来启动容器。

复制代码
Recreating myapp_db_1...
Recreating myapp_web_1...
Attaching to myapp_db_1, myapp_web_1
myapp_db_1 |
myapp_db_1 | PostgreSQL stand-alone backend 9.1.11
myapp_db_1 | 2014-01-27 12:17:03 UTC LOG: database system is ready to accept connections
myapp_db_1 | 2014-01-27 12:17:03 UTC LOG: autovacuum launcher started
myapp_web_1 | Validating models...
myapp_web_1 |
myapp_web_1 | 0 errors found
myapp_web_1 | January 27, 2014 - 12:12:40
myapp_web_1 | Django version 1.6.1, using settings 'figexample.settings'
myapp_web_1 | Starting development server at http://0.0.0.0:8000/
myapp_web_1 | Quit the server with CONTROL-C.

这个时候,你可以开启浏览器,然后输入 localhost:8000 就可以访问这个 Django 应用。

注意,当你跑起来应用之后,就可以初始化数据库了。这里,请一定要保证 fig up 是在运行中,并另外开启一个命令行窗口执行一下命令:

复制代码
$ fig run web python manage.py syncdb

如果你反复使用了 fig up 之后,可以体会到它一次性会把 web 和 db 两个镜像一起启动,如果你 CONTROL-C 之后,数据库也会停止服务。你甚至可以 fig run web /bin/bash 的方式直接进入到容器里看看。

5. Wordpress 开发环境配置详解

Fig 一样可以应付 PHP 应用的开发需求。

首先,创建一个 Dockerfile 来支持生成开发用 Image。

复制代码
$ curl https://wordpress.org/latest.tar.gz | tar -xvzf -

以上这条命令创建名为 wordpress 的目录。进入到此目录,创建镜像文件 Dockerfile,内容如下:

复制代码
FROM stackbrew/ubuntu:13.10
RUN apt-get update && apt-get install php5 php5-mysql -y
ADD . /code

下一步创建 fig.yml,它将定义出 web 服务和 mysql db 服务。

复制代码
web:
build: .
command: php -S 0.0.0.0:8000 -t /code
ports:
- "8000:8000"
links:
- db
volumes:
- .:/code
db:
image: mysql
environment:
MYSQL_DATABASE: wordpress
MYSQL_ROOT_PASSWORD: wordpress

wordpress 有一个支持文件需要修改,它是 wp-config.php:

复制代码
<?php
define('DB_NAME', 'wordpress');
define('DB_USER', 'root');
define('DB_PASSWORD', 'wordpress');
define('DB_HOST', getenv("DB_1_PORT_3306_TCP_ADDR") . ":" . getenv("DB_1_PORT_3306_TCP_PORT"));
define('DB_CHARSET', 'utf8');
define('DB_COLLATE', '');
define('AUTH_KEY', 'put your unique phrase here');
define('SECURE_AUTH_KEY', 'put your unique phrase here');
define('LOGGED_IN_KEY', 'put your unique phrase here');
define('NONCE_KEY', 'put your unique phrase here');
define('AUTH_SALT', 'put your unique phrase here');
define('SECURE_AUTH_SALT', 'put your unique phrase here');
define('LOGGED_IN_SALT', 'put your unique phrase here');
define('NONCE_SALT', 'put your unique phrase here');
$table_prefix = 'wp_';
define('WPLANG', '');
define('WP_DEBUG', false);
if ( !defined('ABSPATH') )
define('ABSPATH', dirname(__FILE__) . '/');
require_once(ABSPATH . 'wp-settings.php');

以上三个文件都修改后,就可以在此目录下运行 fig up 来启动 wordpress。可以使用浏览器访问 localhost:8000 浏览 Wordpress 首页。

6. 结论

通过 Fig 来构建基于 Docker 的开发环境可以让我们的开发事半功倍。其实如果读者按照我的步骤一步一步搭建开发环境,仍然会有很多挑战。

首先,因为国内带宽的限制,从官方下载 Image 是一个长时间等待的过程。即使 Docker 公司已经使用了 CDN 服务,但在国内网络使用仍然很慢。为了能快速下载,我还常备一条 VPN 线路作为数据通道来下载 Image。目前这是最理想的方式,可以节省很多开发时间。

第二,fig up 并不能保证 Image 能一次成功。但不需要灰心。你可以 fig run web /bin/bash 直接到容器里面调试。我遇到过多次费解的问题都是直接在里面跑命令解决的。当你退出时,fig 会自动更新到相应的 Image 里。当你再次 fig up,它会使用新 Image 去创建这个运行容器。

第三,fig up 会创建两个以上的容器实例,在退出容器后再次启动 fig up,并不会重用之前退出的容器实例,而是新建容器实例。像 fig up 这类常用命令运行多次之后会导致过期的容器文件仍然存储在你的开发机器上并占用硬盘空间,并且 Fig 还没有提供对应的命令自动清理它们。目前可以解决的办法是直接使用 Docker rm/rmi 命令手工清除。

以上总结出的实战经验,仍然不能掩盖 Fig 特有的亮点:运用 Fig 可以定义统一运行步骤,让部署环境可以完全的隔离在一个独立的容器环境里,并且这个隔离环境可以在开发、测试、生产多个步骤中保持一致。当前 Fig 项目还很年轻,需要大家多参与项目的讨论,提出自己的问题才能让 Fig 更好使用,更多信息可以到这里查阅

7. 作者简介

肖德时, Red Hat Engineering Service/HSS 内部工具组 Team Leader。Nodejs 开源项目 nodejs-cantas Lead Developer。擅长企业内部工具的设计以及实现。开源课程 Rails Starter 的发起人。rubygem: lazy_high_charts 的 Maintainer。Twitter 账号:xds2000,邮箱: xiaods@gmail.com

8. 下期预告

Docker 引擎可以在多台主机上启动成千上百的实例,那么如何管理这些容器实例显然会成为一个挑战。所以下期我将给大家介绍一款非常出色的容器集群管理框架 Kubernetes,敬请期待!


感谢郭蕾对本文的审校和策划。

给InfoQ 中文站投稿或者参与内容翻译工作,请邮件至 editors@cn.infoq.com 。也欢迎大家通过新浪微博( @InfoQ )或者腾讯微博( @InfoQ )关注我们,并与我们的编辑和其他读者朋友交流。

2014-10-23 08:3218112

评论

发布
暂无评论
发现更多内容

一文带你了解数仓智能运维框架

华为云开发者联盟

运维 GaussDB(DWS) 智能运维框架 调度框架 任务调度器

百度手机助手存储资源优化实践

百度Geek说

后端 存储

王者荣耀商城异地多活设计

皓月

「架构实战营」

设计模式【13】-- 模板模式怎么弄?

秦怀杂货店

Java 设计模式 23种设计模式

一起玩转LiteOS组件:Pixman

华为云开发者联盟

LiteOS LiteOS组件 Pixman Pixman Demo LiteOS组件仓库

知识库的作用

爱吃小舅的鱼

关于如何构建 Go 代码的思考

宇宙之一粟

Go 语言 2月月更

Java异常处理:如何写出“正确”但被编译器认为有语法错误的程序

华为云开发者联盟

Java 代码 java异常处理 语法 Exception

Linux之vmstat命令

入门小站

Linux

在线ASCII流程图编辑器工具

入门小站

工具

产品经理角色理解

wood

300天创作

领域模型设计该如何落地到数据库设计?

蜜糖的代码注释

Java DDD 领域模型 2月月更

运维与微服务结合?深度解析微服务框架Tars整体解决方案

云智慧AIOps社区

DevOps 微服务 运维 云原生 TARS

Web Components系列(三) —— 创建 Custom Elements

编程三昧

前端 组件化 2月月更 WebContents

BIGO 使用 Flink 做 OLAP 分析及实时数仓的实践和优化

Apache Flink

大数据 flink 编程 后端 实时计算

拒绝做工具小子—编写Python漏洞验证脚本

网络安全学海

黑客 网络安全 信息安全 渗透测试 安全漏洞

固态存储行业领导者硅格半导体加入龙蜥社区,共同推动开源生态建设

OpenAnolis小助手

Linux 开源 社群运营

AI+Science:基于飞桨的AlphaFold2,带你入门蛋白质结构预测

百度大脑

一个人手写APP监控平台是什么体验 | 社区征文

一诺倾心

flutter android 性能优化 APM

react源码解析3.react源码架构

buchila11

React

架构训练营 week10 课程总结

红莲疾风

「架构实战营」

RTE2021 回顾丨HDR 技术产品实践与探索

声网

人工智能 HDR

【架构训练营-模块二】

默光

微信朋友圈 架构训练营5期

B站员工猝死,审核员之殇,谁该反省?谁该惭愧?技术层面解构内容安全审核系统(python3)

刘悦的技术博客

系统架构 内容审核 Python3 应用审核 构架

推荐 3 个 React 动画库

devpoint

React 网页动画 2月月更

mysql 面试总结

yuexin_tech

面试

java培训:SpringBoot技术的理解

@零度

JAVA开发 spring-boot

不会用SpringBoot连接Redis,那就赶紧看这篇

华为云开发者联盟

redis 开发 springboot Redis服务器

在线学习FTRL介绍及基于Flink实现在线学习流程|社区征文

张浩_house

机器学习 大数据 flink 新春征文

一文搞明白直播和点播的区别 | 社区征文

liuzhen007

音视频 新春征文 2月月更

AI象棋,谁与争锋

乌龟哥哥

AI 2月月更

深入浅出Docker(五):基于Fig搭建开发环境_语言 & 开发_肖德时_InfoQ精选文章