Nginx location 正则

问题一:

曾经遇到的一个问题:一个server下配置多个 location

location ^~ /admin {
    alias /var/www/admin/;
    index regist.html;
}
location / {
       root   /data/html/;
       index  index.html index.html;
}

root与alias主要区别在于nginx如何解释location后面的uri,这会使两者分别以不同的方式将请求映射到服务器文件上。
root的处理结果是:root路径+location路径
alias的处理结果是:使用alias路径替换location路径
alias是一个目录别名的定义,root则是最上层目录的定义。
还有一个重要的区别是alias后面必须要用“/”结束,否则会找不到文件的。。。而root则可有可无~~

问题二、rootalias 的区别

location 块中,rootalias 指令后的路径最后面,有没有 "/" 的区别如下:

location ^~ /dev-popularize/ {
            # root 指令中:如果后面没有 "/" 则 dist 这个目录下,必须有 dev-popularize 这个目录,然后你的静态资源文件都在这个目录下
            # root /Users/shounaer/dist;
            # root 指令中:如果后面有 "/" 则 dist 这个目录下,不需要有 dev-popularize 这个目录,你的静态资源文件直接放在dist目录下就行
            # root /Users/shounaer/dist/;
            # alias 指令中:如果后面没有 '/' ,则 dist 目录下,无论有没有dev-popularize 都不行,会报 403。
            alias /Users/shounaer/dist;      # 最终解析效果为:/Users/shounaer/distindex.html  所以会报 403.
            # alias 指令中:如果后面有 '/' ,则 dist 目录下,直接放你的静态资源文件就行
            # alias /Users/shounaer/dist/;       # 最终效果:/Users/shounaer/dist/index.html
            index index.html;
        }

一、语法规则

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

1、 =

表示精确匹配,优先级最高,匹配成功后则停止向下搜索。

# 精确匹配,必须是127.0.0.1/
location = / {
  ...
}
# 精确匹配,必须是127.0.0.1/login
location = /login {
  ...
}

2、 ^~

对uri 起始字符字符串匹配,不是 正则匹配。 区分大小写

## 127.0.0.1/static/js.
location ^~ /static/ {
    ...
}

3、~

对 uri (可以不是起始字符)做 正则匹配,区分大小写

## 区分大小写,以gif,jpg,js结尾
location ~ \.(gif|jpg|png|js|css)$ {
    ...
}

4、~*

对 uri (可以不是起始字符)做 正则匹配,不区分大小写

## 不区分大小写,匹配.png结尾的
location ~* \.png$ {
    ...
}

5、!~!~*

分别为区分大小写不匹配不区分大小写不匹配正则匹配

## 区分大小写,匹配不以.xhtml结尾的
location !~ \.xhtml$ {
    ...
}
location !~* \.xhtml$ {
    ...
}

6、不带符号

匹配起始于此字符串的所有uri,区分大小写,字符串匹配

## 什么都可以
location / {
    ...
}

注意:字符串匹配(无论是 ^~ 还是 无符号 匹配),后面表达式必须以 / 开头,否则永远无法匹配到。

总结:

  • 精确匹配: = , 后面的表达式中写的是纯字符串
  • 字符串匹配: ^~无符号匹配 , 后面的表达式中写的是纯字符串
  • 正则匹配: ~~*!~!~* , 后面的表达式中写的是正则表达式

7、匹配顺序

= > ^~ > ~ > ~* > 不带符号

  • 多个字符串匹配同时存在时,谁的表达式中被匹配到的字符越多,就命中谁。
    比如:访问: localhost:8080/imgs 时, location ^~ /img {} 不会被命中, location /imgs {} 会被命中。
  • 多个正则匹配同时存在且表达式均满足匹配规则时,谁排在前面,谁先命中,不存在匹配到的字符串长度一说。
  • 如果 ^~无符号匹配~~* 同时存在,且他们的表达式均满足匹配规则。则按照如下规则命中
    • 先看字符串匹配,被匹配到的字符长度越长,越优先命中;
    • 如果是无符号匹配被命中(被匹配到的字符长度较长),则再去查看正则匹配,如果正则匹配中也有被匹配到的,那么对应的正则匹配被命中,原本命中到的无符号匹配被舍弃。

location @name { ... }
此种方式的location配置,无法通过外网访问,只能通过内网跳转,实现容错的功能。

8、其他

所以实际使用中,个人觉得至少有三个匹配规则定义,如下:
// 这里是直接转发给后端应用服务器了,也可以是一个静态首页

1)、第一个必选规则
location = / {
  proxy_pass http://127.0.0.1:3000/
}
2)、第二个必选规则

第二个必选规则是处理静态文件请求,这是nginx作为http服务器的强项
有两种配置模式,目录匹配后缀匹配,任选其一或搭配使用

location ^~ /static/ {
  root /webroot/static/;
}

location ~* \.(gif|jpg|jpeg|png|css|js|ico)$ {
  root /webroot/res/;
}
3)、第三个规则就是通用规则,用来转发动态请求到后端应用服务器
location /api/ {
    proxy_pass http://127.0.0.1:3000/api/
}

9、Rewrite 重定向

last – 表示完成rewrite 
break – 中止Rewirte,不在继续匹配 
redirect – 返回临时重定向的HTTP状态302 
permanent – 返回永久重定向的HTTP状态301 
1、下面是可以用来判断的表达式: 
-f和!-f用来判断是否存在文件 
-d和!-d用来判断是否存在目录 
-e和!-e用来判断是否存在文件或目录 
-x和!-x用来判断文件是否可执行 
2、下面是可以用作判断的全局变量 
例:http://localhost:88/test1/test2/test.php 
$host:localhost 
$server_port:88 
$request_uri:/test1/test2/test.php 
$document_uri:/test1/test2/test.php 
$document_root:D:\nginx/html 
$request_filename:D:\nginx/html/test1/test2/test.php 

参考这篇文章:《nginx中location、rewrite用法总结》的Rewrite部分

server { 
  listen 80; 
  server_name start.igrow.cn; 
  root html; 
  index index.html index.php; 
  rewrite ^(.*) https://star.igrow.cn$1 permanent; 
} 

10、return 重定向

rewrite 重定向:需要先做正则匹配,然后再给客户端返回新地址进行重定向(做了两步操作)
return 重定向:不需要做正则匹配,直接给客户端返回新地址进行重定向(做了一步操作)

server { 
  listen 80; 
  server_name start.igrow.cn; 
  return 301 https://$host$request_uri;
}

11、if 模块

详情请:参考这篇文章《nginx中location、rewrite用法总结》的 if 部分

 // 如果用户设备为IE浏览器的时候,重定向 。如果UA包含"MSIE",rewrite请求到/msid/目录下
if ($http_user_agent ~ MSIE) {
    rewrite ^(.*)$ /msie/$1 break;
}
 
 //如果cookie匹配正则,设置变量$id等于正则引用部分
if ($http_cookie ~* "id=([^;]+)(?:;|$)") {
    set $id $1;
}
 
 //如果提交方法为POST,则返回状态405(Method not allowed)。return不能返回301,302
if ($request_method = POST) {
    return 405;
}

 //限速,$slow可以通过 set 指令设置 
if ($slow) {
    limit_rate 10k;
}
 
//如果请求的文件名不存在,则反向代理到localhost 。这里的break也是停止rewrite检查
if (!-f $request_filename){
    break;
    proxy_pass http://127.0.0.1;
} 
 
 //如果query string中包含"post=140",永久重定向到example.com
if ($args ~ post=140){
    rewrite ^ http://example.com/ permanent;
}
 
 //防盗链
location ~* \.(gif|jpg|png|swf|flv)$ {
    valid_referers none blocked www.jefflei.com www.lfa.com;
    if ($invalid_referer) {
        return 404;
    }
}

12、防盗链

location ~* \.(gif|jpg|swf)$ { 
  valid_referers none blocked start.igrow.cn sta.igrow.cn; 
  if ($invalid_referer) { 
    rewrite ^/ http://$host/logo.png; 
  } 
}

说明:

如果 valid_referers 条件判断未通过,nginx 则会赋值 invalid_referertrue
语法: valid_referers none | blocked | server_names | string ...;
参数说明:
none: 不允许 “Referer” 来源头部为空的情况
blocked: 不允许“Referer”值为空情况,有可能Referer的值被代理或者防火墙删除
server_names: “Referer”来源头部包必须含当前的server_names (当前域名)可以多个

13、根据文件类型设置过期时间

location ~* \.(js|css|jpg|jpeg|gif|png|swf)$ { 
  if (-f $request_filename) { 
    expires 1h; 
    break; 
  } 
} 

14、禁止访问某个目录

location ~* \.(txt|doc)${ 
  root /data/www/wwwroot/linuxtone/test; 
  deny all; 
}

15、一些可用的全局变量

$host : 请求主机头字段,否则为服务器名称。
$http_user_agent : 客户端agent信息
$http_cookie : 客户端cookie信息
$request_method : 客户端请求的动作,通常为GET或POST。
$content_length : 请求头中的Content-length字段。
$content_type : 请求头中的Content-Type字段。
$request_uri : 包含请求参数的原始URI,不包含主机名,如:”/foo/bar.php?arg=baz”。
$uri : 不包含请求参数的当前URI,$uri不包含主机名,如”/foo/bar.html”。
$document_uri : 与$uri相同。
$query_string 
$args : 这个变量等于请求行中的参数,同$query_string
$document_root : 当前请求在root指令中指定的值。
$limit_rate : 这个变量可以限制连接速率。
$request_body_file 
$request_filename : 当前请求的文件路径,由root或alias指令与URI请求生成。
$remote_addr : 客户端的IP地址。
$remote_port : 客户端的端口。
$remote_user : 已经经过Auth Basic Module验证的用户名。
$scheme : HTTP方法(如http,https)。
$server_protocol : 请求使用的协议,通常是HTTP/1.0或HTTP/1.1。
$server_addr : 服务器地址,在完成一次系统调用后可以确定这个值。
$server_name : 服务器名称。
$server_port : 请求到达服务器的端口号。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 160,999评论 4 368
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 68,102评论 1 302
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 110,709评论 0 250
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,439评论 0 217
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,846评论 3 294
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,881评论 1 224
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 32,062评论 2 317
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,783评论 0 205
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,517评论 1 248
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,762评论 2 253
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,241评论 1 264
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,568评论 3 260
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,236评论 3 241
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,145评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,941评论 0 201
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,965评论 2 283
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,802评论 2 275