配置server对应的域名

<font style="color:rgb(53, 148, 247);">server name</font>为虚拟服务器的识别路径。因此不同的域名会通过请求头中的HOST字段,匹配到特定的server块,转发到对应的应用服务器中去。<font style="color:rgb(53, 148, 247);">server_name</font>匹配规则:后面可以跟多个域名,第1个是主域名。

精确匹配

如下Nginx配置。

  1. server{
  2. listen 8080;
  3. server_name test1.com;
  4. location / {
  5. return 200 "I am test1!\n";
  6. }
  7. }
  8. server {
  9. listen 8080;
  10. server_name my.test.com;
  11. location / {
  12. return 200 "I am mytest!\n";
  13. }
  14. }
请求结果
  1. curl http://my.test.com:8080 返回:I am mytest!
  2. curl http://test1.com:8080 返回:I am test1!

正则表达式

  • *通配符开始的最长字符串,如下示例
  1. server {
  2. listen 8080;
  3. server_name test1.*;
  4. location / {
  5. return 200 "I am test1!\n";
  6. }
  7. }
  • *通配符结束的最长字符串
  1. server{
  2. listen 8080;
  3. server_name *.test.com;
  4. location / {
  5. return 200 "I am mytest!\n";
  6. }
  7. }
通配符名字只可以在名字的起始处或结尾处包含一个星号,并且星号与其他字符之间用点分隔。所以,“my.*.com“都是非法的。 例如:<font style="color:rgb(53, 148, 247);">server_name my.*.com;</font> 报以下错误:
  1. nginx: [emerg] invalid server name or wildcard "my.*.com" on 0.0.0.0:8080
  • 匹配正则表达式
  1. server {
  2. listen 8080;
  3. server_name ~^my(?<serno>.+).mydomain.com$;
  4. location / {
  5. return 200 $serno;
  6. }
  7. }

解释说明

  1. ~:表示大小写敏感的正则;
  2. ^:匹配字符串的开始;
  3. {.+}:换行符以外的任意自读重复一次或更多次;
  4. ():分组与取值;
  5. ?:表示转义;
  6. serno:设置提取的变量;
  7. $:匹配字符串的结束;

请求结果

  1. curl http://my02.mydomain.com:8080 返回:02%
  2. curl http://my03.mydomain.com:8080 返回:03%
  • server_name的配置顺序是怎样的呢?按照如下顺序匹配:匹配顺序-> ->精确匹配 ->在前的域名 ->在后的域名 ->按文件中的顺序匹配 ->default server:第一个,listen指定default

配置location

Location 匹配规则:仅匹配URI,忽略参数

  1. location [=|~|~*|^~] /uri/ { }

匹配的正则符号如下:

  • = 严格匹配。如果请求匹配这个location,那么将停止搜索并立即处理此请求
  • ~ 区分大小写匹配(可用正则表达式)
  • ~* 不区分大小写匹配(可用正则表达式)
  • !~ 区分大小写不匹配
  • !~* 不区分大小写不匹配
  • ^~ 如果把这个前缀用于一个常规字符串,那么告诉Nginx 如果路径匹配那么不测试正则表达式

举例

  1. 1、匹配任意请求
  2. location [=|~|~*|^~] /uri/ { }
  3. 2、不区分大小写匹配以jsphp结尾的请求
  4. location ~* .(js|php)$ { }
  5. 3、区分大小写匹配以.txt结尾的请求
  6. location ~ ^.+\.txt$

匹配顺序如下图

画板

按照上面的规则配置了如下location
  1. location = /documents {
  2. return 200 'configuration A'
  3. }
  4. location /documents {
  5. return 200 'configuration B'
  6. }
  7. location /documents/txt1 {
  8. return 200 'configuration C'
  9. }
  10. location ^~ /documents/ {
  11. return 200 'configuration D'
  12. }
  13. location ~* /documents/(\w+)$ {
  14. return 200 'configuration E'
  15. }
  16. location ~ /documents/$ {
  17. return 200 'configuration F'
  18. }
  • curl http://test1.com:8080/documents%EF%BC%8C%E7%B2%BE%E7%A1%AE%E5%8C%B9%E9%85%8D%E8%BF%94%E5%9B%9E configuration A
  • curl http://test1.com:8080/documents/ ^~匹配上后不在匹配,返回 configuration D
  • curl http://test1.com:8080/documents/txt1 走到了正则匹配,不会走到/documents/txt1(正则没走完) 返回configuration E
  • curl http://test1.com:8080/documents/txt1/,%E8%BF%94%E5%9B%9Econfiguration C,因为正则都不匹配

如何debug正则呢?

编译的时候加上 --with-debug选项,例如<font style="color:rgb(53, 148, 247);">./configure --with-debug conf</font>文件加上要debug的host,<font style="color:rgb(53, 148, 247);">debug_connection</font>对应要debug的连接。

  1. events {
  2. worker_connections 1024;
  3. debug_connection 192.168.1.3;
  4. debug_connection 127.0.0.1;
  5. }

error.log查看debug日志,图中test location就是正则匹配的过程。

Nginx四种跳转总结,掌握域名、URI、重定向与代理 - 图2

配置rewrite

