百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

Docker部署WordPress LNMP环境实践

nanshan 2024-10-09 12:51 33 浏览 0 评论

更多深度文章,请关注云计算频道:https://yq.aliyun.com/cloud

Docker基于LXC实现了把软件封装到一个完整的文件系统,可以在docker容器中运行所需的一切代码,运行环境,系统工具和系统库。由于docker使用独立于主机的文件系统,可以确保软件在不同的主机环境中仍然保持运行环境不变。docker与主机共用一个操作系统内核,使用docker容器具有轻量级的特点,能占用更少的内存快速启动容器。

下面我们学习使用docker来部署目前非常流行的博客系统wordpress的运行环境nginx php mysql(作者wordpress博客www.centos.bz正是运行在docker容器中)。那么docker部署wordpress的运行环境与我们传统上直接在主机配置环境有什么区别?我们从开发和运维人员角度来说明。运维使用docker制作好wordpress容器,分发给开发人员,开发人员随即只需一个命令就可以部署好完全一样的运行环境,从此只需要关注代码本身,而不再需要把时间浪费在配置环境上。而同时,docker容器确保了开发环境与生产环境的一致性,极大减少由于开发环境与生产环境不一致出现的各种问题。而由于docker容器可以快速部署的特点,运维人员可以很轻松的对服务进行伸缩和扩展。

那么如何使用docker部署wordpress的运行环境?大概步骤是分别编写nginx php mysql的Dockerfile文件,从这些Dockerfile文件中生成各自的镜像,然后使用docker-compose工具来统一管理nginx php mysql。为了能只使用docker-compose.yml一个文件就能快速部署wordpress环境,我们把Dockerfile及环境的相关配置保存到阿里云的Kelude(git代码托管code.aliyun.com),然后使用阿里云的Docker镜像仓库(cr.console.aliyun.com)从Kelude拉取Dockerfile自动构建镜像。国外此类服务有hub.docker.com和github.com,使用阿里云的是因为可以免费设置私有git仓库和私有镜像,因为我们可能需要保存一些不便公开的私密信息(如网站证书,密码)。当然你也可以不使用这类服务,直接把镜像保存到本地环境中。下面开始一步步介绍(如需帮助,请联系QQ 452336092)。

准备工作

使用阿里云Kelude

到https://code.aliyun.com/创建一个项目,如Dockerfile。之后我们把wordpress环境的所有相关Dockerfile及配置文件放置到centosbz目录。

使用阿里云镜像仓库

阿里云docker镜像仓库地址为https://cr.console.aliyun.com,用来存放docker镜像,可以在本地push镜像上去,也可以从Kelude拉取Dockerfile自动构建镜像。我们先登录,然后新建一个namespace,如centos-bz,之后所有的nginx,php,mysql镜像将存放在这个namespace下。

安装docker-compose

需要在运行docker容器的主机上安装docker-compose,可以参照官方文档手动安装,也可以使用ezhttp的一键安装工具(推荐)安装。如:

  1. wget centos.bz/ezhttp.zip

  2. unzip ezhttp.zip

  3. cd ezhttp-master

  4. ./start.sh

之后会弹出一个菜单,输入2选择Some Useful Tools,然后输入18选择安装docker和compose。

编写Dockerfile

clone以上在阿里云Kelude创建的Dockerfile镜像到本地,在此项目中创建centos.bz,然后在centos.bz目录分别创建mysql,nginx,php目录,用于存放它们各自Dockerfile及配置文件。

这里我们还约定以下目录:

/home/docker/nginx/logs/centos.bz:存放www.centos.bz网站的日志 /home/docker/nginx/www/centos.bz: 存放www.centos.bz网站的文件 /home/docker/php: 存放php-fpm的日志 /home/docker/mysql:mysql data目录

nginx Dockerfile

