Nginx 不会在 Docker 中重新解析 DNS 名称

问题描述

我正在运行 Nginx 作为 docker-compose 模板的一部分。 在 Nginx 配置中,我指的是其他服务的 docker 主机名(例如 backendui)。 在我做那个技巧之前,这很好用:

docker stop backend
docker stop ui
docker start ui
docker start backend

使后端和 ui 容器交换 IP 地址(docker 在将 CIDR 中可用的下一个 IP 提供给每个新请求者的基础上提供私有网络 IP)。执行的这 4 个命令模仿了一些罕见的情况,即两个上游容器同时重新启动但 Nginx 容器没有。另外,我相信,在基于 Kubernetes 的集群上运行 pod 时,这应该是一种非常常见的情况。

现在 Nginxbackend 主机解析为 ui 的 IP,将 ui 解析为后端的 IP。 重新加载 Nginx 的配置确实有帮助 (Nginx -s reload)。 另外,如果我从 Nginx 容器内执行 nslookup - IP 总是可以正确解析。

因此,这将问题隔离为围绕 DNS 缓存的纯 Nginx 问题。

我尝试过的东西:

  1. 我在 Nginx 配置中的 http {} 块下设置了解析器:
resolver 127.0.0.11 ipv6=off valid=10s;
  1. 互联网上人们提出的最常见的解决方案是在代理传递中使用变量(这有助于防止 Nginx 在启动时解析和缓存 DNS 记录) - 根本没有任何区别:
server {
  <...>
  set $mybackend "backend:3000";
  location /backend/ {
    proxy_pass http://$mybackend;
  }
}
  1. 尝试将解析器行添加到位置本身
  2. 尝试使用 map 在 http{} 块级别设置变量:
http {  
  map "" $mybackend {
    default backend:3000;
  }
  server {
   ...
  }
}
  1. 尝试将 Nginx (https://hub.docker.com/r/openresty/openresty/) 的 openresty fork 与 resolver local=true 一起使用

所有解决方案都没有产生任何效果。仅当我在容器内重新加载 Nginx 配置或手动重新启动容器时,才会擦除 DNS 缓存。

我目前的解决方法是使用 docker-compose.yml 中声明的静态 docker 网络。但这也有它的缺点。

使用的 Nginx 版本:1.20.0(目前最新) 使用的 Openresty 版本:1.13.6.1 和 1.19.3.1(目前最新)

希望有任何想法

解决方法

可能是因为 nginx 的上游服务器 DNS 解析器只适用于商业版本,nginx plus?

https://www.nginx.com/products/nginx/load-balancing/#service-discovery

相关问答

Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其...
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。...
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbc...