本文主要介绍了Server和Location匹配逻辑相关内容。
上篇文章回顾: 浅谈SYNPROXY
Nginx在决定请求由哪一个server块执行时,主要关注的是server块中的listen和server_name两个字段nginx
listen字段定义server响应的ip和端口,若是没有明确配置listen字段,默认监听0.0.0.0:80(root)或者0.0.0.0:8080(非root)
listen能够被配置为:正则表达式
一个ip和端口的组合浏览器
一个单独的ip,默认监听80端口bash
一个单独的端口,默认监听全部的ip接口运维
一个Unix socket路径dom
其中最后一项一般只用于在不一样的server之间传递请求
选择要使用的server的规则以下:socket
Nginx首先将全部"不完整"的listen指令进行转换,好比没有listen字段的转换为listen 0.0.0.0:80,listen 1.1.1.1转换为listen 1.1.1.1:80等网站
Nginx根据请求的ip和端口建立一个与请求最匹配的server块列表,优先匹配指定了特定ip的server块,其次才会选择listen 0.0.0.0的这种server块.可是不管是哪一种状况,端口必须是彻底匹配的ui
若是只有一个最佳匹配,那么将使用匹配的server块响应请求,不然开始评估每个server块的server_name指令spa
再次强调一遍,只有当listen指令没法找到最佳匹配时才会考虑评估server_name指令.
好比,咱们假设example.com域名指向了192.168.0.1,且位于192.168.0.1上的nginx有且仅有以下两个server块:
# server block 1server {
listen 192.168.0.1;
server_name other.com
...
}
# server block 2server {
listen 80;
server_name example.com
...
}复制代码
若是根据listen指令没法获得最佳匹配,将会开始解析server_name指令.nginx会检查请求中的"Host"头,这个值包含了客户端实际试图请求的域名或者ip地址.nginx会根据这个值去匹配server_name指令,匹配规则以下:
nginx会尝试寻找一个和sever_name和Host值彻底匹配的server块,若是找到多个精确匹配,则会使用第一个匹配的server块
若是没有找到精确匹配的server块,则nginx尝试找到server_name带有*开头的server块,若是找到多个,则选择最长匹配的server块
若是没有找到使用开头的server块,则会寻找以结尾的server块,一样,若是有多个匹配, 选择最长匹配
若是没有找到使用*匹配的server块,则会寻找使用正则表达式(以~开头)定义server_name的server块,若是找到多个匹配,会使用第一个匹配
若是没有找到正则表达式匹配的server块,则nginx将会选择一个匹配listen字段的default server块.每个ip和端口组合均可以配置一个且只能配置一个默认的default_server块,若是没有的话,则会选择可用列表中的第一个server(此时的选择是随机的,顺序不固定)
示例以下:
(1)准确的server_name匹配,例如:
server {
listen 80;
server_name www.domain.com;
...
}复制代码
(2)以*通配符开始的字符串:
server {
listen 80;
server_name *.domain.com;
...
}复制代码
(3)以*通配符结束的字符串:
server {
listen 80;
server_name www.*;
...
}复制代码
(4)匹配正则表达式:
server {
listen 80;
server_name ~^(?.+)\.domain\.com$;
...
}复制代码
(5)若是以上都没有匹配,则使用default_server.若是没有指定default_server,则会选择第一个可用的server.咱们能够指定对于没有匹配的host值时,返回错误到客户端.能够用来防止别人把垃圾流量转到你的网站。
server {
listen 80 default_server;
server_name _; return 444;
}复制代码
经过返回444这个nginx的非标准错误码让nginx断开与浏览器的链接
location optional_modifier location_match {
...
}复制代码
其中可用的modifier修饰符以下
nginx首先检查基于前缀的location匹配(即不包含正则表达式的匹配)
若是有使用=修饰符的location块与请求的URL彻底匹配,则马上使用该location响应请求
若是没有找到带有=修饰符的location块匹配,则会继续计算非精确前缀,根据给定的URI找到最长匹配前缀,而后进行以下处理:
(1)若是最长的匹配location带有^~修饰符,nginx马上使用该location响应请求
(2)若是最长的匹配location不带有^~修饰符,nginx会将该匹配暂时存起来,而后继续后续匹配
在肯定并储存最长匹配的前缀location块后,nginx继续检查正则表达式匹配location(区分大小写/不区分大小写).若是存在正则表达式知足要求的匹配,则会选择与请求的URI匹配的第一个正则表达式的location来相应请求
若是没有找到与请求的URI匹配的正则表达式location,则使用以前存储的最长前缀location响应请求
一般状况下,一旦选择使用某一个location响应请求,那么请求将会在该location内部进行处理,而与其余location无关.可是location中某些指令会触发新的location匹配,好比:
(1)try_files
(2)rewrite
(3)error_page
关于为https配置default_server,参考Properly setting up a “default” nginx server for https
本文首发于公众号“小米运维”,点击查看原文。