在nginx目录创建Dockerfile文件,写入如下内容:

  1. # 从debian:jessie镜像基础上安装nginx

  2. FROM debian:jessie

  3. # 声明此Dockerfile维护者的邮箱,有什么问题可以发到此邮件寻问

  4. LABEL maintainer "admin@centos.bz"

  5. # 定义软件版本及编译工具变量

  6. ENV NGINX_VERSION 1.10.3

  7. ENV OPENSSL_VERSION 1.0.2h

  8. ENV ZLIB_VERSION 1.2.11

  9. ENV PCRE_VERSION 8.40

  10. ENV CONCAT_VERSION 1.2.2

  11. ENV BUILD_TOOLS wget gcc make g++

  12. ENV SRC_DIR /opt/nginx

  13. # 切换到工作目录

  14. WORKDIR ${SRC_DIR}

  15. # 开始编译nginx,我们这里使用编译安装nginx而不是使用官方提供的nginx镜像是因为这里使用到了第三方的concat模块,只能编译了。

  16. # 把所有的安装命令都写在一个RUN指令中是因为这样可以减小镜像层数,缩减镜像大小。推荐使用反斜杠和&&把所有的安装命令放置到一行中。

  17. RUN apt-get update \

  18. && apt-get -y --no-install-recommends install ca-certificates ${BUILD_TOOLS} \

  19. && wget http://nginx.org/download/nginx-${NGINX_VERSION}.tar.gz \

  20. && wget https://www.openssl.org/source/openssl-${OPENSSL_VERSION}.tar.gz \

  21. && wget http://www.zlib.net/zlib-${ZLIB_VERSION}.tar.gz \

  22. && wget https://ftp.pcre.org/pub/pcre/pcre-${PCRE_VERSION}.tar.gz \

  23. && wget https://github.com/alibaba/nginx-http-concat/archive/${CONCAT_VERSION}.tar.gz -O nginx-http-concat-${CONCAT_VERSION}.tar.gz \

  24. && tar xf nginx-${NGINX_VERSION}.tar.gz \

  25. && tar xf openssl-${OPENSSL_VERSION}.tar.gz \

  26. && tar xf zlib-${ZLIB_VERSION}.tar.gz \

  27. && tar xf pcre-${PCRE_VERSION}.tar.gz \

  28. && tar xf nginx-http-concat-${CONCAT_VERSION}.tar.gz \

  29. && cd nginx-${NGINX_VERSION} \

  30. && ./configure --prefix=/usr/local/nginx --with-pcre=../pcre-${PCRE_VERSION} \

  31. --with-zlib=../zlib-${ZLIB_VERSION} \

  32. --with-http_ssl_module \

  33. --with-openssl=../openssl-${OPENSSL_VERSION} \

  34. --add-module=../nginx-http-concat-${CONCAT_VERSION} \

  35. && make -j$(nproc) \

  36. && make install \

  37. && rm -rf ${SRC_DIR} \

  38. && apt-get purge -y --auto-remove ${BUILD_TOOLS} \

  39. && rm -rf /var/lib/apt/lists/*

  40. # 把构建上下文目录conf,即Dockerfile/centos.bz/nginx/conf目录下的文件复制到容器的/usr/local/nginx/conf目录。

  41. COPY conf/ /usr/local/nginx/conf/

  42. # 定义启动容器时运行的命令

  43. ENTRYPOINT ["/usr/local/nginx/sbin/nginx"]

  44. EXPOSE 80 443

对于conf目录下的nginx配置文件,需要把日志,网站目录更改为以下约定的目录位置。

php-fpm Dockerfile

创建Dockerfile/centos.bz/php-fpm目录,在此目录下创建Dockerfile文件,内容如下:

  1. FROM debian:jessie

  2. LABEL maintainer "admin@centos.bz"

  3. # 定义软件版本,编译工具,依赖等变量

  4. ENV PHP_VERSION 5.6.30

  5. ENV BUILD_TOOLS m4 \

  6. autoconf \

  7. autoconf2.13 \

  8. openssl \

  9. wget \

  10. gcc \

  11. make

  12. ENV BUILD_DEPS libcurl4-gnutls-dev \

  13. libxml2-dev \

  14. zlib1g-dev \

  15. libpcre3-dev \

  16. libjpeg-dev \

  17. libpng12-dev \

  18. libfreetype6-dev \

  19. libmhash-dev \

  20. libmcrypt-dev \

  21. libssl-dev \

  22. libtool

  23. ENV PHP_LOCATION /usr/local/php

  24. ENV BUILD_ARG --prefix=${PHP_LOCATION} \

  25. --with-config-file-path=${PHP_LOCATION}/etc \

  26. --enable-fpm \

  27. --enable-bcmath \

  28. --with-pdo_sqlite \

  29. --with-gettext \

  30. --with-iconv \

  31. --enable-ftp \

  32. --with-sqlite3 \

  33. --enable-mbstring \

  34. --enable-sockets \

  35. --enable-zip \

  36. --enable-soap \

  37. --with-openssl \

  38. --with-zlib \

  39. --with-curl \

  40. --with-gd \

  41. --with-jpeg-dir \

  42. --with-png-dir \

  43. --with-freetype-dir \

  44. --with-mcrypt \

  45. --with-mhash \

  46. --with-mysql=mysqlnd \

  47. --with-mysqli=mysqlnd \

  48. --with-pdo-mysql=mysqlnd \

  49. --without-pear \

  50. --with-libdir=lib64 \

  51. --enable-opcache \

  52. --disable-cgi

  53. ENV SRC_DIR /opt/php

  54. WORKDIR ${SRC_DIR}

  55. # 开始编译安装php

  56. RUN apt-get update \

  57. && apt-get -y --no-install-recommends install ${BUILD_DEPS} ${BUILD_TOOLS} \

  58. && wget http://php.net/distributions/php-${PHP_VERSION}.tar.gz \

  59. && tar xf php-${PHP_VERSION}.tar.gz \

  60. && cd php-${PHP_VERSION} \

  61. && ln -s /usr/lib/x86_64-linux-gnu/libssl.so /usr/lib/libssl.so \

  62. && ln -s /usr/lib /usr/lib64 \

  63. && ./configure ${BUILD_ARG} \

  64. && make -j$(nproc) \

  65. && make install \

  66. && cp php.ini-production ${PHP_LOCATION}/etc/php.ini \

  67. && cp ${PHP_LOCATION}/etc/php-fpm.conf.default ${PHP_LOCATION}/etc/php-fpm.conf \

  68. && rm -rf ${SRC_DIR} \

  69. && apt-get purge -y --auto-remove ${BUILD_TOOLS} \

  70. && rm -rf /var/lib/apt/lists/*

  71. WORKDIR ${PHP_LOCATION}/etc/

  72. # 配置php-fpm,即使用sed工具编辑php-fpm.conf和php.ini文件,这里的php-fpm相关配置命令不与上面的编译命令合在一起来减小层数是因为

  73. # 配置文件可能会改动比较多,这样分开当配置文件更改时可以直接使用缓存跳过编译步骤,加快构建速度。

  74. RUN set_php_variable(){ \

  75. local key=$1; \

  76. local value=$2; \

  77. if grep -q -E "^$key\s*=" php.ini;then \

  78. sed -i -r "s#^$key\s*=.*#$key=$value#" php.ini; \

  79. else \

  80. sed -i -r "s#;\s*$key\s*=.*#$key=$value#" php.ini; \

  81. fi; \

  82. if ! grep -q -E "^$key\s*=" php.ini;then \

  83. echo "$key=$value" >> php.ini; \

  84. fi; \

  85. } \

  86. && BASE_DIR=/home/docker/php \

  87. && set_php_variable disable_functions "dl,eval,assert,exec,popen,system,passthru,shell_exec,escapeshellarg,escapeshellcmd,proc_close,proc_open" \

  88. && set_php_variable expose_php Off \

  89. && set_php_variable error_log ${BASE_DIR}/php_errors.log \

  90. && set_php_variable request_order "CGP" \

  91. && set_php_variable cgi.fix_pathinfo 0 \

  92. && set_php_variable short_open_tag on \

  93. && set_php_variable date.timezone Asia/Chongqing \

  94. && sed -i 's/^user =.*/user = www-data/' php-fpm.conf \

  95. && sed -i 's/^group =.*/group = www-data/' php-fpm.conf \

  96. && sed -i "s#;slowlog = log/\$pool.log.slow#slowlog = ${BASE_DIR}/\$pool.log.slow#" php-fpm.conf \

  97. && sed -i 's/;request_slowlog_timeout = 0/request_slowlog_timeout = 5/' php-fpm.conf \

  98. && sed -i 's/^pm.max_children.*/pm.max_children =20/' php-fpm.conf \

  99. && sed -i 's/^pm.start_servers.*/pm.start_servers =5/' php-fpm.conf \

  100. && sed -i 's/^pm.min_spare_servers.*/pm.min_spare_servers =3/' php-fpm.conf \

  101. && sed -i 's/^pm.max_spare_servers.*/pm.max_spare_servers =8/' php-fpm.conf \

  102. && sed -i '/\[global\]/a\daemonize =no' php-fpm.conf \

  103. && sed -i 's/^listen.*/listen =0.0.0.0:9000/' php-fpm.conf \

  104. && echo "[opcache]\n \

  105. zend_extension=opcache.so\n \

  106. opcache.memory_consumption=128\n \

  107. opcache.interned_strings_buffer=8\n \

  108. opcache.max_accelerated_files=4000\n \

  109. opcache.revalidate_freq=60\n \

  110. opcache.fast_shutdown=1 \n" >> php.ini

  111. ENTRYPOINT ["/usr/local/php/sbin/php-fpm"]

  112. EXPOSE 9000

