热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Nginx转发post请求变get请求

最近用nginx转发请求,但是明明是post请求,打到服务上确实get请求,body中的参数都没了,于是去查背后详细的原因。

最近用nginx转发请求,但是明明是post请求,打到服务上确实get请求,body中的参数都没了,于是去查背后详细的原因。


背景


Nginx配置


  • ip及端口:192.20.81.34:8081
  • conf关键配置:

location /select/dosomething/ {proxy_pass http://xx.xx.xx.xx:18000/select/dosomething/;}

请求连接

http://192.20.81.34:8081/select/dosomething

网上查询处理方法

1.http请求被转发至https请求引起重定向,导致post变get,显然这个和我的情况无关

2.location后连接多了个/,把/去掉就好了,如:解决 nginx 转发POST变成GET - 灰信网(软件开发博客聚合)icon-default.png?t=M276https://www.freesion.com/article/99851192529/

第二个好使!!


问题分析

虽然按照第二个把/去掉后正常了,但是那篇文章中并没有解释为什么会导致这样的现象,于是本着好奇的态度,去查了一下原因,结果如下


一、为什么post会转get?

post转get其实不是Nginx导致的!而是重定向导致的!出现这种情况,你可以在nginx的logs/access.log里面看到有两条记录。

"POST /select/dosomthing HTTP/1.1" 301 185 "-" "PostmanRuntime/7.28.4"
"GET /select/dosomthing/ HTTP/1.1" 500 389 "http://192.20.81.34:8081/select/dosomthing" "PostmanRuntime/7.28.4"

这两条记录可以看出,我是使用postman进行请求,第一条post请求nginx收到了并返回了301HTTP状态码,接着postman就发出了第二条请求,第二条请求就变成了GET请求,且请求最后加了/,变成了

http://192.20.81.34:8081/select/dosomething/

查询301的状态码对应的是永久重定向,正常情况下请求发送方接到301状态码,表示需要重定向,于是会改变请求的url再次请求,但是!有的时候会将请求强行改为get,详情可见:https://developer.mozilla.org/zh-CN/docs/Web/HTTP/Redirectionsicon-default.png?t=M276https://developer.mozilla.org/zh-CN/docs/Web/HTTP/Redirections

可以看到 

有些时候处理301状态码时,会把请求方法强行改成get,当然这个问题也被人发现,后面区分出了308状态码,这个状态码不会改变请求方式。

所以是301状态码引起了post请求变成了get请求,而整个流程就较为清晰了:


  1. nginx收到post请求
  2. 返回301状态码,并指定新的访问连接,且该连接就是原连接+/
  3. 发送方收到301状态码,将请求方式改为get,并请求新连接
  4. 服务收到get请求,body参数全部失效


二、为什么会发生重定向?

这个重定向似乎发生的非常隐蔽,似乎是触发了nginx的某种机制,从现象上看,该机制是nginx自动添加末尾斜杠的机制。

我在nginx中配置的连接是

/select/dosomething/

而我实际请求的连接是

/select/dosomething

明显(虽然排查的时候也没看出来)末尾少了一个/,这种情况下nginx会通过301重定向的方式来补全最后/,让发送方重新请求,从而完成匹配。

这个机制也有人提到过,比如:nginx url自动加斜杠问题 - 陈一风 - 博客园一、首先对比说明Nginx以下两个现象: 1. 访问的uri最后带斜杠 http://localhost/product/ >>>> 查找 product下的index页面,存https://www.cnblogs.com/jedi1995/p/11320357.html

但是我并没有在nginx官网中找到官方描述(可能仅仅是因为我没找到)


问题反思

现在整个问题流程就很清晰了


  1. 发送方请求了一个少了最后/的请求
  2. nginx返回301状态码,并返回补全/的重定向连接
  3. 发送方因为301状态码,将请求改为get请求,并按照新连接发送
  4. nginx获得新的连接,完成匹配,并转发
  5. 服务获得get请求

我在想这是不是一个bug?如果nginx返回采用308状态码是不是就能够避免这个问题了?


推荐阅读
  • 解决nginx启动报错epoll_wait() reported that client prematurely closed connection的方法
    本文介绍了解决nginx启动报错epoll_wait() reported that client prematurely closed connection的方法,包括检查location配置是否正确、pass_proxy是否需要加“/”等。同时,还介绍了修改nginx的error.log日志级别为debug,以便查看详细日志信息。 ... [详细]
  • Skywalking系列博客1安装单机版 Skywalking的快速安装方法
    本文介绍了如何快速安装单机版的Skywalking,包括下载、环境需求和端口检查等步骤。同时提供了百度盘下载地址和查询端口是否被占用的命令。 ... [详细]
  • Nginx使用(server参数配置)
    本文介绍了Nginx的使用,重点讲解了server参数配置,包括端口号、主机名、根目录等内容。同时,还介绍了Nginx的反向代理功能。 ... [详细]
  • 这是原文链接:sendingformdata许多情况下,我们使用表单发送数据到服务器。服务器处理数据并返回响应给用户。这看起来很简单,但是 ... [详细]
  • Webmin远程命令执行漏洞复现及防护方法
    本文介绍了Webmin远程命令执行漏洞CVE-2019-15107的漏洞详情和复现方法,同时提供了防护方法。漏洞存在于Webmin的找回密码页面中,攻击者无需权限即可注入命令并执行任意系统命令。文章还提供了相关参考链接和搭建靶场的步骤。此外,还指出了参考链接中的数据包不准确的问题,并解释了漏洞触发的条件。最后,给出了防护方法以避免受到该漏洞的攻击。 ... [详细]
  • Postgresql备份和恢复的方法及命令行操作步骤
    本文介绍了使用Postgresql进行备份和恢复的方法及命令行操作步骤。通过使用pg_dump命令进行备份,pg_restore命令进行恢复,并设置-h localhost选项,可以完成数据的备份和恢复操作。此外,本文还提供了参考链接以获取更多详细信息。 ... [详细]
  • 目录浏览漏洞与目录遍历漏洞的危害及修复方法
    本文讨论了目录浏览漏洞与目录遍历漏洞的危害,包括网站结构暴露、隐秘文件访问等。同时介绍了检测方法,如使用漏洞扫描器和搜索关键词。最后提供了针对常见中间件的修复方式,包括关闭目录浏览功能。对于保护网站安全具有一定的参考价值。 ... [详细]
  • Java如何导入和导出Excel文件的方法和步骤详解
    本文详细介绍了在SpringBoot中使用Java导入和导出Excel文件的方法和步骤,包括添加操作Excel的依赖、自定义注解等。文章还提供了示例代码,并将代码上传至GitHub供访问。 ... [详细]
  • 负载均衡_Nginx反向代理动静分离负载均衡及rewrite隐藏路径详解(Nginx Apache MySQL Redis)–第二部分
    nginx反向代理、动静分离、负载均衡及rewrite隐藏路径详解 ... [详细]
  • Annotation的大材小用
    为什么80%的码农都做不了架构师?最近在开发一些通用的excel数据导入的功能,由于涉及到导入的模块很多,所以开发了一个比较通用的e ... [详细]
  • Nginx使用AWStats日志分析的步骤及注意事项
    本文介绍了在Centos7操作系统上使用Nginx和AWStats进行日志分析的步骤和注意事项。通过AWStats可以统计网站的访问量、IP地址、操作系统、浏览器等信息,并提供精确到每月、每日、每小时的数据。在部署AWStats之前需要确认服务器上已经安装了Perl环境,并进行DNS解析。 ... [详细]
  • 基于PgpoolII的PostgreSQL集群安装与配置教程
    本文介绍了基于PgpoolII的PostgreSQL集群的安装与配置教程。Pgpool-II是一个位于PostgreSQL服务器和PostgreSQL数据库客户端之间的中间件,提供了连接池、复制、负载均衡、缓存、看门狗、限制链接等功能,可以用于搭建高可用的PostgreSQL集群。文章详细介绍了通过yum安装Pgpool-II的步骤,并提供了相关的官方参考地址。 ... [详细]
  • 如何在php中将mysql查询结果赋值给变量
    本文介绍了在php中将mysql查询结果赋值给变量的方法,包括从mysql表中查询count(学号)并赋值给一个变量,以及如何将sql中查询单条结果赋值给php页面的一个变量。同时还讨论了php调用mysql查询结果到变量的方法,并提供了示例代码。 ... [详细]
  • Python SQLAlchemy库的使用方法详解
    本文详细介绍了Python中使用SQLAlchemy库的方法。首先对SQLAlchemy进行了简介,包括其定义、适用的数据库类型等。然后讨论了SQLAlchemy提供的两种主要使用模式,即SQL表达式语言和ORM。针对不同的需求,给出了选择哪种模式的建议。最后,介绍了连接数据库的方法,包括创建SQLAlchemy引擎和执行SQL语句的接口。 ... [详细]
  • 31.项目部署
    目录1一些概念1.1项目部署1.2WSGI1.3uWSGI1.4Nginx2安装环境与迁移项目2.1项目内容2.2项目配置2.2.1DEBUG2.2.2STAT ... [详细]
author-avatar
ID张蕾
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有