Docker Compose 引用环境变量
<p><span style="font-family: Microsoft YaHei; font-size: 15px">在项目中,往往需要在 docker-compose.yml 文件中使用环境变量来控制不同的条件和使用场景。本文集中介绍 docker compose 引用环境变量的方式。</span><br><span style="font-family: Microsoft YaHei; font-size: 15px">说明:本文的演示环境为 ubuntu 16.04。</span></p><h1><span style="font-family: Microsoft YaHei; font-size: 18pt">Compose CLI 与环境变量</span></h1>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">Compose CLI(compose command-line 即 docker-compose 程序)能够识别名称为 COMPOSE_PROJECT_NAME 和 COMPOSE_FILE 等环境变量(具体支持的环境变量请参考这里)。比如我们可以通过这两个环境变量为 docker-compose 指定 project 的名称和配置文件:</span></p>
<div class="cnblogs_code">
<pre>$ export COMPOSE_PROJECT_NAME=<span style="color: rgba(0, 0, 0, 1)">TestVar
$ export COMPOSE_FILE</span>=~/projects/composecounter/docker-compose.yml</pre>
</div>
<p><span style="font-family: Microsoft YaHei; font-size: 15px"><img src="https://img2018.cnblogs.com/blog/952033/201810/952033-20181021193147161-635372632.png"></span></p>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">然后启动应用,显示的 project 名称都是我们在环境变量中指定的: </span></p>
<p><span style="font-family: Microsoft YaHei; font-size: 15px"><img src="https://img2018.cnblogs.com/blog/952033/201810/952033-20181021193239211-1124545580.png"></span></p>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">如果设置了环境变量的同时又指定了命令行选项,那么会应用命令行选项的设置:</span></p>
<div class="cnblogs_code">
<pre>$ docker-compose -p nickproject up -d</pre>
</div>
<p><span style="font-family: Microsoft YaHei; font-size: 15px"><img src="https://img2018.cnblogs.com/blog/952033/201810/952033-20181021193320881-483163814.png"></span></p>
<h1><span style="font-family: Microsoft YaHei; font-size: 18pt">在 compose file 中引用环境变量</span></h1>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">我们还可以在 compose file 中直接引用环境变量,比如下面的 demo:</span></p>
<div class="cnblogs_code">
<pre>version: <span style="color: rgba(128, 0, 0, 1)">'</span><span style="color: rgba(128, 0, 0, 1)">3</span><span style="color: rgba(128, 0, 0, 1)">'</span><span style="color: rgba(0, 0, 0, 1)">
services:
web:
<strong>image: ${IMAGETAG}</strong>
ports:
</span>- <span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(128, 0, 0, 1)">5000:5000</span><span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(0, 0, 0, 1)">
redis:
image: </span><span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(128, 0, 0, 1)">redis:alpine</span><span style="color: rgba(128, 0, 0, 1)">"</span></pre>
</div>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">我们通过环境变量 ${IMAGETAG} 指定了 web 的镜像,下面通过 export 的方式来为 compose 配置文件中的环境变量传值:</span></p>
<p><span style="font-family: Microsoft YaHei; font-size: 15px"><img src="https://img2018.cnblogs.com/blog/952033/201810/952033-20181021193436149-979839196.png"></span></p>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">注意,如果对应的环境变量没有被设置,那么 compose 就会把它替换为一个空字符串:</span></p>
<p><span style="font-family: Microsoft YaHei; font-size: 15px"><img src="https://img2018.cnblogs.com/blog/952033/201810/952033-20181021193516245-1665429749.png"></span></p>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">碰到这种情况,我们可以在 compose 的配置文件中为该变量设置一个默认值:</span></p>
<div class="cnblogs_code">
<pre>version: <span style="color: rgba(128, 0, 0, 1)">'</span><span style="color: rgba(128, 0, 0, 1)">3</span><span style="color: rgba(128, 0, 0, 1)">'</span><span style="color: rgba(0, 0, 0, 1)">
services:
web:
<strong>image: ${IMAGETAG:</strong></span><strong>-</strong><span style="color: rgba(0, 0, 0, 1)"><strong>defaultwebimage}</strong>
ports:
</span>- <span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(128, 0, 0, 1)">5000:5000</span><span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(0, 0, 0, 1)">
redis:
image: </span><span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(128, 0, 0, 1)">redis:alpine</span><span style="color: rgba(128, 0, 0, 1)">"</span></pre>
</div>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">这样,如果没有设置 IMAGETAG 变量,就会应用 defaultwebimage:</span></p>
<p><span style="font-family: Microsoft YaHei; font-size: 15px"><img src="https://img2018.cnblogs.com/blog/952033/201810/952033-20181021193612306-971233538.png"></span></p>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">除了这种方式,我们还可以通过后面将介绍的 .env 文件来为环境变量设置默认值。</span></p>
<h1><span style="font-family: Microsoft YaHei; font-size: 18pt">把环境变量传递给容器</span></h1>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">先来看一下在 compose file 中如何为容器设置环境变量:</span></p>
<div class="cnblogs_code">
<pre><span style="color: rgba(0, 0, 0, 1)">web:
<strong>environment:
DEBUG: </strong></span><strong><span style="color: rgba(128, 0, 128, 1)">1</span></strong></pre>
</div>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">compose file 中的 environment 节点用来为容器设置环境变量,上面的写法等同于:</span></p>
<div class="cnblogs_code">
<pre>$ docker run <strong>-e DEBUG=<span style="color: rgba(128, 0, 128, 1)">1</span></strong></pre>
</div>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">要把当前 shell 环境变量的值传递给容器的环境变量也很简单,去掉上面代码中的赋值部分就可以了:</span></p>
<div class="cnblogs_code">
<pre><span style="color: rgba(0, 0, 0, 1)">web:
<strong>environment:
DEBUG:</strong></span></pre>
</div>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">这种情况下,如果没有在当前的 shell 中导出环境变量 DEBUG,compose file 中会把它解释为 null:</span></p>
<p><span style="font-family: Microsoft YaHei; font-size: 15px"><img src="https://img2018.cnblogs.com/blog/952033/201810/952033-20181021193714885-1722118683.png"></span></p>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">在试试导出环境变量 DEBUG 的情况:</span></p>
<div class="cnblogs_code">
<pre>$ export DEBUG=<span style="color: rgba(128, 0, 128, 1)">1</span></pre>
</div>
<p><span style="font-family: Microsoft YaHei; font-size: 15px"><img src="https://img2018.cnblogs.com/blog/952033/201810/952033-20181021193832687-990288802.png"></span></p>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">这才是我们设计的正确的使用场景!</span></p>
<h1><span style="font-family: Microsoft YaHei; font-size: 18pt">使用文件为容器设置多个环境变量</span></h1>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">如果觉得通过 environment 为容器设置环境变量不够过瘾,我们还可以像 docker -run 的 --env-file 参数一样通过文件为容器设置环境变量:</span></p>
<div class="cnblogs_code">
<pre><span style="color: rgba(0, 0, 0, 1)">web:
env_file:
</span> <span style="color: rgba(0, 0, 0, 1)"><strong>- web-variables.env</strong></span></pre>
</div>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">注意,web-variables.env 文件的路径是相对于 docker-compose.yml 文件的相对路径。上面的代码效果与下面的代码相同:</span></p>
<div class="cnblogs_code">
<pre>$ docker run <span style="color: rgba(0, 0, 0, 1)"><strong>--env-file=web-variables.env</strong></span></pre>
</div>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">web-variables.env 文件中可以定义一个或多个环境变量:</span></p>
<div class="cnblogs_code">
<pre># define web container <span style="color: rgba(0, 0, 255, 1)">env</span><span style="color: rgba(0, 0, 0, 1)">
APPNAME</span>=<span style="color: rgba(0, 0, 0, 1)">helloworld
AUTHOR</span>=<span style="color: rgba(0, 0, 0, 1)">Nick Li
VERSION</span>=<span style="color: rgba(128, 0, 128, 1)">1.0</span></pre>
</div>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">检查下结果:</span></p>
<p><span style="font-family: Microsoft YaHei; font-size: 15px"><img src="https://img2018.cnblogs.com/blog/952033/201810/952033-20181021194210315-791060157.png"></span></p>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">原来 compose 把 env_file 的设置翻译成了 environment!</span></p>
<h1><span style="font-family: Microsoft YaHei; font-size: 18pt">.env 文件</span></h1>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">当我们在 docker-compose.yml 文件中引用了大量的环境变量时,对每个环境变量都设置默认值将是繁琐的,并且也会影响 docker-compose.yml 简洁程度。此时我们可以通过 .env 文件来为 docker-compose.yml 文件引用的所有环境变量设置默认值!</span><br><span style="font-family: Microsoft YaHei; font-size: 15px">修改 docker-compose.yml 文件的内容如下:</span></p>
<div class="cnblogs_code">
<pre>version: <span style="color: rgba(128, 0, 0, 1)">'</span><span style="color: rgba(128, 0, 0, 1)">3</span><span style="color: rgba(128, 0, 0, 1)">'</span><span style="color: rgba(0, 0, 0, 1)">
services:
web:
image: ${IMAGETAG}
environment:
APPNAME:
AUTHOR:
VERSION:
ports:
</span>- <span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(128, 0, 0, 1)">5000:5000</span><span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(0, 0, 0, 1)">
redis:
image: </span><span style="color: rgba(128, 0, 0, 1)">"</span><span style="color: rgba(128, 0, 0, 1)">redis:alpine</span><span style="color: rgba(128, 0, 0, 1)">"</span></pre>
</div>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">然后在相同的目录下创建 .env 文件,编辑其内容如下:</span></p>
<div class="cnblogs_code">
<pre># define <span style="color: rgba(0, 0, 255, 1)">env</span><span style="color: rgba(0, 0, 0, 1)"> var default value.
IMAGETAG</span>=<span style="color: rgba(0, 0, 0, 1)">defaultwebimage
APPNAME</span>=<span style="color: rgba(0, 0, 0, 1)">default app name
AUTHOR</span>=<span style="color: rgba(0, 0, 0, 1)">default author name
VERSION</span>=default version is <span style="color: rgba(128, 0, 128, 1)">1.0</span></pre>
</div>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">检查下结果,此时所有的环境变量都显示为 .env 文件中定义的默认值:</span></p>
<p><span style="font-family: Microsoft YaHei; font-size: 15px"><img src="https://img2018.cnblogs.com/blog/952033/201810/952033-20181021194335736-1906362609.png"></span></p>
<h1><span style="font-family: Microsoft YaHei; font-size: 18pt">配置不同场景下的环境变量</span></h1>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">从前面的部分中我们可以看到,docker compose 提供了足够的灵活性来让我们设置 docker-compose.yml 文件中引用的环境变量,它们的优先级如下:</span></p>
<ol>
<li><span style="font-family: Microsoft YaHei; font-size: 15px">Compose file</span></li>
<li><span style="color: rgba(255, 0, 0, 1)"><strong><span style="font-family: Microsoft YaHei; font-size: 15px">Shell environment variables</span></strong></span></li>
<li><span style="font-family: Microsoft YaHei; font-size: 15px">Environment file</span></li>
<li><span style="font-family: Microsoft YaHei; font-size: 15px">Dockerfile</span></li>
<li><span style="font-family: Microsoft YaHei; font-size: 15px">Variable is not defined</span></li>
</ol>
<p><span style="font-family: Microsoft YaHei; font-size: 15px"><strong>首先</strong>,在 docker-compose.yml 文件中直接设置的值优先级是最高的。</span><br><span style="font-family: Microsoft YaHei; font-size: 15px"><strong>然后</strong>是在当前 shell 中 export 的环境变量值。</span><br><span style="font-family: Microsoft YaHei; font-size: 15px"><strong>接下来</strong>是在环境变量文件中定义的值。</span><br><span style="font-family: Microsoft YaHei; font-size: 15px"><strong>再接下来</strong>是在 Dockerfile 中定义的值。</span><br><span style="font-family: Microsoft YaHei; font-size: 15px"><strong>最后</strong>还没有找到相关的环境变量就认为该环境变量没有被定义。</span></p>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">根据上面的优先级定义,我们可以把不同场景下的环境变量定义在不同的 shell 脚本中并导出,然后在执行 docker-compose 命令前先执行 source 命令把 shell 脚本中定义的环境变量导出到当前的 shell 中。通过这样的方式可以减少维护环境变量的地方,下面的例子中我们分别在 docker-compose.yml 文件所在的目录创建 test.sh 和 prod.sh,test.sh 的内容如下:</span></p>
<div class="cnblogs_code">
<pre>#!/bin/<span style="color: rgba(0, 0, 0, 1)">bash
# define </span><span style="color: rgba(0, 0, 255, 1)">env</span><span style="color: rgba(0, 0, 0, 1)"> var default value.
export IMAGETAG</span>=<span style="color: rgba(0, 0, 0, 1)">web:v1
export APPNAME</span>=<span style="color: rgba(0, 0, 0, 1)">HelloWorld
export AUTHOR</span>=<span style="color: rgba(0, 0, 0, 1)">Nick Li
export VERSION</span>=<span style="color: rgba(128, 0, 128, 1)">1.0</span></pre>
</div>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">prod.sh 的内容如下:</span></p>
<div class="cnblogs_code">
<pre>#!/bin/<span style="color: rgba(0, 0, 0, 1)">bash
# define </span><span style="color: rgba(0, 0, 255, 1)">env</span><span style="color: rgba(0, 0, 0, 1)"> var default value.
export IMAGETAG</span>=<span style="color: rgba(0, 0, 0, 1)">webpord:v1
export APPNAME</span>=<span style="color: rgba(0, 0, 0, 1)">HelloWorldProd
export AUTHOR</span>=<span style="color: rgba(0, 0, 0, 1)">Nick Li
export VERSION</span>=<span style="color: rgba(128, 0, 128, 1)">1</span>.0LTS</pre>
</div>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">在测试环境下,执行下面的命令:</span></p>
<div class="cnblogs_code">
<pre>$ source test.<span style="color: rgba(0, 0, 255, 1)">sh</span><span style="color: rgba(0, 0, 0, 1)">
$ docker</span>-compose config</pre>
</div>
<p><span style="font-family: Microsoft YaHei; font-size: 15px"><img src="https://img2018.cnblogs.com/blog/952033/201810/952033-20181021194545225-651179018.png"></span></p>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">此时 docker-compose.yml 中的环境变量应用的都是测试环境相关的设置。</span></p>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">而在生产环境下,执行下面的命令:</span></p>
<div class="cnblogs_code">
<pre>$ source prod.<span style="color: rgba(0, 0, 255, 1)">sh</span><span style="color: rgba(0, 0, 0, 1)">
$ docker</span>-compose config</pre>
</div>
<p><span style="font-family: Microsoft YaHei; font-size: 15px"><img src="https://img2018.cnblogs.com/blog/952033/201810/952033-20181021194641722-1807047502.png"></span></p>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">此时 docker-compose.yml 中的环境变量应用的都是生产环境相关的设置。</span></p>
<h1><span style="font-family: Microsoft YaHei; font-size: 18pt">总结</span></h1>
<p><span style="font-family: Microsoft YaHei; font-size: 15px">docker compose 对环境变量的使用提供了非常丰富支持和灵活的使用方式。希望通过本文的总结可以帮助大家理清相关的用法,并能够以简洁的方式为不同的使用场景提供支持。</span></p>
<p><strong><span style="font-family: Microsoft YaHei; font-size: 15px">参考:</span></strong><br><span style="font-family: Microsoft YaHei; font-size: 15px">Compose CLI environment variables</span><br><span style="font-family: Microsoft YaHei; font-size: 15px">Environment variables in Compose</span><br><span style="font-family: Microsoft YaHei; font-size: 15px">Compose file variable substitution</span><br><span style="font-family: Microsoft YaHei; font-size: 15px">Declare default environment variables in file</span></p>
</div>
<div id="MySignature" role="contentinfo">
<div>作者:sparkdev</div>
<div>出处:http://www.cnblogs.com/sparkdev/</div>
<div>本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。</div><br><br>
来源:https://www.cnblogs.com/sparkdev/p/9826520.html
頁:
[1]