mysql Dockerfile

创建Dockerfile/centos.bz/mysql/Dockerfile文件,内容如下:

  1. FROM mysql:5.6

  2. LABEL maintainer "admin@centos.bz"

  3. COPY my.cnf /etc/mysql/my.cnf

这个Dockerfile非常简单,直接使用了官方的mysql镜像,唯一区别是我们使用自己定义的my.cnf配置文件。

对于my.cnf配置文件,需要把日志,data目录指向/home/docker/mysql,一个my.cnf示例文件如下:

  1. # Generated by EZHTTP at 2016-02-03 01:05:29

  2. [mysql]

  3. # CLIENT #

  4. port = 3306

  5. socket = /home/docker/mysql/mysql.sock

  6. [mysqld]

  7. # GENERAL #

  8. port = 3306

  9. user = mysql

  10. default-storage-engine = InnoDB

  11. socket = /home/docker/mysql/mysql.sock

  12. pid-file = /home/docker/mysql/mysql.pid

  13. skip-name-resolve

  14. # MyISAM #

  15. key-buffer-size = 32M

  16. # INNODB #

  17. #innodb-flush-method = O_DIRECT

  18. innodb-log-files-in-group = 2

  19. innodb-log-file-size = 64M

  20. innodb-flush-log-at-trx-commit = 2

  21. innodb-file-per-table = 1

  22. innodb-buffer-pool-size = 1G

  23. # CACHES AND LIMITS #

  24. tmp-table-size = 32M

  25. max-heap-table-size = 32M

  26. query-cache-type = 0

  27. query-cache-size = 0

  28. max-connections = 300

  29. thread-cache-size = 50

  30. open-files-limit = 1024

  31. table-definition-cache = 100

  32. table-open-cache = 400

  33. # SAFETY #

  34. max-allowed-packet = 16M

  35. max-connect-errors = 1000000

  36. # DATA STORAGE #

  37. datadir = /home/docker/mysql

  38. # LOGGING #

  39. log-error = /home/docker/mysql/mysql-error.log

  40. log-queries-not-using-indexes = 1

  41. slow-query-log = 1

  42. slow-query-log-file = /home/docker/mysql/mysql-slow.log

  43. # BINARY LOGGING #

  44. log-bin = /home/docker/mysql/mysql-bin

  45. server-id = 1

  46. expire-logs-days = 14

  47. sync-binlog = 1

