`

Nginx如何处理一个请求

 
阅读更多


基于名字的虚拟主机

Nginx首先选定由哪一个虚拟主机来处理请求。让我们从一个简单的配置(其中全部3个虚拟主机都在端口*:80上监听)开始:

    server {
        listen      80;
        server_name example.org www.example.org;
        ...
    }

    server {
        listen      80;
        server_name example.net www.example.net;
        ...
    }

    server {
        listen      80;
        server_name example.com www.example.com;
        ...
    }

在这个配置中,nginx仅仅检查请求的“Host”头以决定该请求应由哪个虚拟主机来处理。如果Host头没有匹配任意一个虚拟主机,或者请求中根本没有包含Host头,那nginx会将请求分发到定义在此端口上的默认虚拟主机。在以上配置中,第一个被列出的虚拟主机即nginx的默认虚拟主机——这是nginx的默认行为。而且,可以显式地设置某个主机为默认虚拟主机,即在"listen"指令中设置"default_server"参数:

    server {
        listen      80 default_server;
        server_name example.net www.example.net;
        ...
    }

    "default_server"参数从0.8.21版开始可用。在之前的版本中,应该使用"default"参数代替。

请注意"default_server"是监听端口的属性,而不是主机名的属性。后面会对此有更多介绍。
如何防止处理未定义主机名的请求

如果不允许请求中缺少“Host”头,可以定义如下主机,丢弃这些请求:

    server {
        listen       80;
        server_name  "";
        return       444;
    }

在这里,我们设置主机名为空字符串以匹配未定义“Host”头的请求,而且返回了一个nginx特有的,非http标准的返回码444,它可以用来关闭连接。

    从0.8.48版本开始,这已成为主机名的默认设置,所以可以省略server_name ""。而之前的版本使用机器的hostname作为主机名的默认值。

基于域名和IP混合的虚拟主机

下面让我们来看一个复杂点的配置,在这个配置里,有几个虚拟主机在不同的地址上监听:

    server {
        listen      192.168.1.1:80;
        server_name example.org www.example.org;
        ...
    }

    server {
        listen      192.168.1.1:80;
        server_name example.net www.example.net;
        ...
    }

    server {
        listen      192.168.1.2:80;
        server_name example.com www.example.com;
        ...
    }

这个配置中,nginx首先测试请求的IP地址和端口是否匹配某个server配置块中的listen指令配置。接着nginx继续测试请求的Host头是否匹配这个server块中的某个server_name的值。如果主机名没有找到,nginx将把这个请求交给默认虚拟主机处理。例如,一个从192.168.1.1:80端口收到的访问www.example.com的请求将被监听192.168.1.1:80端口的默认虚拟主机处理,本例中就是第一个服务器,因为这个端口上没有定义名为www.example.com的虚拟主机。

默认服务器是监听端口的属性,所以不同的监听端口可以设置不同的默认服务器:

    server {
        listen      192.168.1.1:80;
        server_name example.org www.example.org;
        ...
    }

    server {
        listen      192.168.1.1:80 default_server;
        server_name example.net www.example.net;
        ...
    }

    server {
        listen      192.168.1.2:80 default_server;
        server_name example.com www.example.com;
        ...
    }

一个简单PHP站点配置

现在我们来看在一个典型的,简单的PHP站点中,nginx怎样为一个请求选择location来处理:

    server {
        listen      80;
        server_name example.org www.example.org;
        root        /data/www;

        location / {
            index   index.html index.php;
        }

        location ~* \.(gif|jpg|png)$ {
            expires 30d;
        }

        location ~ \.php$ {
            fastcgi_pass  localhost:9000;
            fastcgi_param SCRIPT_FILENAME
                          $document_root$fastcgi_script_name;
            include       fastcgi_params;
        }
    }

首先,nginx使用前缀匹配找出最准确的location,这一步nginx会忽略location在配置文件出现的顺序。上面的配置中,唯一的前缀匹配location是"/",而且因为它可以匹配任意的请求,所以被作为最后一个选择。接着,nginx继续按照配置中的顺序依次匹配正则表达式的location,匹配到第一个正则表达式后停止搜索。匹配到的location将被使用。如果没有匹配到正则表达式的location,则使用刚刚找到的最准确的前缀匹配的location。

请注意所有location匹配测试只使用请求的URI部分,而不使用参数部分。这是因为写参数的方法很多,比如:

    /index.php?user=john&page=1
    /index.php?page=1&user=john

除此以外,任何人在请求串中都可以随意添加字符串:

    /index.php?page=1&something+else&user=john

现在让我们来看使用上面的配置,请求是怎样被处理的:

    请求"/logo.gif"首先匹配上location "/",然后匹配上正则表达式"\.(gif|jpg|png)$"。因此,它将被后者处理。根据"root /data/www"指令,nginx将请求映射到文件/data/www/logo.gif",并发送这个文件到客户端。
    请求"/index.php"首先也匹配上location "/",然后匹配上正则表达式"\.(php)$"。 因此,它将被后者处理,进而被发送到监听在localhost:9000的FastCGI服务器。fastcgi_param指令将FastCGI的参数SCRIPT_FILENAME的值设置为"/data/www/index.php",接着FastCGI服务器执行这个文件。变量$document_root等于root指令设置的值,变量$fastcgi_script_name的值是请求的uri,"/index.php"。
    请求"/about.html"仅能匹配上location "/",因此,它将使用此location进行处理。根据"root /data/www"指令,nginx将请求映射到文件"/data/www/about.html",并发送这个文件到客户端。
    请求"/"的处理更为复杂。它仅能匹配上location "/",因此,它将使用此location进行处理。然后,index指令使用它的参数和"root /data/www"指令所组成的文件路径来检测对应的文件是否存在。如果文件/data/www/index.html不存在,而/data/www/index.php存在,此指令将执行一次内部重定向到"/index.php",接着nginx将重新寻找匹配"/index.php"的location,就好像这次请求是从客户端发过来一样。正如我们之前看到的那样,这个重定向的请求最终交给FastCGI服务器来处理。

作者: Igor Sysoev
编辑: Brian Mercer
翻译: Jinglong & cfsego
分享到:
评论

相关推荐

    Nginx中http请求处理过程

    这些数据结构是 Nginx 高效处理请求的基础。 1.2 HTTP 请求中 phase 的介绍 HTTP 请求中 phase 是指从客户端发送请求到 Nginx 服务器处理完毕的整个过程。这个过程可以分为多个阶段,每个阶段都有其特定的处理逻辑...

    详解nginx请求头数据读取流程

    6. 请求头结束:当遇到空行,即请求头结束时,Nginx会调用后续的处理函数,如检查是否需要读取请求体,然后继续处理请求。 整个请求头读取流程体现了Nginx作为高性能Web服务器的高效性和灵活性,能够快速处理大量...

    补充:Nginx之模块处理流程

    如果多个处理模块映射到了同一个位置,Nginx会在配置文件中处理冲突,确保只有一个模块处理请求。 3. **过滤模块(Filter Modules)**:过滤模块在处理模块之后介入,用于修改或增强处理模块生成的输出。例如,它们...

    nginx中一个请求的count计数跟踪浅析.docx

    这些方法是Nginx处理上游服务器交互的关键组件。`count`在`upstream`创建时增加了两次,一次是因为子请求本身,另一次是因为`upstream`的建立。 6. **请求结束与计数还原**: 当jtcmd模块完成其工作并收到上游服务...

    NGINX实现一个域名访问多个项目1

    标题中的"NGINX实现一个域名访问多个项目1"是指利用Nginx服务器的配置能力,让同一个域名能够根据不同的URL路径指向不同的应用或项目。描述中提到,这是为了解决在一个域名下部署多个项目的问题,避免为每个项目单独...

    第一个Nginx模块的例子

    标题中的“第一个Nginx模块的例子”意味着我们将探讨如何创建一个自定义的Nginx模块。Nginx是一个高性能的Web服务器和反向代理服务器,它以其轻量级、高并发处理能力而闻名。开发自定义模块可以让用户扩展Nginx的...

    nginx如何处理请求.pdf

    nginx如何处理请求

    详解nginx同一端口监听多个域名和同时监听http与https

    当有请求到达时,Nginx会根据HTTP请求头中的Host字段来决定将请求转发到哪一个server块。这里有一个重要的注意事项,就是所有server块中不能使用相同的端口监听跨实例的情况。也就是说,如果在同一个服务器上安装了...

    Nginx转发WebSocket接口配置方式

    3. **处理WebSocket标识符**:WebSocket连接中,每个连接都有一个唯一的`Sec-WebSocket-Key`和`Sec-WebSocket-Accept`头,Nginx会自动处理这些头,无需额外配置。 4. **代理缓冲区**:根据实际需求,可能需要调整...

    nginx一个域名多个项目部署.doc

    在IT行业中,Nginx是一个广泛应用的高性能Web服务器和反向代理服务器,尤其在处理静态内容和高并发场景下表现出色。本篇将详细介绍如何利用Nginx在一个域名下部署多个项目,以实现资源的有效管理和优化。 1. **...

    通过nginx实现跨域请求

    Nginx是一个高性能的HTTP和反向代理服务器,它常用于配置和管理网站的访问规则,包括处理跨域请求。 跨域请求是Web开发中常见的限制,由浏览器的同源策略实施。同源策略不允许一个域名下的文档或脚本获取或操作另一...

    nginx处理http请求实例详解

    这样,下次事件发生时,将会调用`ngx_http_request_handler`函数来处理请求,而不会再调用`ngx_http_process_request`。 在`ngx_http_process_request`函数内部,会进行一些关键的设置。例如,当读事件发生时,会...

    统计Nginx日志里前一个小时的IP数量以及IOS占比

    本话题将围绕“统计Nginx日志里前一个小时的IP数量以及iOS占比”这一主题展开,讲解如何处理Nginx日志中的时间戳,以及编写shell脚本来实现这一目标。 首先,Nginx默认的日志格式通常包含以下字段:远程主机...

    Nginx头部处理:掌握请求与响应的艺术

    Nginx(发音为 "engine-x")是一个高性能的HTTP和反向代理服务器,它以事件驱动和异步非阻塞的方式运行,能够处理数以万计的并发连接,同时保持低内存占用。Nginx最初由俄罗斯的程序员Igor Sysoev开发,并在2004年...

    nginx HTTP处理流程.docx

    3. **处理请求**:根据匹配结果,调用相应的HTTP模块处理请求。可能涉及静态文件服务、反向代理、负载均衡等操作。 4. **返回响应**:生成响应报文,发送回客户端。 5. **事件处理**:Nginx使用异步非阻塞I/O模型,...

    Nginx服务器中使用lua获取get或post参数.docx

    在处理请求体时,我们需要注意请求体的大小是否超过nginx配置中的client_body_buffer_size,如果超过的话,请求体将被缓冲到磁盘临时文件中。 此外,我们还可以使用ngx.req.get_body_file()来获取请求体的文件,...

    Nginx 配置文件 nginx.conf 详解

    `,这将设置 Nginx 服务器只有一个工作进程。 错误日志 错误日志是 Nginx 服务器记录错误信息的重要组件,我们可以通过 `error_log` 指令来设置错误日志的存放路径和级别,例如 `error_log logs/error.log;`,这将...

    nginx lua处理图片

    2. **配置Nginx**:在`nginx.conf`文件中,你需要设置一个location块来处理图片请求。在这个location块内,可以使用`content_by_lua_file`指令来指定一个Lua脚本文件,比如`thumbnail.lua`或`config.lua`,这样Nginx...

    windows下nginx配置https以及同一个端口监听多个网站即监听多个虚拟主机

    Nginx是一个高性能的HTTP和反向代理服务器,以其稳定性、高并发处理能力而受到广泛使用。下面将详细阐述如何进行这项配置。 首先,了解HTTPS协议。HTTPS是HTTP(超文本传输协议)的安全版本,它通过SSL/TLS协议加密...

    跨域WebService请求-Nginx_SOAP服务_Ajax客户端.docx

    【Nginx反向代理】为了解决这个问题,Nginx作为一个反向代理服务器可以被用来作为中间层,将客户端的请求转发到目标服务器。Nginx可以配置为将来自不同源的请求视为同源,从而规避跨域限制。在本案例中,Nginx配置...

Global site tag (gtag.js) - Google Analytics