Skip to content

Nginx 代理 proxy_pass 配置去除前缀

使用Nginx做代理的时候,可以简单的直接把请求原封不动的转发给下一个服务。

shell
# 比如,访问 http://aday.fun/user/list, 要求转发到 localhost:8080/user/list
server {
    listen              80;
    server_name         aday.fun;

    location / {
        proxy_set_header Host $http_host;
        proxy_set_header X-Real-IP $remote_addr;
        proxy_set_header REMOTE-HOST $remote_addr;
        proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
        proxy_pass http://localhost:8080;
    }
}

即,设置 proxy_pass 即可。请求只会替换域名。

但很多时候,我们需要根据 url 的前缀转发到不同的服务。

比如

aday.fun/user/profile 转发到 用户服务 localhost:8080/profile

aday.fun/order/details 转发到 订单服务 localhost:8081/details

即,url的前缀对下游的服务是不需要的,除非下游服务添加 context-path, 但很多时候我们并不喜欢加这个。如果 Nginx 转发的时候,把这个前缀去掉就好了。

方案一:proxy_pass 后面加根路径 /

shell
server {
    listen              80;
    server_name         aday.fun;

    location ^~/user/ {
        proxy_set_header Host $http_host;
        proxy_set_header X-Real-IP $remote_addr;
        proxy_set_header REMOTE-HOST $remote_addr;
        proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
        proxy_pass http://localhost:8080/;
    }

    location ^~/order/ {
        proxy_set_header Host $http_host;
        proxy_set_header X-Real-IP $remote_addr;
        proxy_set_header REMOTE-HOST $remote_addr;
        proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
        proxy_pass http://localhost:8081/;
    }
}

^~/user/ :表示匹配前缀是 user 的请求,proxy_pass 的结尾有 /, 则会把 /user/* 后面的路径直接拼接到后面,即移除 user.

方案二:使用 rewrite

shell
server {
    listen              80;
    server_name         aday.fun;

    location ^~/user/ {
        proxy_set_header Host $http_host;
        proxy_set_header X-Real-IP $remote_addr;
        proxy_set_header REMOTE-HOST $remote_addr;
        proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;

        rewrite ^/user/(.*)$ /$1 break;
        proxy_pass http://localhost:8080;
    }

    location ^~/order/ {
        proxy_set_header Host $http_host;
        proxy_set_header X-Real-IP $remote_addr;
        proxy_set_header REMOTE-HOST $remote_addr;
        proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;

        rewrite ^/order/(.*)$ /$1 break;
        proxy_pass http://localhost:8081;
    }
}

注意到 proxy_pass 结尾没有 /, rewrite 重写了 url。

关于 rewrite

实际上就是正则在 nginx 里面的运用。

  • ^ 符号表示正则以什么前缀开始;
  • $ 符号表示正则以什么后缀结束;
  • 后面的 $1 是对第 1 个括号内容的引用;
  • 后面的 $2 是对第 2 个括号内容的引用;
  • 以此类推,$n 是对第 n 个括号内容的引用。
shell
# 先看一个nginx配置
rewrite ^/(user_\d)/(\d).html$ https://$host/?$1 permanent;

其中,$ 代表的是参数,所以一定是 () 包含的。

  • () : 用于匹配括号之间的内容,通过 1、2 调用;
  • $1 就是 user_\d
  • $2 就是 \d
txt
举个例子:https://aday.fun/user_1/2.html 这里:

$1 就是 user_1

$2 就是 2