构建镜像

把上一步创建的文件推送到阿里云的Kelude。然后我们登录阿里云的docker镜像仓库cr.console.aliyun.com。这里以设置自动构建nginx镜像为例,php和mysql镜像构建设置类似。

1.点击左侧“镜像列表”,在右侧点击仓库镜像,如图:

2.在仓库镜像创建对话框中,说明如下:

地域:选择离部署docker主机最近的位置,国内的话选择华东1或华东2。

Namespace和仓库名称:这里选择centos-bz,nginx。

设置代码源:我们这里选择阿里云code。

构建设置:勾选代码变更时自动构建镜像,海外机器构建(因为国内主机apt-get安装软件时较慢),Dockerfile路径填/centos.bz/nginx

完成后点击创建仓库按钮。

如图:

3.回到镜像列表,找到nginx镜像,点击管理。

4.左侧点击“构建”,右侧点击“立即构建”开始首次构建,之后我们更改Dockerfile及配置文件到Kelude之后就会自动构建了。

5.查看日志,查看构建进程。

然后继续完成php,mysql的镜像构建设置。

启动环境

为了方便统一管理nginx,php,mysql的启动,我们使用docker-compose工具。我们只需要编写一个docker-compose.yml文件,然后使用docker-compose工具就可以快速启动docker容器了。之后把docker-compose.yml传输到任意一台支持docker环境的主机中就可以快速配置wordpress的运行环境。

