Nginx -- Web服务器

历史:2004年俄罗斯人 伊戈尔·赛索耶夫 发布Nginx
官网:http://nginx.org/
第三方模块:https://www.nginx.com/resources/wiki/modules/index.html

概念
正向代理:客户端访问服务器地址,本地的代理模块会把请求改造成访问代理的地址,通过代理访问实际的服务器
反向代理:客户端访问代理的地址,通过代理访问实际的服务器
Nagle算法:将要发送的数据存放在缓存里,当积累到一定量或一定时间,再将它们发送出去
Lua:一种轻量小巧的脚本语言
URI:Uniform Resource Identifier,统一资源标识符,标识一个资源,例如 cay@horstman.com
URL:Uniform Resource Locator,统一资源定位符,定位一个资源,例如http://www.test.com/index.html,是URI的子集

Nginx作为LB 与 HAProxy对比
1、Nginx除了LB 还有 Web 服务器功能
2、HAProxy是单进程,Nginx是多进程,可以利用多核
3、Nginx社区更活跃

Nginx作为Web服务器 与 Apache HTTP Server 对比
1、Nginx轻量级,占用资源少
2、Apache是同步多进程模型,一个连接对应一个进程;Nginx是异步多进程模式,多个连接对应一个进程,支持更多的并发连接
3、Nginx处理静态文件效率高
4、Nginx 配置简洁

CentOs安装:yum install epel-release; yum install nginx
Ubuntu安装:apt-get update; apt-get install nginx
tar.gz安装

tar -zxvf nginx.tar.gz
./configure --without-http_rewrite_module --without-http_gzip_module  # 配置,不联网时忽略rewrite 和 gzip 模块
make
make install   # 会提示所安装的目录
./sbin/nginx  # 在nginx 安装目录里 执行启动

启停:
service nginx start|restart|stop
nginx -s reload|reopen|stop|quit
配置文件检查:nginx -t

配置

/etc/conf/nginx.conf

user nginx;                       # worker进程所属的用户组
worker_processes auto;  # worker进程数量
error_log /var/log/nginx/error.log;  # 错误日志
pid /run/nginx.pid;            # 记录主进程ID的文件

