docker lnmp环境搭建
提示:本文可能只对lnmp环境有一定了解的基础有交流学习的价值。
1、下载相应镜像到本地
[root@localhost lnmp]# docker pull php:7.1-fpm
[root@localhost lnmp]# docker pull mysql:5.7
[root@localhost lnmp]# docker pull nginx:1.13
[root@localhost lnmp]# docker pull redis:3.2
[root@localhost lnmp]# docker pull memcached:1.5
2、创建相应容器
# 创建php:7.1-fpm的容器并命名php7.1,将容器的9000端口映射到主机的9000端口。把主机的/home/lnmp/www/目录挂载到容器的/www目录(这个目录用于存放php脚本文件)
[root@localhost lnmp]# docker run -d -p 9000:9000 --name php7.1 -v /home/lnmp/www/:/www php:7.1-fpm
# 创建nginx:1.13的容器并命名nginx1.13,将容器的80端口映射到主机的80端口。把主机的/home/lnmp/app/nginx1.13/conf/目录挂载到容器的/etc/nginx/conf.d目录;/home/lnmp/www/目录挂载到容器的/www目录。
[root@localhost lnmp]# docker run -d -p 80:80 --name nginx1.13 -v /home/lnmp/app/nginx1.13/conf/:/etc/nginx/conf.d -v /home/lnmp/www/:/www nginx:1.13
# 创建mysql:5.7的容器并命名mysql5.7,将容器的3306端口映射到主机的3306端口。把主机的/home/lnmp/data/mysql目录挂载到容器的/var目录。设置root的密码为123456。
[root@localhost lnmp]# docker run -d -p 3306:3306 --name mysql5.7 -v /home/lnmp/data/mysql:/var/lib/mysql -e MYSQL_ROOT_PASSWORD=123456 mysql:5.7
# 创建redis:3.2的容器并命名redis3.2,将容器的6379端口映射到主机的6379端口。把主机的/home/lnmp/data/redis目录挂载到容器的/data目录。设置redis的持久化保存。
[root@localhost lnmp]# docker run -d -p 6379:6379 --name redis3.2 -v /home/lnmp/data/redis:/data redis:3.2 redis-server --appendonly yes
# 创建memcached:1.5的容器并命名memcached1.5,将容器的11211端口映射到主机的11211端口。设置最大内存空间64M。
[root@localhost lnmp]# docker run -d -p 11211:11211 --name memcached1.5 memcached:1.5 -m 64
3、相关文件的配置
3.1 nignx的配置
在站点目录下(即lnmp环境搭建的目录),结构之后可见github链接地址。
下面以绝对地址说明:在/home/lnmp/app/nginx1.13/conf/目录下新建nginx.conf文件,保持nginx原有文件的面貌,配置如下:
server {
listen 80;
server_name localhost;
index index.php index.html index.htm;
root /www/testweb/project/ac;
#charset koi8-r;
#access_log /var/log/nginx/host.access.log main;
location / {
try_files $uri $uri/ /index.php?$query_string;
index index.php index.html index.htm;
# root /usr/share/nginx/html;
root /www/testweb/project/ac;
}
#error_page 404 /404.html;
# redirect server error pages to the static page /50x.html
#
error_page 500 502 503 504 /50x.html;
location = /50x.html {
root /www;
}
# proxy the PHP scripts to Apache listening on 127.0.0.1:80
#
#location ~ \.php$ {
# proxy_pass http://127.0.0.1;
#}
# pass the PHP scripts to FastCGI server listening on 127.0.0.1:9000
# 关键点在这里,下面会有说明。
location ~ [^/]\.php(/|$) {
fastcgi_pass 172.17.0.2:9000; # 这里的ip是对应PHP版本的容器的ip地址,这个为容器之间的通信,下节会有介绍
fastcgi_index index.php;
fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
include fastcgi_params;
}
# deny access to .htaccess files, if Apache's document root
# concurs with nginx's one
#
#location ~ /\.ht {
# deny all;
#}
}
在/home/lnmp/app/nginx1.13/conf/目录下新建fastcgi_params文件,配置如下:
fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
fastcgi_param PATH_INFO $fastcgi_script_name;
fastcgi_param QUERY_STRING $query_string;
fastcgi_param REQUEST_METHOD $request_method;
fastcgi_param CONTENT_TYPE $content_type;
fastcgi_param CONTENT_LENGTH $content_length;
fastcgi_param SCRIPT_NAME $fastcgi_script_name;
fastcgi_param REQUEST_URI $request_uri;
fastcgi_param DOCUMENT_URI $document_uri;
fastcgi_param DOCUMENT_ROOT $document_root;
fastcgi_param SERVER_PROTOCOL $server_protocol;
fastcgi_param REQUEST_SCHEME $scheme;
fastcgi_param HTTPS $https if_not_empty;
fastcgi_param GATEWAY_INTERFACE CGI/1.1;
fastcgi_param SERVER_SOFTWARE nginx/$nginx_version;
fastcgi_param REMOTE_ADDR $remote_addr;
fastcgi_param REMOTE_PORT $remote_port;
fastcgi_param SERVER_ADDR $server_addr;
fastcgi_param SERVER_PORT $server_port;
fastcgi_param SERVER_NAME $server_name;
# PHP only, required if PHP was built with --enable-force-cgi-redirect
fastcgi_param REDIRECT_STATUS 200;
虚拟主机等配置这里暂不介绍。
3.2 php的扩展的配置
官网文档是最好的说明。
打开链接转How to install more PHP extensions
。
在介绍一个网上看着比较好的博文Docker 中的 PHP 如何安装扩展,有时打开慢,这里再搬来下。
1、PHP 源码
为了保证 Docker 镜像尽量小,PHP 的源文件是以压缩包的形式存在镜像中,官方提供了 docker-php-source 快捷脚本,用于对源文件压缩包的解压(extract)及解压后的文件进行删除(delete)的操作。
FROM php:7.1-apache
RUN docker-php-source extract \
# 此处开始执行你需要的操作 \
&& docker-php-source delete
注意:一定要记得删除,否则解压出来的文件会大大增加镜像的文件大小。
2、安装扩展
a、核心扩展
这里主要用到的是官方提供的 docker-php-ext-configure 和 docker-php-ext-install 快捷脚本,如下
FROM php:7.1-fpm
RUN apt-get update \
# 相关依赖必须手动安装
&& apt-get install -y \
libfreetype6-dev \
libjpeg62-turbo-dev \
libmcrypt-dev \
libpng-dev \
# 安装扩展
&& docker-php-ext-install -j$(nproc) iconv mcrypt \
# 如果安装的扩展需要自定义配置时
&& docker-php-ext-configure gd --with-freetype-dir=/usr/include/ --with-jpeg-dir=/usr/include/ \
&& docker-php-ext-install -j$(nproc) gd
注意:这里的 docker-php-ext-configure 和 docker-php-ext-install 已经包含了 docker-php-source 的操作,所有不需要再手动去执行。
b、PECL 扩展
因为一些扩展并不包含在 PHP 源码文件中,所有需要使用 PECL(PHP 的扩展库仓库,通过 PEAR 打包)。用 pecl install 安装扩展,然后再用官方提供的 docker-php-ext-enable 快捷脚本来启用扩展,如下示例:
FROM php:7.1-fpm
RUN apt-get update \
# 手动安装依赖
&& apt-get install -y libmemcached-dev zlib1g-dev \
# 安装需要的扩展
&& pecl install memcached-2.2.0 \
# 启用扩展
&& docker-php-ext-enable memcached
c、其它扩展
一些既不在 PHP 源码包,也不再 PECL 扩展仓库中的扩展,可以通过下载扩展程序源码,编译安装的方式安装,如下示例:
FROM php:5.6-apache
RUN curl -fsSL 'https://xcache.lighttpd.net/pub/Releases/3.2.0/xcache-3.2.0.tar.gz' -o xcache.tar.gz \
&& mkdir -p xcache \
&& tar -xf xcache.tar.gz -C xcache --strip-components=1 \
&& rm xcache.tar.gz \
&& ( \
cd xcache \
&& phpize \
&& ./configure --enable-xcache \
&& make -j$(nproc) \
&& make install \
) \
&& rm -r xcache \
&& docker-php-ext-enable xcache
注意:官方提供的 docker-php-ext- 脚本接受任意的绝对路径(不支持相对路径,以便与系统内置的扩展程序进行区分)*。
所以,上面的例子也可以这样写:
FROM php:5.6-apache
RUN curl -fsSL 'https://xcache.lighttpd.net/pub/Releases/3.2.0/xcache-3.2.0.tar.gz' -o xcache.tar.gz \
&& mkdir -p /tmp/xcache \
&& tar -xf xcache.tar.gz -C /tmp/xcache --strip-components=1 \
&& rm xcache.tar.gz \
&& docker-php-ext-configure /tmp/xcache --enable-xcache \
&& docker-php-ext-install /tmp/xcache \
&& rm -r /tmp/xcache
这里make -j$(nproc)
这个数字好像是指cpu的核数,也没找到具体的相关说明。
然后这里可以参阅下节Dockerfile的使用
。
3.3 其它相关的配置
没啥好说的,以后来补充。
4、容器之间的通信
1、获取各个容器的ip地址:两种方法
a、参考博文:如何获取 docker 容器(container)的 ip 地址
[root@localhost conf]# docker inspect -f '{{.Name}} - {{range .NetworkSettings.Networks}}{{.IPAddress}}{{end}}' $(docker ps -aq)
/php7.1 - 172.18.0.4
/redis3.2 - 172.18.0.5
/mysql5.7 - 172.18.0.3
/memcached1.5 - 172.18.0.6
/nginx1.13 - 172.18.0.2
b、装完容器后,利用iptables
[root@localhost conf]# iptables -L --line-number
Chain DOCKER (2 references)
num target prot opt source destination
1 ACCEPT tcp -- anywhere 172.18.0.2 tcp dpt:http # nginx
2 ACCEPT tcp -- anywhere 172.18.0.3 tcp dpt:mysql # mysql
3 ACCEPT tcp -- anywhere 172.18.0.4 tcp dpt:cslistener # php 这里就是上面nginx配置需要写的ip地址
4 ACCEPT tcp -- anywhere 172.18.0.5 tcp dpt:6379 # redis
5 ACCEPT tcp -- anywhere 172.18.0.6 tcp dpt:memcache # memcache
上面已经说过nginx配置需要的php-fpm的ip地址就是dpt:cslistener
对应的地址。其它扩展对应的地址在代码中填写连接即可,例我的代码如下:
[
'type'=>'file',
'debug'=>true,
'pconnect'=>0,
'autoconnect'=>0,
],
'memcache'=>[
'hostname'=>'172.17.0.6',
'port'=>11211,
'type'=>'memcache',
'debug'=>true,
'pconnect'=>0,
'autoconnect'=>0,
'pre'=>'@ns_ls_',
'ismaster'=>true,
],
'memcached'=>[
'hostname'=>'172.17.0.6',
'port'=>11211,
'type'=>'memcached',
'debug'=>true,
'pconnect'=>0,
'autoconnect'=>0,
'pre'=>'@ns_ls_',
'ismaster'=>true,
],
'redis'=>[
'hostname'=>'172.17.0.5',
'port'=>6379,
'timeout'=>0,
'type'=>'redis',
'debug'=>true,
'pconnect'=>0,
'autoconnect'=>0,
'pre'=>'@ns_ls_',
'isusecluster'=>false,
],
'apc'=>[
'type'=>'apc',
'pre'=>'@ns_ls_',
],
'xcache'=>[
'type'=>'xcache',
'pre'=>'@ns_ls_',
],
'eaccelerator'=>[
'type'=>'eaccelerator',
'pre'=>'@ns_ls_',
],
'__pre'=>'TnPGcc_',
];
5、一些采坑的记录及建议
1、docker容器卷印射错误
[root@localhost ~]# docker run -d -p 9000:9000 --name php7.1 -v /home/wwwroot/:/var/www/ php:7.1-fpm
71f401361040492798a2deab03ccc86025fbd521f2db6308679ef13109162974
docker: Error response from daemon: oci runtime error: container_linux.go:262: starting container process caused "chdir to cwd (\"/var/www/html\") set in config.json failed: no such file or directory".
工作目录途径导致
2、nginx无法解析php文件或输出页为空白
[error] 5#5: *3 FastCGI sent in stderr: "Primary script unknown" while reading response header from upstream
解决:
location ~ [^/]\.php(/|$) {
fastcgi_pass 172.17.0.4:9000;
fastcgi_index index.php;
fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
include fastcgi_params;
}
建议:
- 单个容器一个个跑着安装积累经验,最后全部删除统一安装。
- 注意过程中的资料查找及笔记记录。
- 官档是最好的说明。