一次 Nginx proxy_set_header 故障问题解析和延升
本文会先由一个问题引入,然后再进行多种情况进行分析。
一、问题和排查步骤
1.1 问题基本信息
我们应用程序从代码层面收到的 Header 中的 Host 的值是 upstream 的 名称。 我们程序是需要获取到实际的值。所以这里存在一个问题。
我们先看看我们的 nginx 配置。
upstream open-hz8443{
server 10.60.6.184:8000 max_fails=1 fail_timeout=3s weight=10;
}
server{
server_name 192.168.80.132;
listen 80;
proxy_set_header Host $host;
proxy_set_header X-Real-IP $remote_addr;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_connect_timeout 3s;
proxy_read_timeout 120s;
proxy_send_timeout 120s;
proxy_next_upstream error timeout invalid_header http_404 http_502 http_504 http_500;
location / {
proxy_pass http://open-hz8443;
}
location ^~ /wss/v1
{
proxy_pass http://open-hz8443;
proxy_set_header Connection "upgrade";
proxy_set_header Upgrade $http_upgrade;
tcp_nodelay on;
}
}
我们请求 http://192.168.80.132/wss/v1, 我们可以在后端(10.60.6.184:8000)获取到 Host 的值为 open-hz8443。
这个是我们做了一个 脚本,用于获取请求的所有的数据的。 脚本具体内容在文末。
Connection: upgrade
Host: open-hz8443 # 获取到的Host 是 open-hz8443
Cache-Control: max-age=0
Upgrade-Insecure-Requests: 1
User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/87.0.4280.88 Safari/537.36 FS
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.9
Accept-Encoding: gzip, deflate
Accept-Language: zh-CN,zh;q=0.9
10.210.20.43 - - [04/Jan/2021 15:54:46] "GET /wss/v1 HTTP/1.0" 200 -
1.2 问题解析
首先这里有知识点:
我们在 Server 配了
proxy_set_header Host $host;, 我们在 location 是只配置了proxy_set_header Connection "upgrade";
proxy_set_header Upgrade $http_upgrade;
我们的 Host 最初我认为是会继承 Server 配置的
proxy_set_header Host $host;, 但是明显是没有继承的,而是直接拿的upstream的名称。说明没有继承,那么这里是有一个什么规则? 我们往下看。查询
Nginx官方文档。http://nginx.org/en/docs/http/ngx_http_proxy_module.html#proxy_set_header
Allows redefining or appending fields to the request header passed to the proxied server. The value can contain text, variables, and their combinations. These directives are inherited from the previous configuration level if and only if there are no proxy_set_header directives defined on the current level. By default, only two fields are redefined:
大概意思就是当我们没有设置
proxy_set_header才会从上一层级继承,当我们设置了proxy_set_header,也就意味着我们不从上面 server 的proxy_set_header Host $host;进行继承。那为什么会显示
open-hz8443, 是因为有一个默认值proxy_set_header Host $proxy_host;
这个
proxy_host,- 当我们设置了
upstream的话,也就是上面的例子$proxy_host就是 upstream的名称值. - 当我们直接使用的
proxy_pass http://10.60.6.184:8000的话,那么$proxy_host表示的就是10.60.6.184:8000
- 当我们设置了
1.3、解决办法
在 location ^~ /wss/v1 下面增加配置 proxy_set_header Host $host;。
location ^~ /wss/v1
{
proxy_pass http://open-hz8443;
proxy_set_header Host $host;
proxy_set_header Connection "upgrade";
proxy_set_header Upgrade $http_upgrade;
tcp_nodelay on;
}
二、扩展-各种情况对比
默认两项
proxy_set_header Host $proxy_host;
proxy_set_header Connection close;
proxy_set_header 其他项等
proxy_set_header Connection "upgrade";
proxy_set_header Upgrade $http_upgrade;
| Server 设置 proxy_set_header 其他项 | Server 设置 proxy_set_header 默认两项 | location 设置 proxy_set_header 默认两项 | location 设置 proxy_set_header 其他项 | 默认值生效 | 默认值不生效 |
|---|---|---|---|---|---|
| 1 | 1 | 1 | 1 | 1 继承location | |
| 1 | 1 | 0 | 1 | 1(不继承Server) | |
| 1 | 1 | 1 | 0 | 1 继承location | |
| 1 | 1 | 0 | 0 | 1 继承server | |
| 1 | 0 | 1 | 0 | 1 继承location | |
| 1 | 0 | 0 | 0 | 1 默认 | |
| 1 | 0 | 0 | 11 | 1 默认 | |
| 1 | 0 | 1 | 1 | 1 继承location | |
| 0 | 1 | 0 | 0 | 1 继承server | |
| 0 | 1 | 0 | 1 | 1 默认 | |
| 0 | 1 | 1 | 0 | 1 继承location | |
| 0 | 1 | 1 | 1 | 1 继承location | |
| 0 | 0 | 0 | 1 | 1 默认 | |
| 0 | 0 | 1 | 0 | 1 继承location | |
| 0 | 0 | 1 | 1 | 1 继承location | |
| 0 | 0 | 0 | 0 | 1 默认 |
总结
- location 设置了 proxy_set_header 就不继承,但继承默认值,默认值优先级低于 location设置。
- location 未设置了proxy_set_header ,就往上继承,直到默认值。
只要调用了 proxy_set_header,并没有设置 host 和 connection ,默认重写host、connection两个头。
三、扩展 ->脚本
proxy_set_header $host $proxy_host $http_host 各个变量含义
记录一次 Nginx 配置 proxy_pass 后 返回404问题
python 获取请求所有数据信息脚本
#!/usr/bin/env python
import SimpleHTTPServer
import SocketServer
PORT = 8000
class GetHandler(SimpleHTTPServer.SimpleHTTPRequestHandler):
def do_GET(self):
print(self.headers)
self.send_response(200, "")
def do_POST(self):
print(self.headers)
content_length = self.headers.getheaders('content-length')
length = int(content_length[0]) if content_length else 0
print(self.rfile.read(length))
self.send_response(200, "")
Handler = GetHandler
httpd = SocketServer.TCPServer(("", PORT), Handler)
httpd.serve_forever()
一次 Nginx proxy_set_header 故障问题解析和延升的更多相关文章
- Nginx(六):配置解析之location解析
nginx成为非常流行的代理服务软件,最根本的原因也许是在于其强悍性能.但还有一些必要的条件,比如功能的完整,配置的易用,能够解决各种各样的实际需求问题,这些是一个好的软件的必备特性. 那么,今天我们 ...
- Nginx缓存了DNS解析造成后端不通--代理
文章转载自:https://segmentfault.com/a/1190000022365954 1 问题现象 我们使用 Nginx 的时候,经常会用到 Proxy 功能,为了方便管理,后端站点或者 ...
- Nginx重要结构request_t解析之http请求的获取
请在文章页面明显位置给出原文连接,否则保留追究法律责任的权利. 本文主要参考为<深入理解nginx模块开发与架构解析>一书,处理用户请求部分,是一篇包含作者理解的读书笔记.欢迎指正,讨论. ...
- nginx源代码分析--配置文件解析
ngx-conf-parsing 对 Nginx 配置文件的一些认识: 配置指令具有作用域,分为全局作用域和使用 {} 创建其他作用域. 同一作用域的不同的配置指令没有先后顺序:同一作用域能否使用同样 ...
- nginx文件类型错误解析漏洞
漏洞介绍:nginx是一款高性能的web服务器,使用非常广泛,其不仅经常被用作反向代理,也可以非常好的支持PHP的运行.80sec发现 其中存在一个较为严重的安全问题,默认情况下可能导致服务器错误的将 ...
- (转)Web服务器磁盘满故障深入解析
Web服务器磁盘满故障深入解析 原文:http://blog.51cto.com/oldboy/612351 ############################################# ...
- 再提供一种解决Nginx文件类型错误解析漏洞的方法
[文章作者:张宴 本文版本:v1.2 最后修改:2010.05.24 转载请注明原文链接:http://blog.zyan.cc/nginx_0day/] 注:2010年5月23日14:00前阅读本文 ...
- Nginx入门篇(二)之Nginx部署与配置文件解析
一.Nginx编译安装 ()查看系统环境 [root@localhost tools]# cat /etc/redhat-release CentOS Linux release (Core) [ro ...
- 附002.Nginx代理相关模块解析
一 ngx_http_proxy_module模块 1.1 proxy_pass配置 proxy_pass URL; Context: location, if in location, limit_ ...
随机推荐
- PyQt(Python+Qt)学习随笔:QListWidget获取指定行对应项的item()方法
老猿Python博文目录 专栏:使用PyQt开发图形界面Python应用 老猿Python博客地址 在列表部件中,可以通过item方法获取指定行对应的项,语法如下: QListWidgetItem i ...
- 第九章、Qt Designer可视化设计界面布局组件介绍
老猿Python博文目录 专栏:使用PyQt开发图形界面Python应用 老猿Python博客地址 一.引言 在Qt Designer中,在左边部件栏的提供了界面布局相关部件,如图: 可以看到共包含有 ...
- print(end="\r") 滚动输出到屏幕
for i in range(10000): print(i, end = "\r") print(end="\r") 滚动输出到屏幕
- 【Alpha冲刺阶段】Day 7
[Alpha冲刺阶段]Scrum Meeting Daily7 1.会议简述 会议开展时间 2020/5/28 8:00-8:30 PM 会议基本内容摘要 讨论合并测试问题 参与讨论人员 项目全体 ...
- rocketMq指定broker ip地址,适合解决云主机部署问题
在工作中遇到了一个这个问题,就是我们rocketmq是部署在云主机上的 但是我们的开发同事在自己的电脑连接rocketmq链接不上 报错显示Caused by: org.apache.rocket ...
- Vue项目上线环境部署,项目优化策略,生成打包报告,及上线相关配置
Node.js简介 Node.js是一个基于Chrome V8引擎的JavaScript运行环境,用来方便快速地搭建易于扩展的网络应用.Node.js使用了一个事件驱动.非阻塞式I/O的模型,使其轻量 ...
- 自搭建jetbrains系列ide授权服务器
1.下载 LicenseServer 地址:https://mega.nz/#!7B5UVY6b!Hae2ceTBPIrTowQN0sV9fQ5lGOKzGxas2ug02RZAdGU,里面有不同的服 ...
- sqli-labs less11-12(post型union注入)
less-11 post型union注入 过程: 输入admin admin 判断username password的闭合方式 对username查字段个数 ' union select 1,2# ' ...
- 图的遍历BFS
图的遍历BFS 广度优先遍历 深度优先遍历 可以进行标记 树的广度优先遍历,我们用了辅助的队列 bool visited[MAX_VERTEX_NUM] //访问标记数组 //广度优先遍历 void ...
- MySQL锁(一)全局锁:如何做全库的逻辑备份?
数据库锁设计的初衷是处理并发问题,这也是数据库与文件系统的最大区别. 根据加锁的范围,MySQL里大致可以分为三种锁:全局锁.表锁和行锁.接下来我们会分三讲来介绍这三种锁,今天要讲的是全局锁. 全局锁 ...