docker-compose.yml

把docker-compose.yml文件放置在/home/docker目录下。

  1. version: '3'

  2. # 定义三个服务nginx,php,mysql

  3. services:

  4. nginx:

  5. # 依赖php服务,意味着在启动nginx之前先启动php

  6. depends_on:

  7. - php

  8. # nginx镜像的路径

  9. image: registry.cn-hangzhou.aliyuncs.com/centos-bz/nginx

  10. # 容器的/home/docker/nginx目录挂载主机中的/home/docker/nginx目录,

  11. # 这样使nginx容器把网站文件和目录存放到主机目录中,持久化和方便管理

  12. volumes:

  13. - /home/docker/nginx:/home/docker/nginx

  14. # nginx意外退出时自动重启

  15. restart: always

  16. # 映射80和443端口

  17. ports:

  18. - "80:80"

  19. - "443:443"

  20. # 容器名称

  21. container_name: nginx

  22. php:

  23. depends_on:

  24. - mysql

  25. image: registry.cn-hangzhou.aliyuncs.com/centos-bz/php-fpm

  26. restart: always

  27. volumes:

  28. - /home/docker/nginx/www:/home/docker/nginx/www

  29. - /home/docker/php:/home/docker/php

  30. container_name: php

  31. mysql:

  32. image: registry.cn-hangzhou.aliyuncs.com/centos-bz/mysql

  33. volumes:

  34. - /home/docker/mysql:/home/docker/mysql

  35. restart: always

  36. # 设置MYSQL_ROOT_PASSWORD环境变量,这里是设置mysql的root密码。这里为root。

  37. environment:

  38. MYSQL_ROOT_PASSWORD: root

  39. container_name: mysql

启动环境

在/home/docker目录执行:

  1. docker-compose up

查看nginx,php,mysql是否正常启动,如果正常,ctrl-c停止,再执行:

  1. docker-compose up -d

这里compose命令就在后台启动了。

执行docker ps查看容器运行状态。

连接问题

容器之间可以通过容器名称来连接,如nginx配置文件中连接php的代码fastcgi_pass php:9000,网站数据库配置文件使用mysql:3306。

日常运维

迁移

比如A主机迁移到B主机。只需要三步。

1.打包A主机的/home/docker目录,传输到B主机相同位置 2.配置B主机docker环境 3.在B主机的/home/docker目录下执行docker-compose up -d

导出导入数据库

把centos.sql.gz数据库文件导入到centos数据库:

  1. gunzip < centos.sql.gz | docker exec -i mysql mysql -uroot -proot centos

把centos数据库导出到centos.sql.gz

  1. docker exec -i mysql mysqldump -uroot -proot centos | gzip > centos.sql.gz

备份

推荐使用ezhttp一键备份设置:

  1. wget centos.bz/ezhttp.zip

  2. unzip ezhttp.zip

  3. cd ezhttp-master

  4. ./start.sh

之后会弹出一个菜单,输入2选择Some Useful Tools,然后输入14选择备份设置。需要注意的是在设置mysql使用mysqldump备份时,在提示输入mysql bin directory时,输入docker exec /usr/bin/。

相关推荐

Linux下C++访问web—使用libcurl库调用http接口发送解析json数据

