[Docker] 如何给 Docker 配置网络代理?

news/2025/3/4 11:25:15/文章来源:https://www.cnblogs.com/johnnyzen/p/18746970

有时因为网络原因,比如公司 NAT,或其它啥的,需要使用代理。Docker 的代理配置,略显复杂,因为有三种场景。但基本原理都是一致的,都是利用 Linux 的 http_proxy 等环境变量。

概述:Docker 网络代理方案

方案1:Dockerd 代理

  • 在执行docker pull时,是由守护进程dockerd来执行。

因此,代理需要配在dockerd的环境中。而这个环境,则是受systemd所管控,因此实际是systemd的配置。

sudo mkdir -p /etc/systemd/system/docker.service.d
sudo touch /etc/systemd/system/docker.service.d/proxy.confls -la /etc/systemd/system/docker.service.d/

在这个proxy.conf文件(可以是任意*.conf的形式)中,添加以下内容:

[Service]
Environment="HTTP_PROXY=http://proxy.example.com:8080/"
Environment="HTTPS_PROXY=http://proxy.example.com:8080/"
Environment="NO_PROXY=localhost,127.0.0.1,.example.com"

其中,http://proxy.example.com:8080 要换成可用的免密代理。

通常使用 cntlm 在本机自建免密代理,去对接公司的代理。可参考《Linux下安装配置Cntlm 代理》。

方案2:Container 代理

  • 容器运行阶段,如果需要代理上网,则需要配置 ~/.docker/config.json

以下配置,只在Docker 17.07及以上版本生效。

{"proxies":{"default":{"httpProxy": "http://proxy.example.com:8080","httpsProxy": "http://proxy.example.com:8080","noProxy": "localhost,127.0.0.1,.example.com"}}
}

这个是用户级的配置,除了 proxies,docker login 等相关信息也会在其中。

而且还可以配置信息展示的格式、插件参数等。

  • 此外,容器的网络代理,也可以直接在其运行时通过 -e 注入 http_proxy 等环境变量。

这两种方法分别适合不同场景。

config.json 非常方便,默认在所有配置修改后启动的容器生效,适合个人开发环境。

在CI/CD的自动构建环境、或者实际上线运行的环境中,这种方法就不太合适,用 -e 注入这种显式配置会更好,减轻对构建、部署环境的依赖。

当然,在这些环境中,最好用良好的设计避免配置代理上网

方案3:Docker Build 代理

  • 虽然 docker build 的本质,也是启动一个容器,但是环境会略有不同,用户级配置无效。

  • 在构建时,需要注入 http_proxy 等参数。

docker build . \--build-arg "HTTP_PROXY=http://proxy.example.com:8080/" \--build-arg "HTTPS_PROXY=http://proxy.example.com:8080/" \--build-arg "NO_PROXY=localhost,127.0.0.1,.example.com" \-t your/image:tag

注意:无论是 docker run 还是 docker build,默认是网络隔绝的。

如果代理使用的是 localhost:3128 这类,则会无效。

这类仅限本地的代理,必须加上 --network host 才能正常使用。

而一般则需要配置代理的外部IP,而且代理本身要开启 Gateway 模式。

重启生效

  • 代理配置完成后,reboot 重启当然可以生效,但不重启也行。

  • docker build 代理是在执行前设置的,所以修改后,下次执行立即生效

    • Container 代理的修改也是立即生效的,但是只针对以后启动的 Container,对已启动的 Container 无效。
  • dockerd 代理的修改比较特殊,它实际上是改 systemd 的配置,因此需要重载 systemd 并重启 dockerd 才能生效。

sudo systemctl daemon-reload
sudo systemctl restart docker

参考文献:如何配置docker通过代理服务器拉取镜像?

  • 如果 docker 所在的环境是通过代理服务器和互联网连通的,那么需要一番配置才能让 docker 正常从外网正常拉取镜像。

  • 然而,仅仅通过配置环境变量的方法是不够的。本文结合已有文档,介绍如何配置代理服务器能使docker正常拉取镜像。

本文使用的docker 版本是

docker --version
Docker version 24.0.2, build cb74dfc

问题现象

如果不配置代理服务器就直接拉镜像,docker 会直接尝试连接镜像仓库,并且连接超时报错。如下所示:

$ docker pull busybox
Using default tag: latest
Error response from daemon: Get https://registry-1.docker.io/v2/: net/http: request canceled 
while waiting for connection (Client.Timeout exceeded while awaiting headers)

问题分析

容易误导的官方文档

有这么一篇关于 docker 配置代理服务器的 官方文档 ,如果病急乱投医,直接按照这篇文章配置,是不能成功拉取镜像的。

