序
容器是用来提供服务的,每个容器都是运行一个进程,或许是一个web程序,或许是一个数据库服务,而在每个容器之间都是需要相互访问的,从而在这里构建一个python的程序,一个容器运行python的应用程序,一个容器用来运行redis服务,在应用程序中访问redis,具体架构如下:
运行redis的容器
运行redis的时候,步骤如下:
首先下载到redis的镜像,然后根据镜像运行一个镜像的实例,也就是redis这个实例,在其中需要注意的是,容器对外暴露的端口就是6379端口,从而在主机上对应监听一个6379的端口,差不多就是端口映射了。
运行应用程序容器
构建应用程序的dockerfile如下:
[root@docker appdockerfile]# ls -l
total 12
-rw-r--r--. 1 root root 665 Dec 18 04:16 app.py
-rw-r--r--. 1 root root 146 Dec 18 04:18 dockerfile
-rw-r--r--. 1 root root 12 Dec 18 04:16 requirements.txt
[root@docker appdockerfile]# cat app.py
from flask import Flask
from redis import Redis, RedisError
import os
import socket
# Connect to Redis
redis = Redis(host="redis", db=0, socket_connect_timeout=2, socket_timeout=2)
app = Flask(__name__)
@app.route("/")
def hello():
try:
visits = redis.incr("counter")
except RedisError:
visits = "cannot connect to Redis, counter disabled"
html = "Hello !" \
"Hostname:
" \
"Visits: "
return html.format(name=os.getenv("NAME", "world"), hostname=socket.gethostname(), visits=visits)
if __name__ == "__main__":
app.run(host='0.0.0.0', port=80)
[root@docker appdockerfile]# cat dockerfile
FROM python:2.7-slim
WORKDIR /app
ADD . /app
EXPOSE 80
CMD ["python","app.py"]
[root@docker appdockerfile]# cat requirements.txt
Flask
Redis
创建镜像:
查看生成的镜像:
运行应用程序容器并进行测试:
参数--link表示为tagname:alias,一个是标签的名称,一个是别名。
坑
1、容器的内部网络
其实redis暴露不暴露端口是无所谓的,这个端口是给宿主机访问的,而app的容器和redis的容器的交互实际上是通过内部网络进行的,如下:
容器默认使用的都是那个桥接网络,而不是使用宿主机的IP来进行通信,如果你使用的是宿主机的IP,然后来访问容器暴露的端口的话,会显示没有路由到这个redis的主机。
在这里连接的是redis的主机名,容器的网络是可以解析这个主机名的:
2、 dockerfile里面进行yum显示权限不足
在构建dockerfile的时候,为了进行调试为啥无法连接到redis主机,从而需要进行安装相关的包进行调试,但是构建dockerfile的时候,总是出现如下报错:
要想使用root权限,必须在dockerfile中添加指令如下:
表示使用root的权限运行程序。
3、COPY和ADD的区别
在使用dockerfile的时候,ADD指令和COPY指令很相似,都是将文件复制到对应的路径之中。首先,源路径都必须是相对的路径,相对于dockefile的路径。而ADD所做的可以更多,ADD的源路径可以是一个url,而COPY必须是一个目录或者文件;ADD的源文件如果是一个压缩文件,那么在复制的时候,会自动进行解压,而COPY不会。
推荐使用COPY。
就算使用绝对路径,也会直接变成相对路径。
本文来自企鹅号 - 运维Linux和python媒体
如有侵权,请联系 cloudcommunity@tencent.com 删除。
本文来自企鹅号 - 运维Linux和python媒体
如有侵权,请联系 cloudcommunity@tencent.com 删除。