docker本质是个进程,有很多资源是使用宿主机的,比如系统时间。正常使用时是会觉得很方便,但涉及到对系统资源的修改时,就比较麻烦了。
场景
使用docker部署了一个后端服务,测试需要改系统时间,如果直接改宿主机时间则会影响到其他的docker(不管是在宿主机上改还是通过–cap-add SYS_TIME参数在docker中修改)。有没有什么既能满足测试要求又不影响其他docker的方法呢?答案就是通过faketime来欺骗docker,达到此目的。
faketime
github地址
https://github.com/wolfcw/libfaketime
安装
git clone https://github.com/wolfcw/libfaketime
cd libfaketime
make
make install
安装完成后,在/usr/local/lib/下有个kaketime的目录
使用
比如想修改时间为2019-08-12 10:30:22,执行如下命令
export LD_PRELOAD=/usr/local/lib/faketime/libfaketime.so.1 FAKETIME="2019-08-12 10:30:22"
docker实践
dockerfile
先看一个使用Ubuntu基础镜像做的,可用于laravel的dockerfile
FROM ubuntu:16.04
# 设置时区
ENV TZ=Asia/Shanghai
RUN ln -snf /usr/share/zoneinfo/$TZ /etc/localtime && echo $TZ > /etc/timezone
COPY php-7.3.0.tar.bz2 php-7.3.0.tar.bz2
# 更新源,安装nginx
RUN sed -i 's/archive.ubuntu.com/mirrors.aliyun.com/g' /etc/apt/sources.list \
&& sed -i 's/security.ubuntu.com/mirrors.aliyun.com/g' /etc/apt/sources.list \
&& apt-get clean \
&& rm -fR /var/lib/apt/lists/* \
&& mkdir /var/lib/apt/lists/partial \
&& apt-get update \
&& apt-get upgrade -y \
&& apt-get install -y wget autoconf make gcc nginx procps libfreetype6-dev libpng-dev libzip-dev libcurl3-openssl-dev libbz2-dev libjpeg-dev libxpm-dev libfreetype6-dev libmcrypt-dev libmysql++-dev libxslt1-dev pkg-config libssl-dev libsslcommon2-dev zip unzip \
&& tar xvf php-7.3.0.tar.bz2 \
&& cd php-7.3.0/ \
&& ./configure --prefix=/usr/local/php --enable-fpm --enable-sockets --enable-mbstring=all -enable-mysqlnd --with-config-file-path=/usr/local/php/etc --with-mysqli=mysqlnd --with-pdo-mysql=mysqlnd --with-curl --with-gd --with-openssl \
&& make \
&& make install \
&& cd ext/zip \
&& /usr/local/php/bin/phpize \
&& ./configure --with-php-config=/usr/local/php/bin/php-config \
&& make ; make install \
&& cd ../../ext/bcmath \
&& /usr/local/php/bin/phpize \
&& ./configure --with-php-config=/usr/local/php/bin/php-config \
&& make ; make install \
&& cd ../../.. ; rm -fr php-7.3.0 ; rm -fr php-7.3.0.tar.bz2
# 安装 Composer
ENV COMPOSER_HOME /root/composer
COPY composer /usr/bin/composer
COPY composer /usr/local/bin/composer
RUN cp /usr/local/php/bin/php /usr/bin && composer config -g repo.packagist composer https://mirrors.aliyun.com/composer/
# 复制文件和代码
WORKDIR /data
COPY app/ /data/
COPY php.ini /usr/local/etc/php/php.ini
COPY libfaketime/ /data/
RUN echo 'extension=zip.so' >> /usr/local/php/etc/php.ini \
&& echo 'extension=bcmath.so' >> /usr/local/php/etc/php.ini \
&& cd /data/libfaketime && make ; make install \
&& rm -fr /data/libfaketime
COPY php-fpm.conf /usr/local/php/etc/php-fpm.conf
COPY nginx.conf /etc/nginx/nginx.conf
COPY env-example /data/.env
# Write Permission
RUN chmod -R 777 /data && usermod -u 1000 www-data
CMD ["nginx", "-g", "daemon off;"]
说明
- dockerfile的大概过程是:安装基本工具和依赖;编译安装php;复制composer,faketime到docker,编译安装faktime;复制项目代码,nginx配置,php配置到docker;使用nginx做前台进程启动docker。
- 用Ubuntu而不用PHP做docker基础镜像的原因是: 测试修改时间时要重启php-fpm,如果直接使用PHP的docker则重启php-fpm后docker会重启,faketime修改无效。
- 在docker里面下载php-7.3.0.tar.bz2,composer和faketime,如果网络不好则时间会比较长,这改成先下载好,然后拷贝到docker里打包,节省点时间。nginx,php-fpm的配置也要事先准备好。
- 由于是编译安装php,时间会长一点
使用
先使用该dockerfile打个包
docker build -t laravel:v1 .
打完之后运行起来
docker run -itd -p 9091:80 laravel:v1
docker ps
CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
d5fcc8f42bd4 laravel:v1 "nginx -g 'daemon of…" 37 seconds ago Up 35 seconds 0.0.0.0:9091->80/tcp trusting_darwin
进入docker
docker exec -it d5fcc8f42bd4 /bin/bash
root@d5fcc8f42bd4:/data#
修改时间下时间
root@d5fcc8f42bd4:/data# date
Sun Aug 25 15:24:40 Asia 2019
root@d5fcc8f42bd4:/data#
root@d5fcc8f42bd4:/data# export LD_PRELOAD=/usr/local/lib/faketime/libfaketime.so.1 FAKETIME="2019-08-12 10:30:22"
root@d5fcc8f42bd4:/data# date
Mon Aug 12 10:30:22 Asia 2019
可以看到当前终端的时间已经改了。
看看当前终端下php的时间是不是也改了:
root@d5fcc8f42bd4:/data# echo "<?php echo date('Y-m-d H:i:s'); " > time.php
root@d5fcc8f42bd4:/data# echo 'echo "\n";' >> time.php
root@d5fcc8f42bd4:/data# date
Mon Aug 12 10:30:22 Asia 2019
root@d5fcc8f42bd4:/data# php time.php
2019-08-12 10:30:22
可以看到也修改了。
再来改下php-fpm的时间
pkill php-fpm ; /usr/local/php/sbin/php-fpm
访问http://127.0.0.1:9091/time.php ,可以看到当前的运行时间也改了。
不过目前测试php-fpm的生效时间比较短,只有10秒左右,之后会还原,后续在找下原因完善下。
-------------- 关于php-fpm时间还原问题的更新
对于php-fpm模式,可以在配置文件中加一个参数clear_env = no,启动php-fpm时则不会清空环境变量,修改的时间就能一直生效了。
对于CLI或swoole模式则没有这个问题,修改时间后重启即可。