我们来理解一下这篇文档,文档关键的原文摘录如下:

If your container needs to use an HTTP, HTTPS, or FTP proxy server, you can configure it in different ways: Configure the Docker client On the Docker client, create or edit the file ~/.docker/config.json in the home directory of the user that starts containers....When you create or start new containers, the environment variables are set automatically within the container.

这篇文档说:如果你的 容器 需要使用代理服务器,那么可以以如下方式配置: 在运行容器的用户 home 目录下,配置 ~/.docker/config.json 文件。重新启动容器后,这些环境变量将自动设置进容器,从而容器内的进程可以使用代理服务。

所以这篇文章是讲如何配置运行 容器 的环境与如何拉取镜像无关

如果按照这篇文档的指导,如同南辕北辙。

要解决问题,我们首先来看一般情况下命令行如何使用代理。

环境变量

  • 常规的命令行程序如果要使用代理,需要设置两个环境变量:HTTP_PROXYHTTPS_PROXY 。但是仅仅这样设置环境变量,也不能让 docker 成功拉取镜像。

我们仔细观察 上面的报错信息,有一句说明了报错的来源:

Error response from daemon:

  • 因为镜像的拉取和管理都是 docker daemon 的职责,所以我们要让 docker daemon 知道代理服务器的存在。

  • docker daemon 是由 systemd 管理的,所以我们要从 systemd 配置入手。

正确的官方文档

关于 systemd 配置代理服务器的 官方文档在这里,原文说:

The Docker daemon uses the HTTP_PROXY, HTTPS_PROXY, and NO_PROXY environmental variables in its start-up environment to configure HTTP or HTTPS proxy behavior. You cannot configure these environment variables using the daemon.json file.This example overrides the default docker.service file.If you are behind an HTTP or HTTPS proxy server, for example in corporate settings, you need to add this configuration in the Docker systemd service file.

这段话的意思是,docker daemon 使用 HTTP_PROXY, HTTPS_PROXY, 和 NO_PROXY 三个环境变量配置代理服务器,但是你需要在 systemd 的文件里配置环境变量,而不能配置在 daemon.json 里。

具体操作

下面是来自 Docker 官方文档 的操作步骤和详细解释:

1、创建 dockerd 相关的 systemd 目录,这个目录下的配置将覆盖 dockerd 的默认配置

$ sudo mkdir -p /etc/systemd/system/docker.service.d

新建配置文件 /etc/systemd/system/docker.service.d/http-proxy.conf,这个文件中将包含环境变量

[Service]
Environment="HTTP_PROXY=http://proxy.example.com:80"
Environment="HTTPS_PROXY=https://proxy.example.com:443"

如果你自己建了私有的镜像仓库,需要 dockerd 绕过代理服务器直连,那么配置 NO_PROXY 变量:

[Service]
Environment="HTTP_PROXY=http://proxy.example.com:80"
Environment="HTTPS_PROXY=https://proxy.example.com:443"
Environment="NO_PROXY=your-registry.com,10.10.10.10,*.example.com"

多个 NO_PROXY 变量的值用逗号分隔,而且可以使用通配符(*),极端情况下,如果 NO_PROXY=*,那么所有请求都将不通过代理服务器。

重新加载配置文件,重启 dockerd

$ sudo systemctl daemon-reload
$ sudo systemctl restart docker

检查确认环境变量已经正确配置:

$ sudo systemctl show --property=Environment docker

从 docker info 的结果中查看配置项。

这样配置后,应该可以正常拉取 docker 镜像。

结论

  • docker 镜像由 docker daemon 管理

    所以不能用修改 shell 环境变量的方法使用代理服务,而是从 systemd 角度设置环境变量。

    在一些特定环境下,需要在代理环境下使用Docker的某些功能,本文介绍一些场景下 如何配置网络代理

Y 推荐文献

  • [Docker] Docker Compose 基础教程(概念/基础操作) - 博客园/千千寰宇
  • [Docker] Docker 基础教程(概念/原理/基础操作) - 博客园/千千寰宇

X 参考文献

  • 如何配置docker通过代理服务器拉取镜像 - 醉马踏千秋 - 博客园
  • https://www.lfhacks.com/tech/pull-docker-images-behind-proxy/#correct
  • https://stackoverflow.com/questions/69047394/cant-pull-docker-image-behind-a-proxy
  • https://mikemylonakis.com/unix/docker-proxy/
  • https://docs.docker.com/config/daemon/systemd/

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/892423.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

牛客题解 | 验证IP地址

牛客题库题解题目 题目链接 题目主要信息:IPv4只有十进制数和分割点,其中数字在0-255之间,共4组,且不能有零开头的非零数,不能缺省 IPv6由8组16进制数组成,会出现a-fA-F,通过冒号分割,不可缺省,可以零开头,或者为一个单独零,每组最多4位。举一反三: 学会了本题的思…

牛客题解 | 链表中环的入口结点

牛客题库题解题目 题目链接 题目主要信息:给定一个链表,首先判断其是否有环,然后找到环的入口举一反三: 学习完本题的思路你可以解决如下题目: BM4.合并有序链表 BM5.合并k个已排序的链表 BM6.判断链表中是否有环 BM8.链表中倒数最后k个节点 BM9.删除链表的倒数第n个节点 B…

牛客题解 | 链表中环的入口结点_1

牛客题库题解题目 题目链接 描述 这是一篇针对初学者的题解,共用两种方法解决。 知识点:单链表,哈希,双指针 难度:二星题解 题目抽象:给定一个单链表,如果有环,返回环的入口结点,否则,返回nullptr 方法一:哈希法遍历单链表的每个结点 如果当前结点地址没有出现在set…

通过工艺对象的双轴同步设置

硬件导入导入GSDmlGSDml文件是从Starter导入的,所有驱动合成一个gsdml文件,需要同步的两个驱动使用的标准报文3:starter配置一些必要电机和铭牌信息即可,主要组态在工艺对象,所以电机测需要配置的不多。工艺对象配置添加主动轴,选择TO_PositioningAxis基本设置,直线运动就…

AI时代的灵魂拷问:我们真正的核心竞争力到底是什么?

“当所有人都在谈论AI+的时候,今天我想聊一点不一样的...” 上周,朋友看着我用Cursor在30分钟内完成了他过去需要两天才能完成的工作。 那一刻,一种强烈的危机感涌上心头,他问我:“如果AI可以做得和我们一样好,甚至更好,那我们存在的价值是什么?” 这恐怕是当下许多人都…

EscapeTwo

EscapeTwo题目中给出了一组账号和密码: As is common in real life Windows pentests, you will start this box with credentials for the following account: rose / KxEPkKe6R8sunmap扫一下: root@lbz-lenovo:/mnt/c/Users/86138# nmap -sC -sV -T4 10.129.221.182 Starting…

从零开始的web前端学习-css

CSS 称为层叠样式表,是一种样式表语言,用来描述 HTML 文档的呈现,是对 HTML 的美化 选择器 {属性: 属性值; }目录1CSS引入方式2选择器3区域填充4文字控制属性5复合选择器6伪类选择器(一般用于超链接)7CSS特性8背景属性9显示模式10结构伪类选择器11伪元素选择器12盒子模型-…

关于启用CDN加速域名

在微信小程序连接cos云对象存储时候,需要通过CDN加速域名,需要自己租赁域名 关于启用CDN域名加速的好处: 1. 加速内容分发 (1)全球加速CDN(内容分发网络)通过将内容缓存到全球多个节点,使用户可以从离他们最近的节点获取资源。无论用户位于何处,都能享受快速的访问速度…

redis面试知识

redis是什么介绍 redis是储存在内存中,读写很快,是基于c语言编写的,是原子性的,常用于缓存。redis可以将数据储存到磁盘中实现持久化。 redis的优点 1.基于内存,访问速度很快。 2.支持多种数据类型,如hash,String,map等 3.支持持久化 4.redis的操作是单线程的,安全 5.r…

五星红旗python代码

import turtle def draw_star(size, angle):turtle.begin_fill()for _ in range(5):turtle.forward(size)turtle.right(angle)turtle.end_fill() 初始化画布 turtle.setup(900, 600) turtle.bgcolor("red") 绘制大星 turtle.penup() turtle.goto(-400, 200) turtle.c…

蜀道山2024复现笔记

蜀道山2024复现笔记蜀道山2024复现笔记Map_maze PE32文件,ida分析,由题名可知是一道迷宫题initial函数是地图的初始化,sub_101C40是验证函数 先看验证部分 char __cdecl sub_101C40(_DWORD *a1, _DWORD *a2, int a3) {char result; // alint i; // [esp+0h] [ebp-8h]for ( i…

白话解读 Dapr 1.15:你的「微服务管家」又秀新绝活了

大家好!今天给大家带来一个好消息,Dapr(Distributed Application Runtime)1.15版本正式发布啦!对于不熟悉Dapr的朋友来说,Dapr是一个开源的、跨平台的运行时,它简化了微服务架构中的许多复杂问题,比如服务发现、配置管理、状态管理等。那么,Dapr 1.15版本都带来了哪些…