一、背景这两天由于一些原因研究了研究如何在客户端C++代码中调用web服务端接口,需要访问url,并传入json数据,拿到返回值,并解析。 现在的情形是远程服务端的接口参数和返回类型都是json的字符...

干货 | 这 3 个超经典的Linux实战项目,让你分分钟入门Linux系统

编译安装nginx搭建小游戏网站编译安装流程下载nginx代码wget-P/server/tools/http:nginx.org/download/nginx1.22.0.tar.gz解压并进...

权限管理-树莓派linux⑦

前言当你在看这篇README,我感到非常荣幸。作为支持开源、分享的理念的我,给大家带来一些学习上的乐趣。由于本人并非专业的教育领域人士,很多时候天马行空,随心所欲的表达方式,可能让部分人感到不适。请根...

每天Linux学习:linux文件属性

ls-lih先通过这个命令来观察(-l列表显示目录内容详细,-i第一列显示inode,-h将文件大小显示为我们常见的kb,mb等单位)从截图中我们能看出文件属性由这9列信息组成:第1列:inod...

Linux ln、unlink命令用法

ln命令可以用来创建软链接或硬链接。1.创建软链接:ln-s源文件目标文件例如:ln-s/usr/lib/libc.so/usr/local/lib/libc.so.6这样就创建了一...

Linux 系统启动完整流程

一、启动系统流程简介如上图,简述系统启动的大概流程:1:硬件引导UEFi或BIOS初始化,运行POST开机自检2:grub2引导阶段系统固件会从MBR中读取启动加载器,然后将控制权交给启动加载器GRU...

最火的 CI/CD 平台 Jenkins 详细搭建教程(for Linux)

在正式学习Jenkins之前我们需要对两个名词有一定了解,其一是DevOps,另外一个就是CI/CD。何为DevOps?来自wiki百科介绍DevOps是一系列软件开发实践,强调开发人员(Dev)和测...

hadoop集群搭建详细方法

第一步:搭建配置新的虚拟机格式化之前先把tmp目录下所有与Hadoop有关的信息全部删除rm-rf/tmp/hadoop-centos*开启之后jps只有Java的进程:sudovi/et...

Linux 常用命令集合

系统信息arch显示机器的处理器架构(1)uname-m显示机器的处理器架构(2)uname-r显示正在使用的内核版本dmidecode-q显示硬件系统部件-(SMBIOS/DM...

inode文件索引,你了解嘛?你的Linux基础真的扎实嘛?

一、inode是什么?深入了解inode,就要从文件存储说起来!文件储存在硬盘上,硬盘的最小存储单位叫做"扇区"(Sector)。每个扇区储存512字节。读取硬盘的时候,不会一个个扇区地读取,这样效率...

linux实例之创建service服务

前面我们讲过可以通过service命令来启动,重启,停止指定的服务程序。service服务可以在系统启动时,自动运行该服务,我们可以利用这一特点,创建service文件,并且让系统重启时,自动执行命令...

linux之软连接和硬连接的区别

硬连接硬链接是通过索引节点进行的链接。在Linux中,多个文件指向同一个索引节点是允许的,像这样的链接就是硬链接。硬链接只能在同一文件系统中的文件之间进行链接,不能对目录进行创建。如果删除硬链接对应的...

Linux inode 详解

简介索引节点(IndexNode)是Linux/类unix系统文件系统上的一种数据结构,用于存储有关文件或目录的元数据。它包含文件的所有信息,除了文件名和数据。inode在文件系统如何存储和检...

Bash 脚本实例:获取符号链接的目标位置

我们都熟悉Linux中的符号链接,通常称为符号链接或软链接,符号链接是指向任何文件系统中的另一个文件或目录的特定文件。本文将介绍Linux中符号链接的基础知识,并创建一个简单的bash脚本...

windows快捷方式,符号链接,软链接和硬链接

当一个软件大量的向C盘写入数据,而我们又无法修改软件保存数据的位置时,可以使用windows系统的“符号链接”(SymbolicLink)功能,将保存数据的位置修改到其它分区中。符号链接类似于我们熟...

取消回复欢迎 发表评论: