制作一个定制PHP环境Docker镜像
目标:采用Dockerfile,定制一个 PHP 基础镜像。基础镜像,通常为含最小功能的系统镜像,之后的应用镜像都以此为基础。
制作基础镜像
选择 Ubuntu 官方的 14.04 版本为我们依赖的系统镜像。
1 | FROM ubuntu:trusty |
因所有官方镜像均位于境外服务器,为了确保所有示例能正常运行,DaoCloud 提供了一套境内镜像源,并与官方源保持同步。如果使用 DaoCloud 的镜像源,则指向:
FROM daocloud.io/ubuntu:trusty
设置镜像的维护者,相当于镜像的作者或发行方。
1 | MAINTAINER Misael Yau <misael@qq.com> |
用 RUN 命令调用 apt-get 包管理器安装 PHP 环境所依赖的程序包。
安装依赖包相对比较固定,因此该动作应该尽量提前,这样做有助于提高镜像层的复用率。
1 | RUN apt-get update \ |
用 RUN 命令调用 Linux 命令对 Apache 服务和 PHP 参数进行配置。
1 | RUN echo "ServerName localhost" >> /etc/apache2/apache2.conf \ |
用 RUN 命令调用 mkdir 来准备一个干净的放置代码的目录。
1 | RUN mkdir -p /app && rm -rf /var/www/html && ln -s /app /var/www/html |
将本地的代码添加到目录,并指定其为当前的工作目录。
1 | COPY . /app |
设置启动脚本的权限,指定暴露的容器内端口地址。
最后指定容器启动的进程。
1 | RUN chmod 755 ./start.sh |
start.sh
1 |
|
至此一个 PHP 的基础镜像制作完毕,你可以在本地运行 docker build -t my-php-base .
来构建出这个镜像并命名为 my-php-base
。
由于网络环境的特殊情况,在本地运行
docker build
的时间会很长,并且有可能失败。推荐使用 DaoCloud 加速器 和 DaoCloud 的云端 代码构建 功能。
完整 Dockerfile
1 | # Ubuntu 14.04,Trusty Tahr(可靠的塔尔羊)发行版 |
开发一个PHP的Docker化应用
目标:基于 PHP 的 Docker 基础镜像,开发一个 Docker 化的示例 PHP 应用 。
基于官方镜像
本次基础镜像使用 PHP 官方镜像,也可以根据自己的项目需求与环境依赖使用 定制的 PHP 基础镜像。
因所有官方镜像均位于境外服务器,为了确保所有示例能正常运行,DaoCloud 提供了一套境内镜像源,并与官方源保持同步。
官方镜像维护了自 5.4 版本起的所有 PHP 基础镜像,所有镜像均采用 debian:jessie
作为系统镜像。
首先,选择官方的 php:5.6-cli
镜像作为项目的基础镜像。
1 | FROM daocloud.io/php:5.6-cli |
由于该示例代码较为简单,我们采用仅安装 PHP CLI 的 Docker 镜像来运行。
接着,将代码复制到目标目录。
1 | COPY . /app |
ADD
与 COPY
的区别,总体来说 ADD
和 COPY
都是添加文件的操作,其中 ADD
比 COPY
功能更多,ADD
允许后面的参数为 URL,还有 ADD
添加的文件为压缩包的话,它将自动解压。
CMD
为本次构建出来的镜像运行起来时候默认执行的命令,我们可以通过 docker run
的启动命令修改默认运行命令。
Dockerfile 具体语法请参考:Dockerfile。
有了 Dockerfile 以后,我们可以运行下面的命令构建 PHP 应用镜像并命名为 my-php-app
:
docker build -t my-php-app .
最后,让我们从镜像启动容器:
docker run my-php-app
1 | Welcome the world of Docker ! |
如果看到这段字符串,那么就说明你成功进入到了一个 Docker 化的世界。
欢迎来到 Docker 的世界,这个世界有你意想不到的精彩!
1 | FROM daocloud.io/php:5.6-cli |
开发一个PHP+MySQL Docker化应用
目标:基于典型的 LAMP 技术栈,用 Docker 镜像的方式搭建一个 Linux + Apache + MySQL + PHP 的应用 。
本项目代码维护在 DaoCloud/php-apache-mysql-sample 项目中。
创建 PHP 应用容器
因所有官方镜像均位于境外服务器,为了确保所有示例能正常运行,DaoCloud 提供了一套境内镜像源,并与官方源保持同步。
首先,选择官方的 PHP 镜像作为项目的基础镜像。
1 | FROM daocloud.io/php:5.6-apache |
接着,用官方 PHP 镜像内置命令docker-php-ext-install
安装 PHP 的 MySQL 扩展依赖。
1 | RUN docker-php-ext-install pdo_mysql |
- 依赖包通过
docker-php-ext-install
安装,如果依赖包需要配置参数则通过docker-php-ext-configure
命令。 - 安装
pdo_mysql
PHP 扩展。
然后,将代码复制到目标目录。
1 | COPY . /var/www/html/ |
因为基础镜像内已经声明了暴露端口和启动命令,此处可以省略。
至此,包含 PHP 应用的 Docker 容器已经准备好了。PHP 代码中访问数据库所需的参数,是通过读取环境变量的方式声明的。
1 | $serverName = env("MYSQL_PORT_3306_TCP_ADDR", "localhost"); |
这样做是因为在 Docker 化应用开发的最佳实践中,通常将有状态的数据类服务放在另一个容器内运行,并通过容器特有的 link
机制将应用容器与数据容器动态的连接在一起。
绑定 MySQL 数据容器(本地)
首先,需要创建一个 MySQL 容器。
1 | docker run --name some-mysql -e MYSQL_ROOT_PASSWORD=my-secret-pw -d daocloud.io/mysql:5.5 |
之后,通过 Docker 容器间的 link
机制,便可将 MySQL 的默认端口(3306)暴露给应用容器。
1 | docker run --name some-app --link some-mysql:mysql -d app-that-uses-mysql |
绑定 MySQL 数据服务(云端)
比起本地创建,在云端创建和绑定 MySQL 数据服务会更简单。
- 在 GitHub 上 Fork DaoCloud/php-apache-mysql-sample 或者添加自己的代码仓库。
- 注册成为 DaoCloud 用户。
- 在 DaoCloud 「控制台」中选择「代码构建」。
- 创建新项目,选择代码源,开始构建镜像。
- 在「服务集成」创建 MySQL 服务实例。
- 将构建的应用镜像关联 MySQL 服务实例并部署在云端。
DaoCloud 使用图文介绍
- 了解如何用 DaoCloud 进行代码构建:参考代码构建。
- 了解如何用 DaoCloud 进行持续集成:参考持续集成。
- 了解如何用为应用准备一个数据库服务:参考服务集成。
- 了解如何部署一个刚刚构建好的应用镜像:参考应用部署。
应用截图
Dockerfile
1 |
|
配置Docker化持续集成的PHP环境
目标:我们将为之前创建的 PHP+MySQL 应用,编写测试代码和创建持续集成环境。
利用 PHPUnit 编写单元测试(本地)
使用以下命令安装 PHPUnit 4.0:
1 | composer global require "phpunit/phpunit=~4.0" |
假设我们的工程包含两个文件,一个源代码文件 Cal.php
和一个测试代码文件 CalTest.php
。
1 |
|
1 |
|
使用以下命令来启动测试:
1 | phpunit CalTest |
利用 DaoCloud 配置持续集成环境(云端)
当我们写完测试代码之后,我们需要一个持续集成环境来自动执行测试,报告项目的健康状况。这里我们使用 DaoCloud 云端的持续集成能力来为我们的 PHP + MySQL 应用配置持续集成环境。
我们只需要在源代码的根目录放置 daocloud.yml
文件便可以接入 DaoCloud 持续集成系统,每一次源代码的变更都会触发一次 DaoCloud 持续集成。关于 daocloud.yml
的格式,请参考 这里。
以下是我们为 PHP + MySQL 应用编写的测试代码和 daocloud.yml
。
1 |
|
1 |
|
开发一个 PHP + NewRelic 的生产级 Docker 化应用
目标:我们将为之前创建的 PHP + MySQL 应用,配置由 NewRelic 提供的应用监控探针。
创建 PHP 应用容器
因所有官方镜像均位于境外服务器,为了确保所有示例能正常运行,DaoCloud 提供了一套境内镜像源,并与官方源保持同步。
首先,选择官方的 PHP 镜像作为项目的基础镜像。
1 | FROM daocloud.io/php:5.6-apache |
接着,按照 NewRelic 官方 PHP 安装教程,进行脚本的编写。
1 | ## 安装 NewRelic |
apt-get update
下载从仓库的软件包列表并获取软件包版本信息。apt-get install -y newrelic-php5
安装 NewRelic PHP5 扩展。- Docker 镜像采用分层数据存储格式,镜像的大小等于所有层次大小的总和,所以我们应该尽量精简每次构建所产生镜像的大小。
然后,修改 NewRelic 配置文件。
1 | # 覆盖 NewRelic 配置文件 |
- 主要将
newrelic.appname
和newrelic.license
按照 DaoCloud 最佳实践通过环境变量的方式暴露出来。
至此,我们 NewRelic 的配置全部完成了,将代码复制到指定目录,并执行构建镜像命令完成我们镜像构建的最后一步
1 | # /var/www/html/ 为 Apache 目录 |
docker build -t php-newrelic-image .
启动 php-newrelic 容器(本地)
最后,我们将构建好的镜像运行起来
1 | docker run \ |
- 使用
--name
参数,指定容器的名称。 - 使用
-e
参数,容器启动时会将环境变量注入到容器中。 - 使用
-d
参数,容器在启动时会进入后台运行。 php-newrelic-image
是由我们上面的Dockerfile
构建出来的镜像。
启动 php-newrelic 容器(云端)
比起本地创建,在云端创建会更简单。
- 在 GitHub 上 Fork DaoCloud/php-newrelic-sample 或者添加自己的代码仓库。
- 注册成为 DaoCloud 用户。
- 在 DaoCloud 「控制台」中选择「代码构建」。
- 创建新项目,选择代码源,开始构建镜像。
- 将构建的应用镜像部署在云端并指定
NEW_RELIC_APP_NAME
和NEW_RELIC_LICENSE_KEY
环境变量。
DaoCloud 使用图文介绍
NewRelic Dashboard 截图
开发一个 Laravel + MySQL 框架的 Docker 化应用
目标:基于主流的 PHP 框架,用 Docker 镜像的方式搭建一个 Laravel + MySQL 的应用。
创建 Laravel 应用容器
因所有官方镜像均位于境外服务器,为了确保所有示例能正常运行,DaoCloud 提供了一套境内镜像源,并与官方源保持同步。
首先,选择官方的 PHP 镜像作为项目的基础镜像。
1 | FROM daocloud.io/php:5.6-apache |
其次,通过安装脚本安装 Laravel 应用所需要的 PHP 依赖。
1 | # 安装 PHP 相关的依赖包,如需其他依赖包在此添加 |
- 依赖包通过
docker-php-ext-install
安装,如果依赖包需要配置参数则通过docker-php-ext-configure
命令。 - Docker 镜像采用分层数据存储格式,镜像的大小等于所有层次大小的总和,所以我们应该尽量精简每次构建所产生镜像的大小。
- Composer 为 Laravel 下载第三方 Vendor 包所必需的插件,Composer 同时也是 PHP 最流行的包管理工具。
接着,创建 Laravel 目录结构:
1 | # 开启 URL 重写模块 |
- Laravel 是通过统一的项目的入口文件进入应用系统的。进而需要 Apache 开启链接重写模块。
- Apache 默认的文档目录为
/var/www/html
,将/app
定义为 Laravel 应用目录,而 Laravel 的项目入口文件位于/app/public
。为了方便管理,把/var/www/html
软链接到/app/public
。
紧接着,根据 DaoCloud 的最佳实践,我们需要把第三方依赖预先加载好。
1 | # 预先加载 Composer 包依赖,优化 Docker 构建镜像的速度 |
- 复制
composer.json
和composer.lock
到/app
,composer.lock
会锁定 Composer 加载的依赖包版本号,防止由于第三方依赖包的版本不同导致的应用运行错误。 --no-autoloader
为了阻止composer install
之后进行的自动加载,防止由于代码不全导致的自动加载报错。--no-scripts
为了阻止composer install
运行时composer.json
所定义的脚本,同样是防止代码不全导致的加载错误问题。
然后,将 Laravel 应用程序复制到 /app
:
1 | # 复制代码到 App 目录 |
composer install
执行之前阻止的操作,完成自动加载及脚本运行- 修改
/app
与/app/storage
的权限,保证 Laravel 在运行中有足够的权限 - 因为基础镜像内已经声明了暴露端口和启动命令,此处可以省略。
至此,包含 Laravel 应用的 Docker 容器已经准备好了。Laravel 代码中访问数据库所需的参数,是通过读取环境变量的方式声明的。
config/database.php
修改变量为更贴近 Docker 的方式:
1 | 'host' => env('MYSQL_PORT_3306_TCP_ADDR', 'localhost'), |
这样做是因为在 Docker 化应用开发的最佳实践中,通常将有状态的数据类服务放在另一个容器内运行,并通过容器特有的 link
机制将应用容器与数据容器动态的连接在一起。
绑定 MySQL 数据容器(本地)
首先,需要创建一个 MySQL 容器。
1 | docker run --name some-mysql -e MYSQL_ROOT_PASSWORD=my-secret-pw -d daocloud.io/mysql:5.5 |
之后,通过 Docker 容器间的 link
机制,便可将 MySQL 的默认端口 (3306) 暴露给应用容器。
1 | docker run --name some-app --link some-mysql:mysql -d app-that-uses-mysql |
绑定 MySQL 数据服务(云端)
比起本地创建,在云端创建和绑定 MySQL 数据服务会更简单。
- 在 GitHub 上 Fork DaoCloud/php-laravel-mysql-sample 或者添加自己的代码仓库。
- 注册成为 DaoCloud 用户。
- 在 DaoCloud 「控制台」中选择「代码构建」。
- 创建新项目,选择代码源,开始构建镜像。
- 在「服务集成」创建 MySQL 服务实例。
- 将构建的应用镜像关联 MySQL 服务实例并部署在云端。
DaoCloud 使用图文介绍
- 了解如何用 DaoCloud 进行代码构建:参考代码构建。
- 了解如何用 DaoCloud 进行持续集成:参考持续集成。
- 了解如何用为应用准备一个数据库服务:参考服务集成。
- 了解如何部署一个刚刚构建好的应用镜像:参考应用部署。
php-laravel-mysql-sample 应用截图
参考 DaoCloud