语法如下:
  1. 指令语法:rewrite regex replacement[flag];
  2. 默认值:none
  3. 应用位置:serverlocationif
  4. rewrite是实现URL重定向的重要指令,他根据regex(正则表达式)来匹配内容跳转到replacement,结尾是flag标记.
flag标记 说明
last 本条规则匹配完成后继续向下匹配新的location URI规则
break 本条规则匹配完成后终止,不在匹配任务规则
redirect 返回302临时重定向
permanent 返回301永久重定向

重定向

<font style="color:rgb(53, 148, 247);">return</font>三种code,code url和url。返回状态码:444表示关闭连接 301表示http1。0中永久重定向,302表示临时重定向,进制缓存。http1.1后,303表示临时重定向,允许改变方法,进制缓存,307表示临时重定向,不允许改变方法,禁止被缓存,308表示永久重定向,不允许改变方法。

  • 返回code
  1. location / {
  2. return 301 https://www.xxxx.com$request_uri;
  3. }
  • 通过$request_uri变量匹配所有的URI。
  1. rewrite ^ https://www.xxxx.com$request_uri? permanent;
  • 通过正则匹配所有的URI后再去掉开头第一个/(反斜线)。
  1. rewrite ^/(.*)$ https://www.xxxx.com/$1;
  • 与if指令结合
  1. server {
  2. listen 80;
  3. server_name test1.net test2.net;
  4. if ($host != 'test1.net' ) {
  5. rewrite ^/(.*)$ http://www.baidu.net/$1 permanent;
  6. }
  7. }

如何查看rewrite日志

打开日志开关<font style="color:rgb(53, 148, 247);">rewrite_log on;</font> 可以配置到http,server,location和if上下文中。 示例:<font style="color:rgb(53, 148, 247);">curl test1.com:8080/first/2.txt</font>
  1. location /first {
  2. rewrite_log on;
  3. rewrite /first(.*) /second$1 last;
  4. }
效果图如下:

Nginx四种跳转总结,掌握域名、URI、重定向与代理 - 图3

配置 proxy

对上游服务使用<font style="color:rgb(53, 148, 247);">http/https</font>协议进行反向代理。<font style="color:rgb(53, 148, 247);">proxy_pass</font>后面跟 url,可以仿造 location,if in location和<font style="color:rgb(53, 148, 247);">limit_except</font>上下文中。 这个功能是默认编译到Nginx中的。本文重点讨论http proxy。

url参数规则

  • url必须以http或者https开头,接下来是域名、ip、unix socket或者upstream名字,都可以就端口。后面是可选的uri
http示例:<font style="color:rgb(53, 148, 247);">proxy_pass http://localhost:8000/uri/;</font> UNIX域套接字路径来定义示例:<font style="color:rgb(53, 148, 247);">proxy_pass http://unix:/tmp/backend.socket:/uri/;</font> + url中是否携带uri,结果也不一样,如果在<font style="color:rgb(89, 89, 89);">proxy_pass</font>后面的url加/,相当于是绝对根路径,则Nginx不会把location中匹配的路径部分代理走;如果没有/,则会把匹配的路径部分给代理走。 目录结构如下
  1. ├── first
  2. └── index.html
  3. ├── index.html
  4. └── second
  5. └── index.html
Nginx配置如下
  1. server {
  2. listen 8081;
  3. server_name my.test.com;
  4. }
  5. server {
  6. listen 8082;
  7. # 第一种情况
  8. location /first {
  9. proxy_pass http://my.test.com:8081;
  10. proxy_set_header Host $host;
  11. proxy_set_header X-Real-IP $remote_addr;
  12. proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
  13. }
  14. # 第二种情况
  15. location /first {
  16. proxy_pass http://my.test.com:8081/;
  17. proxy_set_header Host $host;
  18. proxy_set_header X-Real-IP $remote_addr;
  19. proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
  20. }
  21. }
不带<font style="color:rgb(53, 148, 247);">/</font>,然后<font style="color:rgb(53, 148, 247);">curl http://127.0.0.1:8082/first/index.html</font>直接返回index.html。 <font style="color:rgb(53, 148, 247);">/</font>,然后 curl <font style="color:rgb(53, 148, 247);">http://127.0.0.1:8082/first/index.html</font>返回first下的index.html。
  • Url参数中可以携带变量proxy_pass http://$host$uri;
  • 可以配合rewrite break语句
  1. location /nameb/ {
  2. rewrite /nameb/([^/]+) /test?nameb=$1 break;
  3. proxy_pass http://127.0.0.1:8801/;
  4. }

小结

配置Nginx的路由,有多种方式,域名可以用<font style="color:rgb(53, 148, 247);">server_name</font>配置,uri可以用location配置,复杂的可以加rewrite配置修改请求。还有就是配置proxy代理,在代理中转发id等。本文开头的几个问题,同端口不同域名使用<font style="color:rgb(53, 148, 247);">server_name</font>配置就可以,可以放到不同的server中;location的匹配顺序可以参考文中的那张流程图;重定向的情况下客户端拿到302和url就会去请求这个url指定的资源,proxy不会,因此在Nginx+Tomcat的场景下,proxy更适合。