Proxying multiple 302s with proxy_intercept_errors and recursive_error_pages 302 is a frequently used status code in the HTTP protocol. It is one of multiple redirection methods, and its semantics is often interpreted as "Moved Temporarily". By the way, 302 is often used in reality due to misuse (mixed with 303 and 307). In HTTP/1.1, its semantics is "Found". 302 is sometimes very obvious and sometimes more hidden. The simplest case is when we enter a URL A in the browser, and the browser address bar automatically jumps to B, and then opens a web page. This situation is most likely 302. More subtle situations often occur in players embedded in web pages. For example, when you open a Youku video playback page, you will often find the shadow of 302 by capturing the packet and observing it. However, since these URLs are not opened directly in the browser, no changes can be seen in the browser's address bar. Of course, if you specifically pick out these specific URLs and copy them into the browser address bar, you can still observe them. Youku was mentioned in the previous paragraph. In fact, most online video websites now use 302. The reason is very simple. Video websites generally have large traffic and will use CDN. The only difference is whether to use self-built CDN or commercial CDN. And because of the redirect semantics of 302 (I repeat, the semantics of 302 is widely misused. When using 302, we should probably use 303 or 307, but we will not dwell on this later), it can be well combined with the scheduling in CDN. Let's take a look at an example. Open a NetEase video playback page, grab the package, and find the URL with a 302 status. For example:
If we copy it into the browser address bar, we will find that the address bar quickly changes to another URL. This URL is uncertain and may be:
Using the curl tool will show the whole process more clearly: 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 As you can see, there were two 302 errors in the process. Let's put this example aside for now and talk about another important term: proxy. We often jokingly say that some leaders are 302 type and some leaders are proxy type. A 302 type leader will quickly pass on a task to others once it passes through his hands, while a proxy type leader will participate in the task and even complete it all. Back to the example above, if there are multiple 302s when accessing a URL, what should we do if we need to design a proxy with Nginx to hide all these 302s in the middle? 1. Original Proxy We know that Nginx itself is an excellent proxy server. Therefore, first let's set up an Nginx forward proxy with a server IP of 192.168.109.128 (one of my test virtual machines). The initial configuration is simplified as follows: 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; } } The function achieved is that when using
When accessing the proxy, you will be proxied to the real server represented by xxxxxx. The test results are as follows: 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 It can be seen that although a proxy is used, the process is no different from the original access. The access process is as follows:
When Nginx proxies the request to
The latter will immediately return a 302, so Nginx, as a proxy, will pass the 302 back to the client, and the client will re-initiate the request, and then repeat the previous 302. This illustrates a problem. Once the backend of Nginx's proxy returns 302, the client is disconnected from the Nginx proxy, and Nginx cannot play the role of a complete proxy. 2. First revision Modify the configuration file to: 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); '; } } The difference from the above is that an error_page is used. The purpose is that when it is found that the proxy backend returns 302, the destination location of this 302 will continue to proxy, instead of returning it directly to the client. And this logic contains the meaning of recursion, tracking 302 all the way until it finally returns to the address of 200. The test results are as follows: 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 It can be seen that this modification is still unsuccessful! Why? After analyzing, we have added a header print statement in the @error_page_302 location, but in the test, the header is not printed out, which shows that the process has not entered the @error_page_302 location. The reason is error_page 302 = @error_page_302; By default, error_page is the return code of this process. As a proxy, in this processing, as long as the response of the upstream server is forwarded successfully, the status code should be 200. That is, what we really need to check is the status code returned by the proxy's backend server, not the status code returned by the proxy itself. Looking at the Nginx wiki, the proxy_intercept_errors directive does exactly this: 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. Second revision 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); '; } } Compared with the previous modification, the only difference is the addition of a proxy_intercept_errors directive. The test results are as follows: 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 This time it's even more magical. It directly returns a 302 status and doesn't jump any further. The problem is that although the first 302 request successfully enters @error_page_302, the subsequent error_page instructions do not work. That is to say, error_page only checks the status code returned by the backend for the first time, and does not continue to check subsequent backend status codes. Check the information, at this time, another command recursive_error_pages comes in handy. 4. Third revision 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); '; } } Compared with the last time, only the recursive_error_pages on instruction was added. The test results are as follows: 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 It can be seen that Nginx finally successfully returned 200. At this point, Nginx truly plays the role of a Proxy, hiding the multiple 302 links of a request and returning only one final result to the client. 5. Summary In summary, by using the proxy_pass, error_page, proxy_intercept_errors, and recursive_error_pages instructions together, you can hide the redirect details of a request from the client and directly return a final result with a status code of 200 to the user. The above is the full content of this article. I hope it will be helpful for everyone’s study. I also hope that everyone will support 123WORDPRESS.COM. You may also be interested in:
|
<<: Detailed explanation of common for loop in JavaScript statements
>>: How to quickly log in to MySQL database without password under Shell
Repetition: Repeat certain page design styles thr...
count(*) accomplish 1. MyISAM: Stores the total n...
Table of contents Effects Documentation first ste...
Table of contents 1. Project environment: 2: DNS ...
This article example shares the specific code of ...
1. Inline reference: used directly on the label, ...
In summary: 1. Consider performance when designin...
This article uses examples to describe the operat...
Because the project needs to use https service, I...
Table of contents Some basic configuration About ...
I recently used nginx in a project, and used Java...
[LeetCode] 177.Nth Highest Salary Write a SQL que...
When using apt-get to install, it will be very sl...
The HTML structure is as follows: The CCS structu...
mysql basic data types Overview of common MySQL d...