Nginx 从入门到入坟(十)- Rewrite 功能详解与案例实操
Rewrite 是 Nginx 服务器提供的一个重要基本功能,是 Web 服务器产品中几乎必备的功能,主要的作用是用来实现 URL 的重写。
注意:Nginx 服务器的 Rewrite 功能的实现依赖于 PCRE 的支持,因此在编译安装 Nginx 服务器之前,需要安装 PCRE 库。Nginx 使用的是ngx_http_rewrite_module
模块来解析和处理 Rewrite 功能的相关配置。
1. 地址重写与地址转发
重写和转发的区别:
- 地址重写浏览器地址会发生变化而地址转发则不变;
- 一次地址重写会产生两次请求而一次地址转发只会产生一次请求;
- 地址重写到的页面必须是一个完整的路径而地址转发则不需要;
- 地址重写因为是两次请求所以request范围内属性不能传递给新页面而地址转发因为是一次请求所以可以传递值;
- 地址转发速度快于地址重写。
2. Rewrite规则
2.1 set指令
该指令用来设置一个新的变量。
语法 | set $variable value; |
---|---|
默认值 | - |
位置 | server、location、if |
- variable:变量的名称,该变量名称要用 “$” 作为变量的第一个字符,且不能与 Nginx 服务器预设的全局变量同名。
- value:变量的值,可以是字符串、其他变量或者变量的组合等。
2.1.1 Rewrite常用全局变量
变量 | 说明 |
---|---|
$args | 变量中存放了请求URL中的请求指令。比如http://10.7.2.205:8080?arg1=value1&args2=value2 中的arg1=value1&arg2=value2 ,功能和 $query_string 一样 |
$http_user_agent | 变量存储的是用户访问服务的代理信息(如果通过浏览器访问,记录的是浏览器的相关版本信息) |
$host | 变量存储的是访问服务器的server_name值 |
$document_uri | 变量存储的是当前访问地址的 URI。比如http://10.7.2.205/server?id=10&name=zhangsan 中的/server ,功能和 $uri 一样 |
$document_root | 变量存储的是当前请求对应 location 的 root 值,如果未设置,默认指向 Nginx 自带 html 目录所在位置 |
$content_length | 变量存储的是请求头中的 Content-Length 的值 |
$content_type | 变量存储的是请求头中的 Content-Type 的值 |
$http_cookie | 变量存储的是客户端的 cookie 信息,可以通过 add_header Set-Cookie ‘cookieName=cookieValue’ 来添加 cookie 数据 |
$limit_rate | 变量中存储的是 Nginx 服务器对网络连接速率的限制,也就是 Nginx 配置中对 limit_rate 指令设置的值,默认是 0,不限制 |
$remote_addr | 变量中存储的是客户端的 IP 地址 |
$remote_port | 变量中存储了客户端与服务端建立连接的端口号 |
$remote_user | 变量中存储了客户端的用户名,需要有认证模块才能获取 |
$scheme | 变量中存储了访问协议 |
$server_addr | 变量中存储了服务端的地址 |
$server_name | 变量中存储了客户端请求到达的服务器的名称 |
$server_port | 变量中存储了客户端请求到达服务器的端口号 |
$server_protocol | 变量中存储了客户端请求协议的版本,比如 “HTTP/1.1” |
$request_body_file | 变量中存储了发给后端服务器的本地文件资源的名称 |
$request_method | 变量中存储了客户端的请求方式,比如 “GET”,“POST” 等 |
$request_filename | 变量中存储了当前请求的资源文件的路径名 |
$request_uri | 变量中存储了当前请求的 URI,并且携带请求参数,比如http://10.7.2.205/server?id=10&name=zhangsan 中的/server?id=10&name=zhangsan |
2.2 if指令
该指令用来支持条件判断,并根据条件判断结果选择不同的 Nginx 配置。
语法 | if (condition){…} |
---|---|
默认值 | - |
位置 | server、location |
condition 为判定条件,可以支持以下写法:
- 变量名,如果变量名对应的值为空或者是 0,if 都判断为 false,其他条件为 true。
1
2
3if ($param){
} - 使用 “=” 和 “!=” 比较变量和字符串是否相等,满足条件为 true,不满足为 false。
1
2
3if ($request_method = POST){
return 405;
} - 使用正则表达式对变量进行匹配,匹配成功返回 true,否则返回 false。变量与正则表达式之间使用
~
,~*
,!~
,!~*
来连接。注意:正则表达式字符串一般不需要加引号,但是如果字符串中包1
2
3
4
5
6
7"~"代表匹配正则表达式过程中区分大小写,
"~*"代表匹配正则表达式过程中不区分大小写
"!~"和"!~*"刚好和上面取相反值,如果匹配上返回false,匹配不上返回true
if ($http_user_agent ~ MSIE){
#$http_user_agent的值中是否包含MSIE字符串,如果包含返回true
}
含"}"
或者是";"
等字符时,就需要把引号加上。 - 判断请求的文件是否存在使用
"-f"
和"!-f"
,当使用"-f"
时,如果请求的文件存在返回 true,不存在返回 false;当使用"!-f"
时,如果请求文件不存在,但该文件所在目录存在返回 true,文件和目录都不存在返回 false,如果文件存在返回 false。1
2
3
4
5
6if (-f $request_filename){
#判断请求的文件是否存在
}
if (!-f $request_filename){
#判断请求的文件是否不存在
} - 判断请求的目录是否存在使用
"-d"
和"!-d"
,当使用"-d"
时,如果请求的目录存在,if 返回 true,如果目录不存在则返回 false;当使用"!-d"
时,如果请求的目录不存在但该目录的上级目录存在则返回 true,该目录和它上级目录都不存在则返回 false,如果请求目录存在也返回 false。 - 判断请求的目录或者文件是否存在使用
"-e"
和"!-e"
,当使用"-e"
,如果请求的目录或者文件存在时,if 返回 true,否则返回 false;当使用"!-e"
,如果请求的文件和文件所在路径上的目录都不存在返回 true,否则返回 false。 - 判断请求的文件是否可执行使用
"-x"
和"!-x"
,当使用"-x"
,如果请求的文件可执行,if 返回 true,否则返回 false;当使用"!-x"
,如果请求文件不可执行,返回 true,否则返回 false。
2.3 break指令
该指令用于中断当前相同作用域中的其他 Nginx 配置。与该指令处于同一作用域的 Nginx 配置中,位于它前面的指令配置生效,位于后面的指令配置无效。
语法 | break |
---|---|
默认值 | - |
位置 | server、location、if |
例子:
1 | location /{ |
2.4 return指令
该指令用于完成对请求的处理,直接向客户端返回响应状态代码。在return后的所有 Nginx 配置都是无效的。
语法 | return code [text] return code URL return URL |
---|---|
默认值 | - |
位置 | server、location、if |
- code:为返回给客户端的 HTTP 状态代理。可以返回的状态代码为 0~999 的任意 HTTP 状态代理;
- text:为返回给客户端的响应体内容,支持变量的使用;
- URL:为返回给客户端的 URL 地址;
2.5 rewrite指令
该指令通过正则表达式的使用来改变 URI。可以同时存在一个或者多个指令,按照顺序依次对 URL 进行匹配和处理。
URL 和 URI 的区别:
1 | URI:统一资源标识符 |
语法 | rewrite regex replacement [flag] |
---|---|
默认值 | - |
位置 | server、location、if |
regex:用来匹配 URI 的正则表达式。
replacement:匹配成功后,用于替换 URI 中被截取内容的字符串。如果该字符串是以http://
或者https://
开头的,则不会继续向下对 URI 进行其他处理,而是直接返回重写后的 URI 给客户端。
flag:用来设置 rewrite 对 URI 的处理行为,可选值有如下:
- last
- break
- redirect
- permanent
2.6 rewrite_log指令
该指令配置是否开启 URL 重写日志的输出功能。
语法 | rewrite_log on|off |
---|---|
默认值 | rewrite_log off |
位置 | http、server、location、if |
开启后,URL 重写的相关日志将以 notice 级别输出到 error_log 指令配置的日志文件汇总。
3. Rewrite的案例
3.1 域名跳转
问题分析
先来看一个效果,如果我们想访问京东网站,大家都知道我们可以输入www.jd.com
,但是同样的我们也可以输入www.360buy.com
同样也都能访问到京东网站。这个其实是因为京东刚开始的时候域名就是www.360buy.com
,后面由于各种原因把自己的域名换成了www.jd.com
,虽然说域名变量,但是对于以前只记住了www.360buy.com
的用户来说,我们如何把这部分用户也迁移到我们新域名的访问上来,针对于这个问题,我们就可以使用 Nginx 中 Rewrite 的域名跳转来解决。
环境准备
- 准备两个域名
www.360buy.com
|www.jd.com
1
2
3
4vim /etc/hosts
10.7.2.205 www.360buy.com
10.7.2.205 www.jd.com - 在
/usr/local/nginx/html/hm
目录下创建一个访问页面:1
2
3
4
5
6<html>
<title></title>
<body>
<h1>欢迎来到我们的网站</h1>
</body>
</html> - 通过 Nginx 实现当访问
www.hm.com
访问到系统的首页:1
2
3
4
5
6
7
8server {
listen 80;
server_name www.hm.com;
location /{
root /usr/local/nginx/html/hm;
index index.html;
}
} - 通过 Rewrite 完成将
www.360buy.com
的请求跳转到www.jd.com
:1
2
3
4
5server {
listen 80;
server_name www.360buy.com;
rewrite ^/ http://www.jd.com permanent;
}
问题描述:如何在域名跳转的过程中携带请求的URI?
修改配置信息:
1 | server { |
问题描述:我们除了上述说的
www.jd.com
、www.360buy.com
其实还有我们也可以通过www.jingdong.com
来访问,那么如何通过 Rewrite 来实现多个域名的跳转?
添加域名:
1 | vim /etc/hosts |
修改配置信息:
1 | server{ |
3.2 域名镜像
上述案例中,将www.360buy.com
和www.jingdong.com
都能跳转到www.jd.com
,那么www.jd.com
我们就可以把它起名叫主域名,其他两个就是我们所说的镜像域名,当然如果我们不想把整个网站做镜像,只想为其中某一个子目录下的资源做镜像,我们可以在 location 块中配置 rewrite 功能,比如:
1 | server { |
3.3 独立域名
一个完整的项目包含多个模块,比如购物网站有商品商品搜索模块、商品详情模块已经购物车模块等,那么我们如何为每一个模块设置独立的域名。
1 | http://search.hm.com 访问商品搜索模块 |
1 | server{ |
3.4 目录自动添加“/”
通过一个例子来演示下问题:
1 | server { |
要想访问上述资源,很简单,只需要通过http://10.7.2.205
直接就能访问,地址后面不需要加/
,但是如果将上述的配置修改为如下内容:
1 | server { |
这个时候,要想访问上述资源,按照上述的访问方式,我们可以通过http://10.7.2.205/hm/
来访问,但是如果地址后面不加斜杠,页面就会出问题。如果不加斜杠,Nginx 服务器内部会自动做一个 301 的重定向,重定向的地址会有一个指令叫server_name_in_redirect on|off
来决定重定向的地址:
1 | 如果该指令为on |
所以就拿刚才的地址来说,http://10.7.2.205/hm
如果不加斜杠,那么按照上述规则,如果指令server_name_in_redirect
为on
,则 301 重定向地址变为http://localhost/hm/
,如果为off
,则 301 重定向地址变为http://10.7.2.205/ht/
。后面这个是正常的,前面地址就有问题。
注意server_name_in_redirect
指令在 Nginx 的 0.8.48 版本之前默认都是on
,之后改成了off
,所以现在我们这个版本不需要考虑这个问题,但是如果是 0.8.48 以前的版本并且server_name_in_redirect
设置为on
,我们如何通过 rewrite 来解决这个问题?
解决方案:
我们可以使用 rewrite 功能为末尾没有斜杠的 URL 自动添加一个斜杠:
1 | server { |
3.5 合并目录
搜索引擎优化(SEO)是一种利用搜索引擎的搜索规则来提供目的网站的有关搜索引擎内排名的方式。我们在创建自己的站点时,可以通过很多中方式来有效的提供搜索引擎优化的程度。其中有一项就包含 URL 的目录层级一般不要超过三层,否则的话不利于搜索引擎的搜索也给客户端的输入带来了负担,但是将所有的文件放在一个目录下又会导致文件资源管理混乱并且访问文件的速度也会随着文件增多而慢下来,这两个问题是相互矛盾的,那么使用 rewrite 如何解决上述问题?
举例:网站中有一个资源文件的访问路径是/server/11/22/33/44/20.html
,也就是说20.html
存在于第 5 级目录下,如果想要访问该资源文件,客户端的 URL 地址就要写成http://www.web.name/server/11/22/33/44/20.html
。
1 | server { |
但是这个是非常不利于 SEO 搜索引擎优化的,同时客户端也不好记,使用 rewrite 我们可以进行如下配置:
1 | server { |
这样的话,客户端只需要输入http://www.web.name/server-11-22-33-44-20.html
就可以访问到20.html
页面了。这里也充分利用了 rewrite 指令支持正则表达式的特性。
3.6 防盗链
防盗链之前我们已经介绍过了相关的知识,在 rewrite 中的防盗链和之前将的原理其实都是一样的,只不过通过 rewrite 可以将防盗链的功能进行完善下,当出现防盗链的情况,我们可以使用 rewrite 将请求转发到自定义的一张图片和页面,给用户比较好的提示信息。下面我们就通过根据文件类型实现防盗链的一个配置实例:
1 | server{ |
根据目录实现防盗链配置:
1 | server{ |