nginx配置url转发

worker_processes  1;

events {
    worker_connections  1024;
}

http {
    include       mime.types;
    default_type  application/octet-stream;

    sendfile        on;

    keepalive_timeout  65;

    server {
        listen       80;
     
        location /a {
			rewrite ^/a/(.*)$ /$1 break;
			proxy_pass http://127.0.0.1:8080;
        }
		 location /b {
			rewrite ^/b/(.*)$ /$1 break;
			proxy_pass http://127.0.0.1:8081;
        }
		
    }
}

nginx 的 rewrite

其实nginx的rewrite模块是在日常nginx配置中十分常用的模块,应该是除了 real_ip 模块、匹配时用到的find_config模块和前端资源配置的 static 模块外的最常用的了吧。

Nginx的处理顺序
nginx的处理顺序是11 个模块,处在realip模块后的就是 rewrite 模块了。而rewrite的模块中的return一旦执行就不会再处理接下来的模块了。而rewrite 分两部分。在 postread阶段,rewrite -> find_config -> rewrite 这样去执行的。

Rewrite 模块

return 指令

rewrite 模块提供了一个指令 return 。
return 的语法可以是

return code [text];
return code URL;
return URL;

而再返回状态码上。

  • Nginx自定义
  • 444 关闭连接 (客户端是收不到这样的返回码)
  • HTTP 1.0 标准
  • 301 http1.0 永久重定向 (会将 post方法变成 get)
  • 302 临时重定向,禁止被缓存 (会将post方法变成 get)
  • HTTP 1.0 标准
  • 303 临时重定向,允许改变方法,禁止被缓存
  • 307 临时重定向,不允许改变方法,禁止被缓存
  • 308 永久重定向,不允许改变方法。

error_page 指令

error_page 指令是在发现错误返回码的时候重新定义返回的页面.
这里需要知道 return 指令会先于 error_page 执行。如果return指令被执行到了,其他的指令都将无法被执行。

1. error_page 404 /404.html;
2. error_page 500 502 503 504 /50x.html;
3. error_page 404 =200 /empty.gif;  (返回码重新置为200)
4. error_page 404 = /404.php;
5. location / {
         error_page 404 = @fallback;
}
    location @fallback {
          proxy_pass http://backend;
}
6. error_page 403 http://example.com/forbidden.html;
7. error_page 404 =301 http://example.com/notfound.html;

rewrite 指令

rewrite 的语法是这样的,可以通过 rewrite_log on; 记录rewrite。

rewrite regex replacement [flag];
  • 将regex指定的url替换成 replacement 这个新的url。
  • 当replacement以 http:// 或者 $schema 开头,直接会返回302重定向。
  • 替换后的url根据flag指定的方式进行处理。
  • last:用replacement这个url 重新进行location匹配。
  • break:break指令停止当前脚本的执行,不再匹配。
  • redirect:返回302重定向。
  • permanent:返回301重定向。

具体的例子来访问一下。
目录的结构为

file
├── first
│   └── 1.txt    # 1
├── second
│   └── 2.txt    # 2
└── third
    └── 3.txt    # 3

如下的配置

root html/file/;
        location /first {
                rewrite /first(.*) /second$1 last;
                return 200 'first';
        }
        location /second {
                rewrite /second(.*) /third$1 break;
                return 200 'second';
        }
        location /third {
                return 200 'third';
        }

这里再重申一次,last表示匹配上后再匹配,break是中止往下执行,如果第二个没加 break的话,会 return 200 的。

curl -H"Host:rewrite.example.wjx" http://127.0.0.1/first/3.txt
3
curl -H"Host:rewrite.example.wjx" http://127.0.0.1/second/3.txt
3
curl -H"Host:rewrite.example.wjx" http://127.0.0.1/third/3.txt
third

如果不要 last ,不要 break会怎么样呢,他们的作用就是中止下面的return等指令运行
而last是rewrite后的url还可以再作rewrite,break就直接rewrite,不再做新的rewrite了

curl -H"Host:rewrite.example.wjx" http://127.0.0.1/first/3.txt
first
curl -H"Host:rewrite.example.wjx" http://127.0.0.1/second/3.txt
second
curl -H"Host:rewrite.example.wjx" http://127.0.0.1/third/3.txt
third

最后就是 permanent 和 redirect了。其实这两就是301 和 302 ,且如果 replacement 中有协议的话,默认就是 302
如下配置

location /redirect1 {
                rewrite /redirect1(.*) $1 permanent;
        }
        location /redirect2 {
                rewrite /redirect2(.*) $1 redirect;
        }
        location /redirect3 {
                rewrite /redirect3(.*) http://rewrite.example.wjx$1;
        }
        location /redirect4 {
                rewrite /redirect4(.*) http://rewrite.example.wjx$1 permanent;
        }

访问后如下,第一个会返回301,第二,三个会返回302,第四个返回301。

if 指令

if 指令也是rewrite模块中的一个指令。
其可作用于 server 和 location 块中。
一般可使用 字符串与变量做匹配 '=' 和 '!=' 。
也可以正则匹配 '~' 或 '~!'。
检测文件是否存在'-f' 或 '!-f' ,目录则是 '-d'。
检查文件,目录,软链接是否存在' -e' 或者 '!-e'。
检查是否为可执行文件,使用'-x' 或者 '!-x'。
以下为示例。

if ($http_user_agent ~ MSIE) {
    rewrite ^(.*)$ /msie/$1 break;
}
if ($http_cookie ~* "id=([^;]+(?:;|$))") {
    set $id $1;
}
if ($request_method = POST) {
    return 405;
}
if ($slow) {
    limit_rate 10k;
}
if ($invalid_referer) {
    return 403;
}