我的编程空间,编程开发者的网络收藏夹
学习永远不晚

nginx代理出现302如何解决

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

nginx代理出现302如何解决

这期内容当中小编将会给大家带来有关nginx代理出现302如何解决,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。

用proxy_intercept_errors和recursive_error_pages代理多次302

302是HTTP协议中的一个经常被使用状态码,是多种重定向方式的一种,其语义经常被解释为“Moved Temporarily”。这里顺带提一下,现实中用到的302多为误用(与303,307混用),在HTTP/1.1中,它的语义为“Found”.

302有时候很明显,有时候又比较隐蔽。最简单的情况,是当我们在浏览器中输入一个网址A,然后浏览器地址栏会自动跳到B,进而打开一个网页,这种情况就很可能是302。

比较隐蔽的情况经常发生在嵌入到网页的播放器中。例如,当你打开一个优酷视频播放页面时,抓包观察一下就会经常发现302的影子。但由于这些url并不是直接在浏览器中打开的,所以在浏览器的地址栏看不到变化,当然,如果将这些具体的url特意挑出来复制到浏览器地址栏里,还是可以观察到的。

上一段提到了优酷。其实现在多数在线视频网站都会用到302,原因很简单,视频网站流量一般较大,都会用到CDN,区别只在于是用自建CDN还是商业CDN。而由于302的重定向语义(再重复一遍,302的语义广泛的被误用,在使用302的时候,我们很可能应该使用303或307,但后面都不再纠结这一点),可以与CDN中的调度很好的结合起来。

我们来看一个例子,打开一个网易视频播放页面,抓一下包,找到302状态的那个url。例如:

http://flv.bn.netease.com/tvmrepo/2014/8/5/P/EA3I1J05P/SD/EA3I1J05P-mobile.mp4

我们把它复制到浏览器地址栏中,会发现地址栏迅速的变为了另外一个url,这个Url是不定的,有可能为:

http://14.18.140.83/f6c00af500000000-1408987545-236096587/data6/flv.bn.netease.com/tvmrepo/2014/8/5/P/EA3I1J05P/SD/EA3I1J05P-mobile.mp4

用curl工具会更清楚的看到整个过程:

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

curl -I "http://flv.bn.netease.com/tvmrepo/2014/8/5/P/EA3I1J05P/SD/EA3I1J05P-mobile.mp4" -L

HTTP/1.1 302 Moved Temporarily

Server: nginx

Date: Mon, 25 Aug 2014 14:49:43 GMT

Content-Type: text/html

Content-Length: 154

Connection: keep-alive

NG: CCN-SW-1-5L2

X-Mod-Name: GSLB/3.1.0

Location: http://119.134.254.9/flv.bn.netease.com/tvmrepo/2014/8/5/P/EA3I1J05P/SD/EA3I1J05P-mobile.mp4

 

HTTP/1.1 302 Moved Temporarily

Server: nginx

Date: Mon, 25 Aug 2014 14:49:41 GMT

Content-Type: text/html

Content-Length: 154

Connection: keep-alive

X-Mod-Name: Mvod-Server/4.3.3

Location: http://119.134.254.7/cc89fdac00000000-1408983581-2095617481/data4/flv.bn.netease.com/tvmrepo/2014/8/5/P/EA3I1J05P/SD/EA3I1J05P-mobile.mp4

NG: CHN-SW-1-3Y1

 

HTTP/1.1 200 OK

Server: nginx

Date: Mon, 25 Aug 2014 14:49:41 GMT

Content-Type: video/mp4

Content-Length: 3706468

Last-Modified: Mon, 25 Aug 2014 00:23:50 GMT

Connection: keep-alive

Cache-Control: no-cache

ETag: "53fa8216-388e64"

NG: CHN-SW-1-3g6

X-Mod-Name: Mvod-Server/4.3.3

Accept-Ranges: bytes

可以看到,这中间经历了两次302。

先暂时将这个例子放在一边,再来说说另一个重要的术语:proxy.我们通常会戏称,某些领导是302类型的,某些领导是proxy类型的。302类型的领导,一件事情经过他的手,会迅速的转给他人,而proxy类型的领导则会参与到事情中来,甚至把事情全部做完。

回到上面的例子,如果访问一个url中途会有多个302,那如果需要用Nginx设计一个proxy,来隐藏掉中间所有的这些302,该怎么做呢?

1.原始Proxy

我们知道,Nginx本身就是一个优秀的代理服务器。因此,首先我们来架设一个Nginx正向代理,服务器IP为192.168.109.128(我的一个测试虚拟机)。

初始配置简化如下:

?

1

2

3

4

5

6

7

8

9

10

11

12

13