include /usr/share/nginx/modules/*.conf;  # 引入其他配置文件

events {
    worker_connections 1024;    # 一个worker的最大连接数
}

http {
    log_format  main  '$remote_addr - $remote_user [$time_local] "$request" '
                      '$status $body_bytes_sent "$http_referer" '
                      '"$http_user_agent" "$http_x_forwarded_for"';
    # $upstream_addr 上游地址
    access_log  /var/log/nginx/access.log  main;   # 访问日志 路径 和 格式
    # access_log off; # 关闭访问日志

    sendfile            on;                    # 把文件数据从磁盘直接传到套接字,不经过buffer
    tcp_nopush          on;               # 启用套接字TCP_CORK选项,即sendfile时尽可能一次发完
    tcp_nodelay         on;               # 禁用Nagle算法
    keepalive_timeout   65;           # 在响应头中设置keepalive
    types_hash_max_size 2048;
    client_max_body_size   20m;  # 最大上传量,也可以放在server{} 或  location{} 里

    include             /etc/nginx/mime.types;
    default_type        application/octet-stream;

    include /etc/nginx/conf.d/*.conf;

    server {     # 一个虚拟服务器
        listen       80 default_server;   # default_server表示对80端口的请求,任何 server_name都不匹配时,选用这个server
        listen       [::]:80 default_server;  # 监听IPv6的80端口
        server_name  www.test.com;     # 这个server绑定的域名
        root         /usr/share/nginx/html;  # 静态文件根目录,不设置root,则默认为 /usr/share/nginx/html

        include /etc/nginx/default.d/*.conf;

       if ($scheme = http) {
            return 301 https://$server_name$request_uri;  # 将http重定向到https,要写在监听80端口的server{}下
        }
        location ^~ /static/ {  
            ## 通过 expires 设置 Expires 和 Cache-Control 两个字段,指导浏览器进行缓存验证
            expires 24h;   # 缓存,缓存有效期内不进行服务器端验证
            expires  max; # 使得 Expires 和 Cache-Control 为 10 年
            expires -1s; 或 epoch; # 使得 Cache-Control 为 no-cache,缓存,但缓存过期,进行服务端验证
            ## 单独设置 Cache-Control
            add_header Cache-Control "no-store";  # 禁用缓存
            ## 服务端验证 设置
            if_modified_since exact;  # 提供Last-Modified,处理If-Modified-Since,精确度是秒,HTTP 1.0 就支持
            etag on;  # 提供Etag(跟最后修改时间、文件大小有关),处理If-None-Match,HTTP 1.1 才支持
            # if_modified_since 和 etag 都没变化才会返回304,两项一般会同时变化,先验证etag变化就返回200

            root /webroot/static/;     # 访问本地文件, 该目录下如果没有index文件,会出现403
        }  
    location /contract {
                // alias 和 root 二选一
               alias /usr/;  # 实际访问的是 /usr/
               root /usr/;   # 实际访问的是 /usr/contract 
               add_header Access-Control-Allow-Origin *;  # 允许跨域访问
        }
        location / {
           proxy_pass http://tomcat:8080/   # 转发给后端服务器,如果这里使用了变量,则需要配置 resolver 来指定DNS
           proxy_read_timeout  3600;   # 超时时间
        }

        error_page 403 404 /404.html;  # 碰到403、404 则 跳转到 /404.html
        location = /404.html {
            root /usr/static;
        }

        error_page 500 502 503 504 /50x.html;
            location = /50x.html {
        }

        error_page 403 404 =301 $scheme://$host:$server_port/;   # 重定向
    }
    server{
        # 第二个server
    }
}

location匹配规则
精确匹配:=,例 location = /login
匹配路径:^~,例 location ^~ /static/ 匹配 /static/a.html
区分大小写的正则匹配:~,location ~ .(gif|jpg|png|js|css)$
不区分大小写的正则匹配:~*
区分大小写的正则不匹配:!~
不区分大小写的正则不匹配:!~*
通配:/

匹配优先级:=、^~、正则匹配(按书写顺序)、/

全局变量

$args: 请求行中问号后面的参数
$content_length: 请求头中的Content-length字段。
$content_type: 请求头中的Content-Type字段。
$host: 请求目标主机
$http_user_agent: 客户端agent信息
$http_cookie: 客户端cookie信息
$request_method: 客户端请求的方法
$remote_addr: 客户端的IP地址。
$remote_port: 客户端的端口。
$request_filename: 当前所请求文件的路径
$scheme: HTTP协议(如http,https)
$server_protocol: 请求使用的协议,通常是HTTP/1.0或HTTP/1.1。
$server_port: 请求到达服务器的端口号。
$request_uri: 请求参数的原始URI,不包含主机名,如:”/foo/bar.php?arg=baz”。
$uri: 不带参数的当前URI,不包含主机名,如”/foo/bar.html”。
$document_uri: 与$uri相同

反向代理

location ^~ /uri {
    proxy_pass http://tomcat:8080/newuri;   # 转发时可以改变端口 和 路径,如果location是正则匹配,转发时路径不会变
    
   proxy_http_version 1.1  # 默认为1.0
    # 设置给上游服务器的 头
    proxy_set_header Host $host;                            # 客户端请求的目标host,即Nginx
    proxy_set_header Proxy-Host $proxy_port;       # Nginx请求的目标host,即上游服务器
    proxy_set_header X-Real-IP $remote_addr;      # 客户端host
    proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; # 请求头中的X-Forwarded-For + $remote_addr
    # client -> nginx -> nginx 结构中,第一台nginx的$remote_addr是客户端,$proxy_add_x_forwarded_for也为$remote_addr,第二台nginx的$remote_addr为第一台nginx,$proxy_add_x_forwarded_for为第一台nginx的$proxy_add_x_forwarded_for + 第一台nginx,即客户端 + 第一台nginx
    sub_filter 'a' 'b';  # 响应内容替换
    sub_filter_once off;  # 非单处替换
}
upstream tomcats{   # 定义上游服务器
    server 10.0.100.10:11211;
    server 10.0.100.20:11211;
    keepalive 64;
}
location ^~ /uri {
    proxy_pass http://tomcat;   # 使用upstream,转发时会有 /uri
    proxy_pass http://tomcat/;   # 转发时没有 /uri
}

负载均衡算法
1、轮询(默认算法)
2、IP hash
3、最少连接数

upstream tomcats{   # 定义上游服务器
    server 10.0.100.10:11211;
    server 10.0.100.20:11211;
    least_conn;   # 开启最少连接数 算法
}

HTTPS

server {
    listen 443 default ssl;
    server_name www.example.com example.com;

    ssl_session_cache shared:SSL:1m;
    ssl_session_timeout  10m;
    ssl_ciphers HIGH:!aNULL:!MD5;
    ssl_prefer_server_ciphers on;

    ssl_certificate /usr/local/etc/nginx/www.example.com.crt;  # 证书
    ssl_ certificate_ key /usr/local/etc/nginx/www.example.com.key;  # 私钥
或
        ssl_certificate "/etc/nginx/cert/5810237_example.com.pem";
        ssl_certificate_key "/etc/nginx/cert/5810237_example.com.key";

    location / {
        # https页面 访问的资源也必须是 https
        add_header Content-Security-Policy upgrade-insecure-requests;  # 如果html引用了http资源,强制改为https
        proxy_set_header X-FORWARDED-PROTO https;  # 让上游服务器知道原始请求是https
        proxy_pass http://upstream;  # nginx和上游之间不用https
        # proxy_pass https://upstream;  # nginx和上游之间也用https
    }
}

gzip 压缩

http {
    gzip on;
    gzip_http_version 1.0;
    gzip_comp_level 2;
    gzip_disable msie6;

    gzip_types text/plain text/css application/x-javascript text/xml
        application/xml application/xml+rss text/javascript
        application/javascript application/json;
}

rewrite

执行顺序
1、执行server块的rewrite
2、匹配location
3、执行location块的rewrite,如果URL被重写,则重新执行1-3,直到匹配location 并 不被 rewrite
4、如果循环超过10次,则返回500 Internal Server Error错误

格式

rewrite 正则 新串 [标志位];
例:rewrite ^/images/(.*)_(\d+)x(\d+)\.(png|jpg|gif)$ /resizer/$1.$4?width=$2&height=$3? last;
含义:原url中的路径部分 去匹配 正则,把匹配串替换为新串,新串中的$1即引用匹配串中的第1个分组(小括号)

标志位flag
last: 后续只匹配,不再rewrite
break: 用在location中,不再rewrite,采用当前location
redirect: 返回302临时重定向,地址栏会显示跳转后的地址
permanent: 返回301永久重定向,地址栏会显示跳转后的地址

server {
    if ($request_method = POST) {    # if里的判断运算符 与 location 判断运算符 含义一致
        return 405;     # 直接返回状态码
    }
    location / {
        rewrite ^/listings/(.*)$ /$remote_port.html?listing=$1 last;  # 新串中可以引用全局变量$remote_port等
        
    set $flag 0;  # 由于 if 不能嵌套使用 也 不能多重条件,因此用 变量来实现复杂条件
        if ($http_user_agent ~* "(Android|iPhone|Windows Phone|UC|Kindle|Mobile)"){  # 移动端,正则表达式匹配
            set $flag "${flag}1";
        }
        if ($request_uri !~ /mobile){   # 正则表达式不匹配
            set $flag "${flag}1";
        }
       if ($request_uri !~ \.){   # 正则表达式 转义
            set $flag "${flag}1";
        }
        if ($flag = "0111"){
            return 301 http://$server_name/mobile/index;  # 重定向
        }

        proxy_pass http://upstream;
    }


    location / {
        try_files $uri $uri/ /index.html;
    }
        
    location = / {
        set $flag 0;
            if ($http_user_agent ~* "(Android|iPhone|Windows Phone|UC|Kindle|Mobile)"){
                set $flag "${flag}1";
            }
            if ($flag = "01"){
                return 301 http://$server_name/mobile/index;    # if 会使 try_files 失效
            }
        if ($flag != "01"){
                return 301 http://$server_name/home;
            }
    }
}

try_file

location ^~ /test/ {
    index index.html;
    # try_files 表示找到一个可用的文件返回
    try_files /2.html /1.html /test/test2.html @bd;  # 引用location
}
        
location @bd {      # 定义location
    rewrite ^/(.*)$ http://www.google.com;
}

案例

1、小程序里需要打开第三方网页,用我方域名代理第三方域名

location ~ "^\/.*\..*\/" {    # 识别出 "https://我方域名/某域名/path" 格式的请求;需要转义的正则表达式得用引号包起来
       proxy_pass https:/$request_uri;  # 拼接出 "https://某域名/path"
       proxy_set_header Referer 'www.目标.com';
}

location / {
  proxy_pass https://www.目标.com;
  proxy_set_header Referer 'www.目标.com';
        
  proxy_set_header Accept-Encoding "";  # 避免gzip,使得sub_filter能正常工作
  sub_filter_once off;   # 替换所有
  sub_filter https:// https://www.我方.cn/;   # 从目标获取的响应内容里,在所有写明的域名前加一个我方域名,形成https://我方域名/某域名/path;避免页面直接向别的域名发送请求,造成跨域限制(CORS、防盗链)
    }
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,911评论 5 460
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 82,014评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 142,129评论 0 320
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,283评论 1 264
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,159评论 4 357
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,161评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,565评论 3 382
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,251评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,531评论 1 292
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,619评论 2 310
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,383评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,255评论 3 313
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,624评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,916评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,199评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,553评论 2 342
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,756评论 2 335