当前位置: 首页 > news >正文

【Docker】docker-compose中的nginx为何突然访问不到服务了?

docker-compose中的nginx为何突然访问不到服务了?

我使用docker-compose的方式启动了一个Nginx、服务A、服务B,nginxconfig内部是填写的服务名,
在重建服务A之后,有时就会出现服务A无法访问的问题,此时重启一下nginx就可以访问到了。

场景复现准备

目录结构
docker-repro/
├── docker-compose.yml
├── nginx/
│   └── nginx.conf
└── service-a/├── Dockerfile└── app.py

1. 编写 docker-compose.yml

在根目录 docker-repro/docker-compose.yml 中写入:

version: '3.8'services:nginx:image: nginx:latestvolumes:- ./nginx/nginx.conf:/etc/nginx/nginx.conf:roports:- "8080:80"depends_on:- service_aservice_a:build: ./service-acontainer_name: service_aservice_b:image: busyboxcommand: ["sh", "-c", "while true; do echo B; sleep 5; done"]container_name: service_b
  • service_a:一个简单的 Python Flask 服务,返回 “Hello A”
  • service_b:一个不停输出字符 B 的占位容器
  • nginx:反向代理到 service_a,端口转发到本机 8080

2. 编写 Service A

docker-repro/service-a/ 下:

app.py
from flask import Flask
app = Flask(__name__)@app.route('/')
def hello():return 'Hello A\n'if __name__ == '__main__':# 监听 5000 端口app.run(host='0.0.0.0', port=5000)
Dockerfile
FROM dockerpull.cn/library/python:3.9-slim
WORKDIR /app
COPY app.py /app/
RUN pip install flask  -i https://mirrors.aliyun.com/pypi/simple/
EXPOSE 5000
CMD ["python", "app.py"]

3. 编写 Nginx 配置

docker-repro/nginx/nginx.conf 中:

worker_processes  1;events {worker_connections  1024;
}http {upstream service_a {server service_a:5000;}server {listen 80;location / {proxy_pass http://service_a;}}
}

注意:这里直接在 upstream 里写了 service_a:5000,Nginx 启动时只解析一次。


4. 启动服务并验证启动成功

在项目根目录运行:

docker-compose up -d --build

待启动完成之后执行:

curl -i http://localhost:8080

问题复现

  • 先查看下目前service_a的IP
docker inspect -f '{{range .NetworkSettings.Networks}}{{.IPAddress}}{{end}}' service_a
销毁并重新创建 Service A
docker-compose stop service_a
docker-compose rm -f service_a
docker-compose up -d service_a

如果你是这样就开始实验了,那么你大概率会发现 IP 没变,服务正常访问,本次实验就没有找出原因。
因为 Docker 的网络默认会“填补空缺”——只要那个槽位(IP)在你重建前没有被别的容器占用,
它就会再分配给新的 service_a。
想要让它真正拿到一个不同的 IP,你需要“先占坑”。

正确复现的做法是:

docker-compose stop service_a
docker-compose rm -f service_a# 使用一个临时的容器占坑
docker run -d --name dummy --network docker-repro_default busybox sleep 3600docker-compose up -d service_a

这样你再次执行查看ip命令就能发现,IP变了,服务访问不到了,问题复现出来了。

总结以下问题的核心:
Nginx 默认只解析服务名一次,DNS 结果缓存不变。
Docker 中服务 A 重建后 IP 会变,但 Nginx 仍然访问旧 IP,导致连接失败。

解决方案

解决方案有很多,在这里使用较为简单的一种方式,仅需要修改nginx配置文件。

worker_processes  1;events {worker_connections  1024;
}http {resolver 127.0.0.11 valid=5s;server {listen 80;location / {set $backend "service_a:5000";proxy_pass http://$backend;proxy_set_header Host $host;proxy_set_header X-Real-IP $remote_addr;proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;}}
}
配置项作用
resolver 127.0.0.11使用 Docker 内置 DNS 进行域名解析
valid=5sDNS 记录最多缓存 5 秒,之后会重新解析
set $backend动态构造代理地址,触发每次请求前的 DNS 查询
proxy_pass http://$backend代理到动态地址
proxy_set_header添加常用的请求头,方便后端服务识别请求来源
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.dtcms.com/a/255479.html

相关文章:

  • 【CS创世SD NAND征文】STM32户外无线终端管理设备的数据存储方案
  • 拼多多消息对接、支付服务策略(策略+工厂)
  • supervisor /usr/bin/dotnet: cannot execute binary file
  • cili3d笔记20 正交投影3d重建笔记1
  • GoFrame的Gtoken--基于电商项目以及GPT-o3的辅助解惑
  • 如何轻松地将联系人从 iPhone 转移到 iPhone?
  • window显示驱动开发—输出合并器阶段
  • django FileSystemStorage is located outside of the base path component
  • 【前端隐蔽 Bug 深度剖析:SVG 组件复用中的 ID 冲突陷阱】
  • QT的一些介绍
  • Pinia在多步骤表单中的实践应用
  • DDoS防护体系构建——从基础限速到智能调度
  • C++容器之 forward_list (单向链表)使用说明
  • 层级冲突的处理,弹窗生成遮罩问题
  • http通信测试,模拟客户端
  • 力扣-169.多数元素
  • 三种经典算法无人机三维路径规划对比(SMA、HHO、GWO三种算法),Matlab代码实现
  • 医学图像处理期末复习
  • Ruoyi(若依)整合websocket实现信息推送功能(消息铃铛)
  • Logback示例解析
  • 云徙科技 OMS:让订单管理变得轻松又高效
  • python高校工作室管理系统
  • 【服务器R环境架构】基于 micromamba下载 R 库包
  • linux的基本运维
  • 【网络安全】文件上传型XSS攻击解析
  • 音视频合并避坑指南,封装过程中的解决办法
  • 云主机远程连接与防火墙安全配置指南:协同防护是关键
  • Web Splats
  • LeetCode 2942.查找包含给定字符的单词
  • 短剧CPS分销系统开发 - 专业内容变现与渠道拓展解决方案