server {

    listen 80;

    location / {

        rewrite_by_lua '

            ngx.exec("/proxy-to" .. ngx.var.request_uri)

        ';

    }

 

    location ~ /proxy-to/([^/]+)(.*) {

        proxy_pass http://$1$2$is_args$query_string;

 

    }

}

实现的功能是,当使用

http://192.168.109.128/xxxxxx

访问该代理时,会proxy到xxxxxx所代表的真实服务器。

测试结果如下:

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

curl -I "http://192.168.109.128/flv.bn.netease.com/tvmrepo/2014/8/5/P/EA3I1J05P/SD/EA3I1J05P-mobile.mp4" -L

HTTP/1.1 302 Moved Temporarily

Server: nginx/1.4.6

Date: Mon, 25 Aug 2014 14:50:54 GMT

Content-Type: text/html

Content-Length: 154

Connection: keep-alive

NG: CCN-SW-1-5L2

X-Mod-Name: GSLB/3.1.0

Location: http://183.61.140.24/flv.bn.netease.com/tvmrepo/2014/8/5/P/EA3I1J05P/SD/EA3I1J05P-mobile.mp4

 

HTTP/1.1 302 Moved Temporarily

Server: nginx

Date: Mon, 25 Aug 2014 14:50:55 GMT

Content-Type: text/html

Content-Length: 154

Connection: keep-alive

X-Mod-Name: Mvod-Server/4.3.3

Location: http://183.61.140.20/540966e500000000-1408983655-236096587/data1/flv.bn.netease.com/tvmrepo/2014/8/5/P/EA3I1J05P/SD/EA3I1J05P-mobile.mp4

NG: CHN-ZJ-4-3M4

 

HTTP/1.1 200 OK

Server: nginx

Date: Mon, 25 Aug 2014 14:50:55 GMT

Content-Type: video/mp4

Content-Length: 3706468

Last-Modified: Mon, 25 Aug 2014 00:31:03 GMT

Connection: keep-alive

Cache-Control: no-cache

ETag: "53fa83c7-388e64"

NG: CHN-ZJ-4-3M4

X-Mod-Name: Mvod-Server/4.3.3

Accept-Ranges: bytes

可见,虽然使用proxy,但过程与原始访问没有什么区别。访问过程为,当访问

http://192.168.109.128/flv.bn.netease.com/tvmrepo/2014/8/5/P/EA3I1J05P/SD/EA3I1J05P-mobile.mp4

时,Nginx会将该请求proxy到

http://flv.bn.netease.com/tvmrepo/2014/8/5/P/EA3I1J05P/SD/EA3I1J05P-mobile.mp4

而后者马上就会返回一个302,所以Nginx作为proxy,将该302传回到客户端,客户端重新发起请求,进而重复之前的多次302.这里说明一个问题,一旦Nginx的proxy的后端返回302后,客户端即与Nginx这个proxy脱离关系了,Nginx无法起到完整的代理的作用。

2. 第1次修改

将配置文件修改为:

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

server {

    listen 80;

    location / {

        rewrite_by_lua '

            ngx.exec("/proxy-to" .. ngx.var.request_uri)

        ';

    }

 

    location ~ /proxy-to/([^/]+)(.*) {

        proxy_pass http://$1$2$is_args$query_string;

        error_page 302 = @error_page_302;

 

    }

    location @error_page_302 {

        rewrite_by_lua '

            local _, _, upstream_http_location = string.find(ngx.var.upstream_http_location, "^http:/(.*)$")

            ngx.header["zzzz"] = "/proxy-to" .. upstream_http_location

            ngx.exec("/proxy-to" .. upstream_http_location);

        ';

 

    }

}

与上面的区别在于,使用了一个error_page,目的是当发现proxy的后端返回302时,则用这个302的目的location继续proxy,而不是直接返回给客户端。并且这个逻辑里面包含着递归的意思,一路跟踪302,直到最终返回200的那个地址。测试结果如下:

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

curl -I "http://192.168.109.128/flv.bn.netease.com/tvmrepo/2014/8/5/P/EA3I1J05P/SD/EA3I1J05P-mobile.mp4" -L

HTTP/1.1 302 Moved Temporarily

Server: nginx/1.4.6

Date: Mon, 25 Aug 2014 15:01:17 GMT

Content-Type: text/html

Content-Length: 154

Connection: keep-alive

NG: CCN-SW-1-5L2

X-Mod-Name: GSLB/3.1.0

Location: http://183.61.140.24/flv.bn.netease.com/tvmrepo/2014/8/5/P/EA3I1J05P/SD/EA3I1J05P-mobile.mp4

 

HTTP/1.1 302 Moved Temporarily

Server: nginx

Date: Mon, 25 Aug 2014 15:01:17 GMT

Content-Type: text/html

Content-Length: 154

