0%

  • rsync 的时候 --include --exclude 不会使用绝对路径,全都是相对于 rsync 源目录的路径

  • --include 需要在 --exclude 前面

  • include 和 exclude 的路径前加 "/" 指的是相对 rsync 源目录路径的目录,如果不加,所有子目录下匹配上该路径的都会被匹配

  • 例子:

    • rsync /data/c --include="/e/" --exclude="e" root@127.0.0.1:/data/d,/data/c/d/e 会被同步, /data/c/e 不会被同步
    • rsync /data/c --exclude="e" root@127.0.0.1:/data/d, /data/c/d/e 和 /data/c/e 不会被同步,所有 /data/c 下目录名为 e 的都不会被同步(包括子目录下的)
    • rsync /data/c --include='/c/d/e/' --exclude="e" root@127.0.0.1:/data/d, /data/c/d/e 会被同步,所有 /data/c 下其他目录名为 e 的都不会被同步(包括子目录下的)

upgrade-swoole.sh

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
#!/usr/bin/env bash

# 参考资料:
# https://blog.csdn.net/hutianyou123/article/details/78133309 (CentOS6.5升级手动安装GCC4.8.2 与 CentOS 6.4 编译安装 gcc 4.8.1)
# https://www.cnblogs.com/coyu/p/5750627.html (CentOS yum升级GCC到4.8)

# swoole 4.3.3 依赖 4.8 以后的 gcc, 需要先升级 gcc
wget http://ftp.gnu.org/gnu/gcc/gcc-4.8.2/gcc-4.8.2.tar.bz2
tar -jxvf gcc-4.8.2.tar.bz2

cd gcc-4.8.2
bash ./contrib/download_prerequisites

mkdir gcc-build-4.8.2
cd gcc-build-4.8.2
../configure -enable-checking=release -enable-languages=c,c++ -disable-multilib
make -j4
sudo make install

wget http://people.centos.org/tru/devtools-2/devtools-2.repo
mv devtools-2.repo /etc/yum.repos.d
yum install devtoolset-2-gcc devtoolset-2-binutils devtoolset-2-gcc-c++

mv /usr/bin/gcc /usr/bin/gcc-4.4.7
mv /usr/bin/g++ /usr/bin/g++-4.4.7
mv /usr/bin/c++ /usr/bin/c++-4.4.7
ln -s /opt/rh/devtoolset-2/root/usr/bin/gcc /usr/bin/gcc
ln -s /opt/rh/devtoolset-2/root/usr/bin/c++ /usr/bin/c++
ln -s /opt/rh/devtoolset-2/root/usr/bin/g++ /usr/bin/g++
gcc --version


# 安装 swoole 4.3.3
wget https://github.com/swoole/swoole-src/archive/v4.3.3.tar.gz
tar -xvf v4.3.3.tar.gz
cd swoole-src-4.3.3
phpize
./configure
make
sudo make install

php --re swoole | grep version

1
2
3
location ^~ /abc/ {
return 301 http://192.168.2.123:8503$uri;
}

含义

  • ^~ 正则匹配后面的路径 (这里是 /abc/),这里 location 的作用是匹配所有 uri 前缀是 /abc/ 的请求

  • 301 永久重定向

  • $uri 加上这个防止重定向之后 uri 丢失

很多人都对 Volume 有一个误解,他们认为 Volume 是为了持久化。如此想法是因为他们觉得容器不能持久化,所以 Volume 应该是为了满足这个需求设计的。其实容器会一直存在,除非你删除它们:

这可能来自于容器不是持久的想法,这样确实是不对的。容器是持久的,直到你删除他们,并且你只能这样做:

1
docker rm my_container

如果你没有执行此命令,那么你的容器会一直存在,依旧可以启动、停止等。如果你找不到你的容器,可以运行此命令:

1
docker ps -a

docker ps 只能显示正在运行的容器,但是容器也会处于停止状态,这种情况下,上面的命令会显示所有的容器,无论它们处于什么状态。

综上,再次声明:Volume 并不是为了持久化。

什么是 Volume

Volume 可以将容器以及容器产生的数据分离开来,这样,当你使用 docker rm my_container 删除容器时,不会影响相关的数据。

Volume 可以使用以下两种方式创建:

  • 在 Dockerfile 中指定 VOLUME /some/dir

  • 执行 docker run -v /some/dir 命令来指定

无论哪种方式都是做了同样的事情。它们告诉 Docker 在主机上创建一个目录(默认情况下是在 /var/lib/docker 下),然后将其挂载到指定的路径(例子中是 /some/dir)。当删除使用该 Volume 的容器时,Volume 本身不会受到影响,它可以一直存在下去。

如果在容器中不存在指定的路径,那么该目录将会被自动创建。

你可以告诉 Docker 同时删除容器和其 Volume:

1
docker rm -v my_container

有时候,你想在容器中使用主机上的某个目录,你可以通过其它的参数来指定:

1
docker run -v /host/path:/some/path

这明确地告诉 Docker 使用指定的主机路径来代替 Docker 自己创建的根路径并挂载到容器内指定的路径(以上例子为 /some/path)。需要注意的是,这种方式同样支持文件。在 Docker 术语中,这通常被称为 bind-mount(虽然技术层面上是这样讲的,但是实际的感觉是所有的 Volume 都是 bind-mounts 的)。 如果主机上的路径不存在,目录将自动在给定的路径中创建。

对待 bind-mounts Volume 和一个正常的 Volume 有一点不同,它不会修改主机上那些非 Docker 创建的东西:

  • 一个正常的 Volume,Docker 会自动将指定 Volume 路径(如上面的示例 /some/path)上的数据复制到由 Docker 创建的新目录下,如果是 "bind-mount"。Volume 就不会这样做。(这样做会将主机上的目录复制到容器)

  • 当你执行 docker rm -v my_container 命令时,"bind-mount" 类型的 Volume 不会被删除

容器也可以与其它容器共享 Volume。

1
2
docker run --name my_container -v /some/path ...
docker run --volume-from my_container --name my_container2 ...

上面的命令告诉 Docker 从第一个容器挂载相同的 Volume 到第二个容器,它可以在两个容器之间共享数据。

如果你执行 docker rm -v my_container 命令,而上方的第二个容器依然存在,那 Volume 不会被删除,如果你不使用 docker rm -v my_container2 命令删除第二个容器,那它会一直存在。

Dockerfile 里的 VOLUME

正如前面提到的,Dockerfile 中的 VOLUME 指令也可以做同样的事情,类似 docker run 命令中的 -v 参数(除了你不能在 Dockerfile 指定主机路径)。也正因为如此,构建镜像时可以得到惊奇的效果。

在 Dockerfile 中的每个命令都会创建一个新的用于运行指定命令的容器,并将容器提交到镜像,每一步都是在前一步的基础上构建的。因此 Dockerfile 中 ENV FOO=bar 等同于:

1
2
cid=$(docker run -e FOO=bar <image>)
docker commit $cid

下面让我们来看看这个 Dockerfile 的例子发生了什么:

1
2
3
FROM debian:jessie
VOLUME /foo/bar
RUN touch /foo/bar/baz
1
docker build -t my_debian .

我们期待的是 Docker 创建一个名为 my_debian 并且 Volume 是 /foo/bar 的镜像,以及在 /foo/bar/baz 下添加了一个空文件,但是让我们看看等同的 CLI 命令行实际上做了哪些:

1
2
3
4
cid=$(docker run -v /foo/bar debian:jessie)
image_id=$(docker commit $cid)
cid=$(docker run $image_id touch /foo/bar/baz)
docker commit $(cid) my_debian

真实过程可能并不是这样,但是类似。

在这里,/foo/bar 会首先创建,所以我们每次通过这个镜像启动一个容器,都会有一个空的 /foo/bar 目录。正如前面所说,Dockerfile 中每个命令都会创建一个新容器。 也就是说,每次都会创建一个新的 Volume。由于例子的 Dockerfile 是先指定 Volume 的,所以当执行 touch /foo/bar/baz 命令的容器创建时,一个 Volume 会被挂载到 /foo/bar,然后 baz 才能被写入此 Volume,而不是实际的容器或镜像的文件系统内。

所以,牢记 Dockerfile 中 VOLUME 指令的位置,因为它在你的镜像内创建了不可改变的目录。