這一章的主要內(nèi)容都是在執(zhí)行命令。我們也重點把命令整理出來,執(zhí)行一遍,看看效果練練手。
cd /data/docker_test/
mkdir sample
cd sample/
touch Dockerfile
vim Dockerfile
dockerfile 里面的內(nèi)容
FROM ubuntu
MAINTAINER huangzelin "huangzelin@example.com"
ENV REFRESHER_AT 202-12-01
RUN apt-get update
RUN apt-get -y -q install nginx
RUN mkdir -p /var/www/html
ADD nginx/global.conf /etc/nginx/conf.d/
ADD nginx/nginx.conf /etc/nginx/
EXPOSE 80
然后給nginx傳入兩份配置
mkdir nginx && cd nginx
vim global.conf
vim nginx.conf
global.conf
server {
listen 0.0.0.0:80;
server_name _;
root /var/www/html/website;
index index.html index.htm;
access_log /var/log/nginx/default_access.log;
error_log /var/log/nginx/default_error.log;
}
nginx.conf
user www-data;
worker_processes 4;
pid /run/nginx.pid;
daemon off;
events { }
http {
sendfile on;
tcp_nopush on;
tcp_nodelay on;
keepalive_timeout 65;
types_hash_max_size 2048;
include /etc/nginx/mime.types;
default_type application/octet-stream;
access_log /var/log/nginx/access.log;
error_log /var/log/nginx/error.log;
gzip on;
gzip_disable "msie6";
include /etc/nginx/conf.d/*.conf;
}
構(gòu)建鏡像
sudo docker build -t huangzelin/nginx .
創(chuàng)建卷目錄,并寫入網(wǎng)頁內(nèi)容
mkdir website && cd website
vim index.html
<head>
<title>Test website</title>
</head>
<body>
<h1>This is a yahaha website</h1>
</body>
運行docker并指定卷目錄
docker run -d -p 80 --name website -v $PWD/website:/var/www/html/website huangzelin/nginx nginx
查看端口
docker ps
CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
87a99d7b5681 huangzelin/nginx "nginx" 13 minutes ago Up 13 minutes 0.0.0.0:32768->80/tcp website

修改卷目錄下面的代碼。
<head>
<title>Test website</title>
</head>
<body>
<h1>This is a yahaha website</h1>
</body>

對應(yīng)內(nèi)容也會馬上生效。
構(gòu)建一個web應(yīng)用程序
創(chuàng)建一個webAPP目錄
mkdir sinatra && cd sinatra
vim Dockerfile
FROM ubuntu
LABEL maintainer="huangzelin@example.com"
ENV REFRESHED_AT 2014-06-01
RUN apt-get -qq update && apt-get -qq install ruby ruby-dev build-essential redis-tools
RUN gem sources --remove https:/rubygems.org
RUN gem sources -a https://gems.ruby-china.com
RUN gem install sinatra json redis
RUN mkdir -p /opt/webapp
EXPOSE 4567
CMD [ "/opt/webapp/bin/webapp" ]
這里把gem轉(zhuǎn)成國內(nèi)的鏡像源,不然太慢了。
然后創(chuàng)建webapp目錄。這下面是作者寫的一個ruby程序。
可以去這里拿
給作者的程序增加可運行權(quán)限。
chmod +x webapp/bin/webapp
然后開始build鏡像。
docker build -t huangzelin/sinatra .
啟動
sudo docker run -d -p 4567 --name webapp -v $PWD/webapp:/opt/webapp huangzelin/sinatra
看一下他的映射端口:
root@DESKTOP-3JK8RKR:/data/docker_test/sinatra# docker ps |grep webapp
c4da61b1d6fa huangzelin/sinatra "/opt/webapp/bin/web…" 7 minutes ago Up 7 minutes 0.0.0.0:32768->4567/tcp webapp

是通的,作者這個小程序有個路由,是post到/json的。
post '/json/?' do
params.to_json
end
也確實是生效的。

啟動redis
老規(guī)矩,新起一個redis目錄寫Dockerfile
mkdir redis && cd redis
vim Dockerfile
FROM ubuntu
LABEL maintainer="huangzelin@example.com"
ENV REFRESHED_AT 2014-06-01
RUN apt-get -qq update && apt-get -qq install redis-server redis-tools
EXPOSE 6379
ENTRYPOINT ["/usr/bin/redis-server" ]
CMD []
啟動
docker run -d -p 6379 --name redis huangzelin/redis
安裝一個redis-tools,看看能不能連上redis
apt-get -y install redis-tools
看看redis被映射到哪里了。
docker ps |grep redis
連接
redis-cli -h 127.0.0.1 -p XXXXX
除此此外,連接redis還可以通過docker的網(wǎng)卡去連接。
ip a show docker0
可以看到這個網(wǎng)卡的信息,容器要訪問外網(wǎng),首先一跳就需要跳到這個網(wǎng)卡上才能連接外網(wǎng),但是通過iptable限制了只有我們填寫了-p配置的端口才能被外網(wǎng)訪問。
所以我們也可以通過網(wǎng)卡來訪問到對應(yīng)的地址。
root@DESKTOP-3JK8RKR:/data/docker_test/redis# docker inspect redis -f '{{ .NetworkSettings.IPAddress}}'
172.18.0.2
查看地址連接。
redis-cli -h 172.18.0.2 -p 6379
但是這兩種連接方式,在重啟容器之后,地址可能會變的。
還有一種方式支持容器互聯(lián)。
我們把redis容器給刪了,重新起一個。
docker stop redis
docker rm redis
docker run -d --name redis huangzelin/redis
這次我們沒有重新起端口了。
docker run -p 4567 --name webapp --link redis:db -t -i -v $PWD/webapp:/opt/webapp huangzelin/sinatra /
bin/bash
我們讓之前的webapp新構(gòu)建一個容器(記得docker里面的name不能相同)然后多了一個配置之前沒見過,--link,標(biāo)識了兩個容器的父子連接。需要兩個參數(shù),一個事實容器名字,另一個是連接后容器的別名。這樣只有使用--link標(biāo)識連接到這個容器才能連接到這個端口,容器的端口不需要對本地宿主機公開。
進(jìn)入該容器之后,我們ping一下db
apt-get install inetutils-ping
ping db
可以看到是ping得通的。
而且在容器的env里面,還多了一些可以直接使用的連接信息。
root@4dab1238532d:/# env
DB_PORT_6379_TCP_ADDR=172.18.0.2
DB_PORT_6379_TCP=tcp://172.18.0.2:6379
DB_PORT=tcp://172.18.0.2:6379
那么我們的程序代碼里就能直接用這些環(huán)境變量去實現(xiàn)容器互聯(lián)了。
但是我們的redis版本是新版本,在沒有指定綁定端口并且沒有設(shè)置密碼的情況下,會啟動保護(hù)模式,我們啟動時加入?yún)?shù)把保護(hù)模式去掉。
docker stop redis
docker rm redis
docker run -d --name redis huangzelin/redis --protected-mode no
新建webapp_redis,用來啟動一個連接redis的應(yīng)用。
里面的內(nèi)容直接在作者的示例代碼里面拷貝。
在dockerbook-code/code/5/sinatra/webapp_redis下。
然后
docker build -t huangzelin/redis_app .
docker run -p 4567 --name redis_app --link redis:db -t -v $PWD/webapp_redis:/opt/webapp huangzelin/redis_app
再次用postman去發(fā)送同樣的請求。或者curl
curl -X POST \
http://localhost:32768/json \
-H 'cache-control: no-cache' \
-H 'content-type: multipart/form-data; boundary=----WebKitFormBoundary7MA4YWxkTrZu0gW' \
-H 'postman-token: f04598b5-7f81-b75b-4b79-eceee00866eb' \
-F name=Foo \
-F status=Bar
之后用之前安裝過redis的容器去連容器中的redis,就會發(fā)現(xiàn)。
docker run -p 4567 --link redis:db -t -i -v $PWD/webapp:/opt/webapp huangzelin/sinatra /bin/bash
root@3ce955ae5f46:/# env |grep DB
DB_PORT_6379_TCP_ADDR=172.18.0.2
root@3ce955ae5f46:/# redis-cli -h 172.18.0.2 -p 6379
172.18.0.2:6379> keys *
1) "params"
172.18.0.2:6379> get params
"[{\"name\":\"Foo\",\"status\":\"Bar\"}]"
172.18.0.2:6379>
redis確實是將參數(shù)存儲起來了。容器有了聯(lián)動。