Connection: keep-alive

X-Mod-Name: Mvod-Server/4.3.3

Location: http://183.61.140.20/a90a952900000000-1408984277-236096587/data1/flv.bn.netease.com/tvmrepo/2014/8/5/P/EA3I1J05P/SD/EA3I1J05P-mobile.mp4

NG: CHN-ZJ-4-3M4

 

HTTP/1.1 200 OK

Server: nginx

Date: Mon, 25 Aug 2014 15:01:17 GMT

Content-Type: video/mp4

Content-Length: 3706468

Last-Modified: Mon, 25 Aug 2014 00:31:03 GMT

Connection: keep-alive

Cache-Control: no-cache

ETag: "53fa83c7-388e64"

NG: CHN-ZJ-4-3M4

X-Mod-Name: Mvod-Server/4.3.3

Accept-Ranges: bytes

可见,本次修改仍然没有成功!

为什么呢?分析一下,我们在@error_page_302这个location里已经加了一个头部打印语句,可是在测试中,该头部并没有打出来,可见流程并没有进入到@error_page_302这个location。

原因在于

?

1

error_page 302 = @error_page_302;

error_page默认是本次处理的返回码。作为proxy,本次处理,只要转发上游服务器的响应成功,应该状态码都是200.即,我们真正需要检查的,是proxy的后端服务器返回的状态码,而不是proxy本身返回的状态码。查一下Nginx的wiki,proxy_intercept_errors指令正是干这个的:

?

1

2

3

4

5

Syntax: proxy_intercept_errors on | off;

Default: 

proxy_intercept_errors off;

Context:  http, server, location

Determines whether proxied responses with codes greater than or equal to 300 should be passed to a client or be redirected to nginx for processing with the error_page directive.

3. 第二次修改

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

server {

    listen 80;

    proxy_intercept_errors on;

    location / {

        rewrite_by_lua '

            ngx.exec("/proxy-to" .. ngx.var.request_uri)

        ';

    }

    location ~ /proxy-to/([^/]+)(.*) {

        proxy_pass http://$1$2$is_args$query_string;

        error_page 302 = @error_page_302;

 

    }

    location @error_page_302 {

        rewrite_by_lua '

            local _, _, upstream_http_location = string.find(ngx.var.upstream_http_location, "^http:/(.*)$")

            ngx.header["zzzz"] = "/proxy-to" .. upstream_http_location

            ngx.exec("/proxy-to" .. upstream_http_location);

        ';

    }

}

与上一次修改相比,区别仅仅在于增加了一个proxy_intercept_errors指令。测试结果如下:

?

1

2

3

4

5

6

7

8

curl -I "http://192.168.109.128/flv.bn.netease.com/tvmrepo/2014/8/5/P/EA3I1J05P/SD/EA3I1J05P-mobile.mp4" -L

HTTP/1.1 302 Moved Temporarily

Server: nginx/1.4.6

Date: Mon, 25 Aug 2014 15:05:54 GMT

Content-Type: text/html

Content-Length: 160

Connection: keep-alive

zzzz: /proxy-to/183.61.140.24/flv.bn.netease.com/tvmrepo/2014/8/5/P/EA3I1J05P/SD/EA3I1J05P-mobile.mp4

这次更神奇了,直接返回一个302状态完事,也不继续跳转了。

问题出在,虽然第一次302,请求成功的进入到@error_page_302,但后续的error_page指令却没起作用。也就是说,error_page只检查了第一次后端返回的状态码,而没有继续检查后续的后端状态码。

查一下资料,这个时候,另一个指令 recursive_error_pages就派上用场了。

4. 第3次修改

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

server {

    listen 80;

    proxy_intercept_errors on;

    recursive_error_pages on;

    location / {

        rewrite_by_lua '

            ngx.exec("/proxy-to" .. ngx.var.request_uri)

        ';

    }

    location ~ /proxy-to/([^/]+)(.*) {

        proxy_pass http://$1$2$is_args$query_string;

        error_page 302 = @error_page_302;

 

    }

    location @error_page_302 {

        rewrite_by_lua '

            local _, _, upstream_http_location = string.find(ngx.var.upstream_http_location, "^http:/(.*)$")

            ngx.header["zzzz"] = "/proxy-to" .. upstream_http_location

            ngx.exec("/proxy-to" .. upstream_http_location);

        ';

    }

}

与上一次相比,仅仅增加了recursive_error_pages on这条指令。测试结果如下:

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

curl -I "http://192.168.109.128/flv.bn.netease.com/tvmrepo/2014/8/5/P/EA3I1J05P/SD/EA3I1J05P-mobile.mp4" -L

HTTP/1.1 200 OK

Server: nginx/1.4.6

Date: Mon, 25 Aug 2014 15:09:04 GMT

Content-Type: video/mp4

Content-Length: 3706468

Connection: keep-alive

zzzz: /proxy-to/14.18.140.83/f48bad0100000000-1408984745-236096587/data6/flv.bn.netease.com/tvmrepo/2014/8/5/P/EA3I1J05P/SD/EA3I1J05P-mobile.mp4

Last-Modified: Mon, 25 Aug 2014 00:21:07 GMT

Cache-Control: no-cache

ETag: "53fa8173-388e64"

NG: CHN-MM-4-3FE

X-Mod-Name: Mvod-Server/4.3.3

Accept-Ranges: bytes

可见,Nginx终于成功的返回200了。此时,Nginx才真正起到了一个Proxy的功能,隐藏了一个请求原本的多个302链路,只返回客户端一个最终结果。

上述就是小编为大家分享的nginx代理出现302如何解决了,如果刚好有类似的疑惑,不妨参照上述分析进行理解。如果想知道更多相关知识,欢迎关注编程网行业资讯频道。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

nginx代理出现302如何解决

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

nginx代理出现302如何解决

这期内容当中小编将会给大家带来有关nginx代理出现302如何解决,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。用proxy_intercept_errors和recursive_error_pages代
2023-06-08

nginx代理后出现503如何解决

这篇文章主要介绍了nginx代理后出现503如何解决的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇nginx代理后出现503如何解决文章都会有所收获,下面我们一起来看看吧。问题:配置serve_name后,并且
2023-07-02

nginx反向代理出现502如何解决

当nginx作为反向代理时,出现502错误通常是因为代理的目标服务器无法提供正确的响应给nginx。以下是一些可能的解决方法:1. 检查目标服务器是否正在运行,并且端口是否正确。2. 检查目标服务器的防火墙设置,确保nginx服务器可以访问
2023-09-16

nginx反向代理不生效如何解决

出现nginx反向代理不生效的问题,可能有以下几个原因:1. 配置错误:请确认nginx配置文件中的反向代理配置是否正确,包括upstream配置和location配置。2. 服务未启动:请确认被代理的后端服务是否已经启动,并监听了正确的端
2023-08-24

nginx反向代理经常超时如何解决

nginx反向代理经常超时的问题可以通过以下几种方式来解决:1. 增加超时时间:可以通过修改nginx的配置文件,增加proxy_connect_timeout和proxy_read_timeout等参数的值,使得超时时间变长。例如:```
2023-08-24

nginx做反向代理后无法跳转如何解决

当使用Nginx作为反向代理时,可能会遇到一些跳转问题。以下是一些常见的解决方法:1. 配置proxy_redirect指令:在Nginx的配置文件中,使用proxy_redirect指令来修改响应头中的Location字段,使其指向正确的
2023-09-09

nginx反向代理请求参数丢失如何解决

如果在使用nginx作为反向代理时发生了请求参数丢失的情况,可能是由于配置不正确或者代理服务器的限制导致的。以下是一些解决方法:检查nginx配置文件:确保代理服务器的配置正确,特别是在location块中的proxy_pass参数。确保p
nginx反向代理请求参数丢失如何解决
2024-02-29

windows出现007b蓝屏代码如何解决

本篇内容主要讲解“windows出现007b蓝屏代码如何解决”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“windows出现007b蓝屏代码如何解决”吧!出现007
2023-02-02

windows出现错误代码0xt000000e如何解决

这篇“windows出现错误代码0xt000000e如何解决”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“windows出现
2023-07-01

win7出现0x80070002错误代码如何解决

这篇文章主要介绍“win7出现0x80070002错误代码如何解决”,在日常操作中,相信很多人在win7出现0x80070002错误代码如何解决问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”win7出现0x8
2023-07-01

win11出现0x80070002错误代码如何解决

今天小编给大家分享一下win11出现0x80070002错误代码如何解决的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。0x8
2023-07-02

电脑蓝屏出现代码0x00000024如何解决

这篇文章主要介绍了电脑蓝屏出现代码0x00000024如何解决的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇电脑蓝屏出现代码0x00000024如何解决文章都会有所收获,下面我们一起来看看吧。首先我们说一下这个
2023-06-27

win7出现错误代码0xc00000d如何解决

错误代码0xc00000d是Windows 7系统启动错误的一种常见错误代码,表示系统无法找到或加载操作系统文件。以下是一些解决方案:1. 修复启动文件:使用Windows 7安装光盘或USB启动盘,进入系统恢复环境,选择修复启动文件选项。
2023-08-23

windows谷歌浏览器代理服务器出现问题如何解决

今天小编给大家分享一下windows谷歌浏览器代理服务器出现问题如何解决的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。解决方
2023-07-01

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录