• Nginx反向代理与负载均衡


    Nginx反向代理与负载均衡

     

    集群简介

    简单地说,集群就是指一组(若干个)相互独立的计算机,利用高速通信网络组成的一个较大的计算机服务系统,每个集群节点(即集群中的每台计算机)都是运行各自服务的独立服务器。这些服务器之间可以彼此通信,协同向用户提供应用程序,系统资源和数据,并以单一系统的模式加以管理。当用户客户机请求集群系统时,集群给用户的感觉就是一个单一独立的服务器,而实际上用户请求的是一组集群服务器。

    打开谷歌,百度的页面,看起来好简单,也许你觉得用几分钟就可以制作出相似的网页,而实际上,这个页面的背后是由成千上万台服务器集群协同工作的结果。而这么多的服务器维护和管理,以及相互协调工作也许就是同学们未来的工作职责了。

    若要用一句话描述集群,即一堆服务器合作做同一件事,这些机器可能需要整个技术团队架构,设计和统一协调管理,这些机器可以分布在一个机房,也可以分布在全国全球各个地区的多个机房。

     

    为什么要使用集群

    (1)高性能

    一些国家重要的计算密集型应用(如天气预报,核试验模拟等),需要计算机有很强的运算处理能力。以全世界现有的技术,即使是大型机,其计算能力也是有限的,很难单独完成此任务。因为计算时间可能会相当长,也许几天,甚至几年或更久。因此,对于这类复杂的计算业务,便使用了计算机集群技术,集中几十上百台,甚至成千上万台计算机进行计算。

    假如你配一个LNMP环境,每次只需要服务10个并发请求,那么单台服务器一定会比多个服务器集群要快。只有当并发或总请求数量超过单台服务器的承受能力时,服务器集群才会体现出优势。

    集群高性能。.png-273.6kB

    (2)价格有效性

    通常一套系统集群架构,只需要几台或数十台服务器主机即可。与动辄价值上百万元的专用超级计算机相比便宜了很多。在达到同样性能需求的条件下,采用计算机集群架构比采用同等运算能力的大型计算机具有更高的性价比。

    早期的淘宝,支付宝的数据库等核心系统就是使用上百万元的小型机服务器。后因使用维护成本太高以及扩展设备费用成几何级数翻倍,甚至成为扩展瓶颈,人员维护也十分困难,最终使用PC服务器集群替换之,比如,把数据库系统从小机结合Oracle数据库迁移到MySQL开源数据库结合PC服务器上来。不但成本下降了,扩展和维护也更容易了。

    (3)可伸缩性

    当服务负载,压力增长时,针对集群系统进行较简单的扩展即可满足需求,且不会降低服务质量。

    通常情况下,硬件设备若想扩展性能,不得不增加新的CPU和存储器设备,如果加不上去了,就不得不够买更高性能的服务器,就拿我们现在的服务器来讲,可以增加的设备总是有限的。如果采用集群技术,则只需要将新的单个服务器加入现有集群架构中即可,从访问的客户角度来看,系统服务无论是连续性还是性能上都几乎没有变化,系统在不知不觉中完成了升级,加大了访问能力,轻松地实现了扩展。集群系统中的节点数目可以增长到几千乃至上万个,其伸缩性远超过单台超级计算机。

    (4)高可用性

    单一的计算机系统总会面临设备损毁的问题,如CPU,内存,主板,电源,硬盘等,只要一个部件坏掉,这个计算机系统就可能会宕机,无法正常提供服务。在集群系统中,尽管部分硬件和软件也还是会发生故障,但整个系统的服务可以是7*24小时可用的。

    集群架构技术可以使得系统在若干硬件设备故障发生时仍可以继续工作,这样就将系统的停机时间减少到了最小。集群系统在提高系统可靠性的同时,也大大减小了系统故障带来的业务损失,目前几乎100%的互联网网站都要求7*24小时提供服务。

    (5)透明性

    多个独立计算机组成的松耦合集群系统构成一个虚拟服务器。用户或客户端程序访问集群系统时,就像访问一台高性能,高可用的服务器一样,集群中一部分服务器的上线,下线不会中断整个系统服务,这对用户也是透明的。

    (6)可管理性

    整个系统可能在物理上很大,但其实容易管理,就像管理一个单一映像系统一样。在理想状况下,软硬件模块的插入能做到即插即用。

    (7)可编程性

    在集群系统上,容易开发及修改各类应用程序。

     

    集群的常见分类

    负载均衡集群,简称LBC或者LB

    高可用性集群,简称HAC --->预防单点也就是高可用

    高性能计算集群,简称HPC

    网格计算集群

    负载均衡集群和高可用性集群是互联网行业常用的集群架构模式,也是我们要学习的重点。

     

    负载均衡集群的作用

    分摊用户访问请求及数据流量(负载均衡)

    保持业务连续性,即7*24小时服务(高可用性)

    应用于Web业务及数据库从库等服务器的业务

    负载均衡集群典型的开源软件包括LVS,Nginx,Haproxy等,如下图所示

    负载均衡集群。.png-200.1kB

     

    负载均衡集群介绍

    负载均衡集群为企业提供了更为实用,性价比更高的系统架构解决方案。负载均衡集群可以把很多客户集中的访问请求负载压力尽可能平均地分摊在计算机集群中处理。客户访问请求负载通常包括应用程序处理负载和网络流量负载。这样的系统非常适合使用同一组应用程序为大量用户提供服务的模式,每个节点都可以承担一定的访问请求负载压力,并且可以实现访问请求在各节点之间动态分配,以实现负载均衡。

    负载均衡集群运行时,一般是通过一个或多个前端负载均衡器将客户访问请求分发到后端的一组服务器上,从而达到整个系统的高性能和高可用性。一般高可用性集群和负载均衡集群会使用类似的技术,或同时具有高可用性与负载均衡的特点。

     

    高可用性集群的作用

    当一台机器宕机时,另外一台机器接管宕机的机器的IP资源和服务资源,提供服务。

    常用于不易实现负载均衡的应用,比如负载均衡器,主数据库,主存储对之间。

    高可用性集群常用的开源软件包括Keepalived,Heartbeat等,其架构图如下图所示

    高可用性集群。.png-205.4kB

     

    高可用性集群介绍

    一般是指在集群中任意一个节点失效的情况下,该节点上的所有任务会自动转移到其他正常的节点上。此过程并不影响整个集群的运行。

    当集群中的一个节点系统发生故障时,运行着的集群服务会迅速作出反应,将该系统的服务分配到集群中其他正在工作的系统上运行。考虑到计算机硬件和软件的容错性,高可用性集群的主要目的是使集群的整体服务尽可能可用。如果高可用性集群中的主节点发生了故障,那么这段时间内将由备节点代替它。备节点通常是主节点的镜像。当它代替主节点时,它可以完全接管主节点(包括IP地址及其他资源)提供服务,因此,使集群系统环境对于用户来说是一致的,既不会影响用户的访问。

    高可用性集群使服务器系统的运行速度和响应速度会尽可能的快。他们经常利用在多台机器上运行的冗余节点和服务来相互跟踪。如果某个节点失败,它的替补者将在几秒钟或更短时间内接管它的职责。因此,对于用户而言,集群里的任意一台机器宕机,业务都不会受影响(理论情况下)。

     

    Nginx负载均衡集群介绍

     

    搭建负载均衡服务的需求

    负载均衡集群提供了一种廉价,有效,透明的方法,来扩展网络设备和服务器的负载,带宽和吞吐量,同时加强了网络数据处理能力,提高了网络的灵活性和可用性。

    搭建负载均衡服务的需求如下

    (1)把单台计算机无法承受的大规模并发访问或数据流量分担到多台节点设备上,分别进行处理,减少用户等待响应的时间,提升用户体验。

    (2)单个重负载的运算分担到多台节点设备上做并行处理,每个节点设备处理结束后,将结果汇总,返回给用户,系统处理能力得到大幅度提高。

    (3)7*24小时的服务保证,任意一个或多个有限后面节点设备宕机,不能影响业务。

    在负载均衡集群中,同组集群的所有计算机节点都应该提供相同的服务。集群负载均衡器会截获所有对该服务的入站请求。然后将这些请求尽可能地平均地分配在所有集群节点上。

     

    反向代理与负载均衡概念简介

     

    反向代理与负载均衡概念简介

    严格地说,Nginx仅仅是作为Nginx Proxy反向代理使用的,因为这个反向代理功能表现的效果是负载均衡集群的效果,所以本文称之为Nginx负载均衡。那么,反向代理和负载均衡有什么区别呢?

    普通负载均衡软件,例如大名鼎鼎的LVS,其实功能只是对请求数据包的转发(也可能会改写数据包),传递,其中DR模式明显的特征是从负载均衡下面的节点服务器来看,接收到的请求还是来自访问负载均衡器的客户端的真实用户,而反向代理就不一样了,反向代理接收访问用户的请求后,会代理用户重新发起请求代理下的节点服务器,最后把数据返回给客户端用户,在节点服务器看来,访问的节点服务器的客户端用户就是反向代理服务器了,而非真实的网站访问用户。

    一句话,LVS等的负载均衡是转发用户请求的数据包,而Nginx反向代理是接收用户的请求然后重新发起请求去请求其后面的节点。

    Nginx是七层结构的,通过nginx本身来转发,可以有多种方式进行负载均衡,实现的功能多,nginx可以分业务

    LVS是四层结构的,通过转发数据包mac或ip头部,不能对数据进行修改,来实现的负载均衡,实现的功能少但效率高,LVS的WEB内容都给是一样的

     

    实现Nginx负载均衡的组件主要有两个,如下表

    负债均衡组件。.png-45.9kB

     

    反向代理与负载均衡搭建

     

    硬件准备

    准备3台VM虚拟机(有物理服务器更佳)都需要安装Nginx服务

    Nginx反向代理

    NginxWebA

    NginxWebB

    image_1crp57ne91mjek0c1q9v1js41iahs.png-6.8kB

     

    安装依赖软件包

    yum -y install openssl openssl-devel pcre pcre-devel

    rpm -qa openssl openssl-devel pcre pcre-devel

    三台都需要安装支持包

    image_1crp58vobfo7pno7v71usl9pq19.png-20.2kB

     

    软件包准备

    nginx-1.10.2.tar.gz

    image_1crp5ad1rtscc221q9v1hm51j9v1m.png-10.5kB

     

    安装Nginx服务

    useradd -M -s /sbin/nologin nginx

    tar xf nginx-1.10.2.tar.gz -C /usr/src

    cd /usr/src/nginx-1.10.2

    ./configure --user=nginx --group=nginx --prefix=/usr/local/nginx --with-http_stub_status_module --with-http_ssl_module && make && make install

    ln -s /usr/local/nginx/sbin/* /usr/local/sbin/ --->软链接可做可不做

    egrep -v "#|^$" nginx.conf.default > nginx.conf --->把配置文件多余的东西过滤掉

    这里只编译了一个截图,三台虚拟机的步骤都是一样的

    image_1crp6k368e75d611ho2oud9mm9.png-51.9kB

     

    简单的Nginx反向代理配置文件

     
    1. worker_processes 1;
    2. events {
    3. worker_connections 1024;
    4. }
    5. http {
    6. include mime.types;
    7. default_type application/octet-stream;
    8. sendfile on;
    9. keepalive_timeout 65;
    10. upstream www_server {
    11. server 192.168.200.68:80 weight=1;
    12. server 192.168.200.69:80 weight=1;
    13. }
    14. server {
    15. listen 80;
    16. server_name sl.yunjisuan.com;
    17. location / {
    18. proxy_pass http://www_server;
    19. }
    20. }
    21. }
     

    配置文件详解

     
    1. #这里定义Web服务器池,包含了68,69两个Web节点
    2. upstream --->服务器池
    3. www_server --->是服务器池的名字
    4. server --->就是一个RS节点
    5. weight --->权重,负载均衡是按照权重比分配的
    6. #这里定义代理的负载均衡域名虚拟主机
    7. proxy_pass --->访问sl.yunjisuan.com,请求发送给www_server里面的
     

    WEBA配置文件

     
    1. worker_processes 1;
    2. events {
    3. worker_connections 1024;
    4. }
    5. http {
    6. include mime.types;
    7. default_type application/octet-stream;
    8. sendfile on;
    9. keepalive_timeout 65;
    10. server_tokens off;
    11. server {
    12. listen 80;
    13. server_name bbs.yunjisuan.com;
    14. location / {
    15. root html/bbs;
    16. index index.html index.htm;
    17. }
    18. }
    19. server {
    20. listen 80;
    21. server_name sl.yunjisuan.com;
    22. location / {
    23. root html/sl;
    24. index index.html index.htm;
    25. }
    26. }
    27. }

    WEBB跟WEBA配置文件是一样的

     

    填充测试文件

     
    1. mkdir /usr/local/nginx/html/{sl,bbs}
    2. echo "`hostname -I ` sl" >> /usr/local/nginx/html/sl/index.html
    3. echo "`hostname -I ` bbs" >> /usr/local/nginx/html/bbs/index.html

    WEBB跟WEBA测试文件是一样的

     

    添加WEB-A和B虚拟机本地映射

    vim /etc/hosts

    cat /etc/hosts

    image_1crrauh48kkq1ke212k313vu1ftc1p.png-19.5kB

    image_1crrauu0c19j9mhh16q4q083h726.png-19.6kB

     

    配置完两台WEB服务器用curl测试一下

    curl sl.yunjisuan.com

    curl bbs.yunjisuan.com

    image_1crrb5917g3d14li1hhk1k9065l2j.png-16.3kB

    image_1crrb5n6nf1b1bh4jj1bqj22m3g.png-16.9kB

     

    检查两台WEB简单的负载均衡测试结果

    curl sl.yunjisuan.com

    测试结果可以看出来。两个Web节点按照1:1的比例被访问

    image_1crrbvs7i1lm79udopc1ti81p553t.png-32.8kB

     

    Nginx upstream模块介绍

    Nginx的负载均衡功能依赖于ngx_http_upsteam_module模块,所支持的代理方式包括proxy_pass,fastcgi_pass,memcached_pass等,新版Nginx软件支持的方式有所增加。本文主要讲解proxy_pass代理方式。

    ngx_http_upstream_module模块允许Nginx定义一组或多组节点服务器组,使用时可以通过proxy_pass代理方式把网站的请求发送到事先定义好的对应Upstream组的名字上,具体写法为“proxy_pass http:// www_server_pools”,其中www_server_pools就是一个Upstream节点服务器组名字。ngx_http_upstream_module模块官方地址为:http://nginx.org/en/docs/http/ngx_http_upstream_module.html

     

    基本的upstream配置案例

     
    1. upstream www_server_pools {
    2. # upstream是关键字必须有,后面的www_server_pools为一个Upstream集群组的名字,可以自己起名,调用时就用这个名字
    3. server 192.168.0.223:80 weight=5;
    4. server 192.168.0.224:80 weight=10;
    5. server 192.168.0.225:80 weight=15;
    6. # server关键字是固定的,后面可以接域名(门户会用)或IP。如果不指定端口,默认是80端口。weight代表权重,数值越大被分配的请求越多,结尾有分号,别忘了。
    7. }
     

    较完整的upstream配置案例

     
    1. upstream blog_server_pool {
    2. server 192.168.0.223; #这行标签和下行是等价的
    3. server 192.168.0.224:80 weight=1 max_fails=1 fail_timeout=10s; #这行标签和上一行是等价的,此行多余的部分就是默认配置,不写也可以。
    4. server 192.168.0.225:80 weight=1 max_fails=2 fail_timeout=20s backup;
    5. # server最后面可以加很多参数,具体参数作用看下文的表格
    6. }
     

    使用域名及socket的upstream配置案例

     
    1. upstream backend {
    2. server backend1.example.com weight=5;
    3. server backend2.example.com:8080; #域名加端口。转发到后端的指定端口上
    4. server unix:/tmp/backend3; #指定socket文件
    5. #提示:server后面如果接域名,需要内网有DNS服务器或者在负载均衡器的hosts文件做域名解析。
    6. server 192.168.0.223;
    7. server 192.168.0.224:8080;
    8. server backup1.example.com:8080 backup;
    9. #备份服务器,等上面指定的服务器都不可访问的时候会启动,backup的用法和Haproxy中用法一样
    10. server backup2.example.com:8080 backup;
    11. }
     

    权重比测试

     
    1. worker_processes 1;
    2. events {
    3. worker_connections 1024;
    4. }
    5. http {
    6. include mime.types;
    7. default_type application/octet-stream;
    8. sendfile on;
    9. keepalive_timeout 65;
    10. upstream www_server {
    11. server 192.168.200.68:80 weight=5;
    12. server 192.168.200.69:80 weight=1;
    13. }
    14. server {
    15. listen 80;
    16. server_name sl.yunjisuan.com;
    17. location / {
    18. proxy_pass http://www_server;
    19. }
    20. }
    21. }

    测试阶段多少会有点延迟,所有显示的会不太准确

    image_1crrid56v1apt15b861qrg6dtc8k.png-64.5kB

     

    backup备份测试

     
    1. worker_processes 1;
    2. events {
    3. worker_connections 1024;
    4. }
    5. http {
    6. include mime.types;
    7. default_type application/octet-stream;
    8. sendfile on;
    9. keepalive_timeout 65;
    10. upstream www_server {
    11. server 192.168.200.68:80 weight=1 backup;
    12. server 192.168.200.69:80 weight=1;
    13. }
    14. server {
    15. listen 80;
    16. server_name sl.yunjisuan.com;
    17. location / {
    18. proxy_pass http://www_server;
    19. }
    20. }
    21. }

    image_1crriptk41a25150i3lm834ba9h.png-40.7kB

    image_1crriq7ua4ig1pped9rg7j1brs9u.png-14.5kB

     

    特别提示

    如果是两台Web服务器做高可用,常规方案就需要keepalived配合,那么这里使用Nginx的backup参数通过负载均衡功能就可以实现Web服务器集群了,对于企业应用来说,能做集群就不做高可用。

     

    upstream模块相关说明

    upstream模块的内容应放于nginx.conf配置的http{}标签内,其默认调度节点算法是wrr(weighted round-robin,即权重轮询)。下图为upstream模块内部server标签部分参数说明

    image_1crrd4dae1loa19ok1h1el1815cm4a.png-433.8kB

    提示: 
    以上参数与专业的Haproxy参数很类似,但不如Haproxy的参数易懂。

     

    示例,如下

     
    1. upstream backend {
    2. server backend1.example.com weight=5; #如果就是单个Server,没必要设置权重
    3. server 127.0.0.1:8080 max_fail=5 fail_timeout=10s;
    4. #当检测次数等于5的时候,5次连续检测失败后,间隔10s再重新检测。
    5. server unix:/tmp/backend3;
    6. server backup1.example.com:8080 backup; #热备机器设置
    7. }
     

    hash算法访问

     
    1. upstream backend {
    2. ip_hash;
    3. server backend1.example.com;
    4. server backend2.example.com;
    5. server backend3.example.com down;
    6. server backend4.example.com;
    7. }

    需要特别说明的是,如果是Nginx代理Cache服务,可能需要使用hash算法,此时若宕机,可通过设置down参数确保客户端用户按照当前的hash算法访问,这一点很重要

     

    开启对后端服务器的健康检测

     
    1. #开启对后端服务器的健康检测,通过GET /test/index.php来判断后端服务器的健康情况
    2. server php_server_1 192.168.0.223:80 cookie 1 check inter 2000 rise 3 fall 3 weight 2
    3. server php_server_2 192.168.0.224:80 cookie 2 check inter 2000 rise 3 fall 3 weight 1
    4. server php_server_bak 192.168.0.225:80 cookie 3 check inter 1500 rise 3 fall 3 backup
     

    健康检测命令详解

     
    1. weight:调节服务器的请求分配权重。
    2. check:开启对该服务器健康检查。
    3. inter:设置连续两次的健康检查间隔时间,单位毫秒,默认值2000
    4. rise:指定多少次连续成功的健康检查后,即可认定该服务器处于可用状态。
    5. fall:指定多少次不成功的健康检查后,即认为服务器为宕机状态,默认值3.
    6. maxconn:指定可被发送到该服务器的最大并发连接数。
     

    upstream模块调度算法

    调度算法一般分为两类: 
    第一类为静态调度算法,即负载均衡器根据自身设定的规则进行分配,不需要考虑后端节点服务器的情况,例如:rr,wrr,ip_hash等都属于静态调度算法。 
    第二类为动态调度算法,即负载均衡器会根据后端节点的当前状态来决定是否分发请求,例如:连接数少的优先获得请求,响应时间短的优先获得请求。例如:least_conn,fair等都属于动态调度算法。

    (1) rr轮询(默认调度算法,静态调度算法)

    按客户端请求顺序把客户端的请求逐一分配到不同的后端节点服务器,这相当于LVS中的rr算法,如果后端节点服务器宕机(默认情况下Nginx只检测80端口),宕机的服务器会被自动从节点服务器池中剔除,以使客户端的用户访问不受影响。新的请求会分配给正常的服务器。

    (2)wrr(权重轮询,静态调度算法)

    在rr轮询算法的基础上加上权重,即为权重轮询算法,当使用该算法时,权重和用户访问成正比,权重值越大,被转发的请求也就越多。可以根据服务器的配置和性能指定权重值大小,有效解决新旧服务器性能不均带来的请求分配问题。

    (3)ip_hash(静态调度算法)(会话保持)

    每个请求按客户端IP的hash结果分配,当新的请求到达时,先将其客户端IP通过哈希算法哈希出一个值,在随后的客户端请求中,客户IP的哈希值只要相同,就会被分配至同一台服务器,该调度算法可以解决动态网页的session共享问题,但有时会导致请求分配不均,即无法保证1:1的负载均衡,因为在国内大多数公司都是NAT上网模式,多个客户端会对应一个外部IP,所以,这些客户端都会被分配到同一节点服务器,从而导致请求分配不均。LVS负载均衡的-p参数,Keepalived配置里的persistence_timeout 50参数都类似这个Nginx里的ip_hash参数,其功能都可以解决动态网页的session共享问题。

    当负载调度算法为ip_hash时,后端服务器在负载均衡调度中的状态不能有weight和backup,即使有也不会生效。

     
    1. upstream yunjisuan_lb{
    2. ip_hash;
    3. server 192.168.0.223:80;
    4. server 192.168.0.224:8080;
    5. }
    6. upstream backend{
    7. ip_hash;
    8. server backend1.example.com;
    9. server backend2.example.com;
    10. server backend3.example.com down;
    11. server backend4.example.com;
    12. }

    (4)fair(动态调度算法)

    此算法会根据后端节点服务器的响应时间来分配请求,响应时间短的优先分配。这是更加智能的调度算法。此种算法可以根据页面大小和加载时间长短智能地进行负载均衡,也就是根据后端服务器的响应时间来分配请求,响应时间短的优先分配。Nginx本身不支持fair调度算法,如果需要使用这种调度算法,必须下载Nginx相关模块upstream_fair。

     
    1. upstream yunjisuan_lb{
    2. server 192.168.0.223;
    3. server 192.168.0.224;
    4. fair;
    5. }

    (5)least_conn(最小连接)

    least_conn算法会根据后端节点的连接数来决定分配情况,哪个机器连接数少就分发。 
    除了上面介绍的这些算法外,还有一些第三方调度算法,例如:url_hash,一致性hash算法等,介绍如下。

    (6)url_hash算法(web缓存节点)

    与ip_hash类似,这里是根据访问URL的hash结果来分配请求的,让每个URL定向到同一个后端服务器,后端服务器为缓存服务器时效果显著。在upstream中加入hash语句,server语句中不能写入weight等其他的参数,hash_method使用的是hash算法。

    url_hash按访问URL的hash结果来分配请求,使每个URL定向到同一个后端服务器,可以进一步提高后端缓存服务器的效率命令率。Nginx本身是不支持url_hash的,如果需要使用这种调度算法,必须安装Nginx的hash模块软件包。

     
    1. upstream yunjisuan_lb {
    2. server squid1:3128;
    3. server squid2:3128;
    4. hash $request_uri;
    5. hash_method crc32;
    6. }

    (7)一致性hash算法(Nginx没有)

    一致性hash算法一般用于代理后端业务为缓存服务(如Squid,Memcached)的场景,通过将用户请求的URI或者指定字符串进行计算,然后调度到后端的服务器上,此后任何用户查找同一个URI或者指定字符串都会被调度到这一台服务器上,因此后端的每个节点缓存的内容都是不同的,一致性hash算法可以解决后端某个或几个节点宕机后,缓存的数据动荡最小,一致性hash算法知识比较复杂,详细内容可以参考百度上的相关资料,这里仅仅给出配置示例

     
    1. http {
    2. upstream test {
    3. consistent_hash $request_uri;
    4. server 127.0.0.1:9001 id=1001 weight=3;
    5. server 127.0.0.1:9002 id=1002 weight=10;
    6. server 127.0.0.1:9003 id=1003 weight=20;
    7. }
    8. }

    虽然Nginx本身不支持一致性hash算法,但Nginx得分支Tengine支持。详细可参考http://tengine.taobao.org/document_cn/http_upstream_consistent_hash_cn.html

     

    http_proxy_module模块

     

    proxy_pass指令介绍

    proxy_pass指令属于ngx_http_proxy_module模块,此模块可以将请求转发到另一台服务器,在实际的反向代理工作中,会通过location功能匹配指定的URI,然后把接收到的符合匹配URI的请求通过proxy_pass抛给定义好的upstream节点池。该指令官方地址1见:http://nginx.org/en/docs/http/ngx_http_proxy_module.html#proxy_pass

     

    proxy_pass的使用案例

    (1)将匹配URI为name的请求抛给http://127.0.0.1/remote/

     
    1. location /name/ {
    2. proxy_pass http://127.0.0.1/remote/;
    3. }

    (2)将匹配URI为some/path的请求抛给http://127.0.0.1

     
    1. location /some/path/ {
    2. proxy_pass http://127.0.0.1;
    3. }

    (3)将匹配URI为name的请求应用指定的rewrite规则,然后抛给http://127.0.0.1

     
    1. location /name/ {
    2. rewrite /name/( [^/]+ ) /username=$1 break;
    3. proxy_pass http://127.0.0.1;
    4. }
     

    http proxy模块参数详解

    Nginx的代理功能是通过http proxy模块来实现的。默认在安装Nginx时已经安装了http proxy模块,因此可直接使用http proxy模块。下面详细解释模块1中每个选项代表的含义,见下表

    image_1crrhm9b11fmi1oe2gerr1mlah4n.png-368kB

     

    反向代理多虚拟主机节点服务器测试

    从测试结果可以看出,已经实现了反向代理,负载均衡功能,但是有一个特殊问题,出来的结果并不是带有sl的字符串,而是bbs的字符串,根据访问结果,我们推测是访问了Web节点下bbs的虚拟主机,明明代理的是sl虚拟主机,为什么结果是访问了后端的bbs虚拟主机

    image_1crrnbfkn530581163m4n8g9.png-19.9kB

    原因是当用户访问域名时确实是携带了sl.yunjisuan.com主机头请求Nginx反向代理服务器,但是反向代理向下面节点重新发起请求时,默认并没有在请求头里告诉节点服务器要找哪台虚拟主机,所以,Web节点服务器接收到请求后发现没有主机头信息,因此,就把节点服务器的第一个虚拟主机发给了反向代理了(节点上第一个虚拟主机放置的是故意这样放置的bbs)。解决这个问题的方法,就是当反向代理向后重新发起请求时,要携带主机头信息,以明确告诉节点服务器要找哪个虚拟主机。具体的配置很简单,就是在Nginx代理sl服务虚拟主机配置里增加如下一行配置即可:

    proxy_set_header host $host;

     
    1. worker_processes 1;
    2. events {
    3. worker_connections 1024;
    4. }
    5. http {
    6. include mime.types;
    7. default_type application/octet-stream;
    8. sendfile on;
    9. keepalive_timeout 65;
    10. upstream www_server {
    11. server 192.168.200.68:80 weight=1;
    12. server 192.168.200.69:80 weight=1;
    13. }
    14. server {
    15. listen 80;
    16. server_name sl.yunjisuan.com;
    17. location / {
    18. proxy_pass http://www_server;
    19. proxy_set_header host $host;
    20. }
    21. }
    22. }

    image_1crrnom2k1c6t1kp9q5krrf17ne16.png-26.6kB

     

    反向代理后的节点服务器记录用户IP测试

    测试会发现,节点服务器对应的sl虚拟主机的访问日志的第一个字段记录的并不是客户端的IP,而是反向代理服务器的IP,最后一个字段也是“-”!

    image_1crro073pbiq1j6opvtmru9pd1j.png-31.5kB

    解决办法同样是在反向代理服务器增加如下一行参数:

    proxy_set_header X-Forwarded-For $remote_addr;

    这是反向代理时,节点服务器获取用户真实IP的必要功能配置

    在反向代理请求后端节点服务器的请求头中增加获取的客户端IP的字段信息,然后节点后端可以通过程序或者相关的配置接收X-Forwarded-For传过来的用户真实IP的信息。

     
    1. worker_processes 1;
    2. events {
    3. worker_connections 1024;
    4. }
    5. http {
    6. include mime.types;
    7. default_type application/octet-stream;
    8. sendfile on;
    9. keepalive_timeout 65;
    10. upstream www_server {
    11. server 192.168.200.68:80 weight=1;
    12. server 192.168.200.69:80 weight=1;
    13. }
    14. server {
    15. listen 80;
    16. server_name sl.yunjisuan.com;
    17. location / {
    18. proxy_pass http://www_server;
    19. proxy_set_header host $host;
    20. proxy_set_header X-Forwarded-For $remote_addr;
    21. }
    22. }
    23. }

    image_1crrp7d581kob2m3auqtl1kdc40.png-24.5kB

    特别注意,虽然反向代理已经配好了,但是节点服务器(WEBA和B)需要的访问日志如果要记录用户的真实IP,还必须进行日志格式配置,这样才能把代理传过来的X-Forwarded-For头信息记录下来,具体配置为

    WEBA和WEBA配置文件是一样的只是IP不同

     
    1. worker_processes 1;
    2. events {
    3. worker_connections 1024;
    4. }
    5. http {
    6. include mime.types;
    7. default_type application/octet-stream;
    8. sendfile on;
    9. keepalive_timeout 65;
    10. server_tokens off;
    11. log_format main '$remote_addr-$remote_user[$time_local]"$request"'
    12. '$status $body_bytes_sent "$http_referer"'
    13. '"$http_user_agent""$http_x_forwarded_for"';
    14. server {
    15. listen 80;
    16. server_name bbs.yunjisuan.com;
    17. location / {
    18. root html/bbs;
    19. index index.html index.htm;
    20. }
    21. access_log logs/access_bbs.log main;
    22. }
    23. server {
    24. listen 80;
    25. server_name sl.yunjisuan.com;
    26. location / {
    27. root html/sl;
    28. index index.html index.htm;
    29. }
    30. access_log logs/access_sl.log main;
    31. }
    32. }
    33. 配置文件详解
    34. log_format --->记录字段顺序
    35. $remote_addr --->来源IP
    36. $remote_user --->来源用户
    37. [$time_local] --->来源时间
    38. $request --->请求
    39. $status --->状态码
    40. $body_bytes_sent --->主体,发送的大小
    41. $http_referer --->来源的浏览器
    42. $http_user_agent --->用户的客户端
    43. $http_x_forwarded_for --->这是反向代理时,节点服务器获取用户真实IP的必要功能配置
    44. 如果希望在第一行显示,可以替换掉第一行的$remote_addr变量

    image_1crrq5nj9iph1ue21e3o17lj170n4d.png-25.5kB

     

    清空NginxWEBA和B日志文件并在浏览器输入域名测试

     
    1. cd /usr/local/nginx/logs
    2. > access_bbs.log
    3. > access_sl.log
    4. 没在WINDOWS做映射的话先去做一下
    5. 因为虚拟机是NAT模式,所以真实IP是网关IP

    image_1crrrjrcptlm1nr2bphc7h1uk9ig.png-11.1kB

    image_1crrqfg1rr4s10r2193b30v1n4d4q.png-63.3kB

     

    反向代理相关重要基础参数的总结

    image_1crrqlv1gqk5mbsu4m1r8h1qn757.png-202.4kB

     

    配置文件用include方式包含到虚拟主机配置里

    cd /usr/local/nginx/conf

    touch proxy.conf

    把下面的重要参数放到proxy.conf在反向代理配置文件用include方式调用

     
    1. proxy_set_header host $host;
    2. proxy_set_header x-forwarded-for $remote_addr;
    3. proxy_connect_timeout 60;
    4. proxy_send_timeout 60;
    5. proxy_read_timeout 60;
    6. proxy_buffer_size 4k;
    7. proxy_buffers 4 32k;
    8. proxy_busy_buffers_size 64k;
    9. proxy_temp_file_write_size 64k;
     
    1. worker_processes 1;
    2. events {
    3. worker_connections 1024;
    4. }
    5. http {
    6. include mime.types;
    7. default_type application/octet-stream;
    8. sendfile on;
    9. keepalive_timeout 65;
    10. upstream www_server {
    11. server 192.168.200.68:80 weight=1;
    12. server 192.168.200.69:80 weight=1;
    13. }
    14. server {
    15. listen 80;
    16. server_name sl.yunjisuan.com;
    17. location / {
    18. proxy_pass http://www_server;
    19. include proxy.conf;
    20. }
    21. }
    22. }

    image_1crrr9ve81cpc4lt1r401qkl1iol5k.png-49kB

     

    include方式在windows浏览器测试

    image_1crrrhevo1r1sokc9sj155krl9e5.png-13kB

    image_1crrrgchr16g01dem5m0u461ei4b5.png-61.2kB

     

    根据URL中的目录地址实现代理转发

    通过Nginx实现动静分离,即通过Nginx反向代理配置规则实现让动态资源和静态资源及其他业务分别由不同的服务器解析,以解决网站性能,安全,用户体验等重要问题。

    下图为企业常见的动静分离集群架构图,此架构图适合网站前端只使用同一个域名提供服务的场景,例如,用户访问的域名是www.yunjisuan.com,然后,当用户请求www.yunjisuan.com/upload/xx地址时候,代理会分配请求到上传服务器池处理数据;当用户请求www.yunjisuan.com/static/xx地址的时候,代理会分配请求到静态服务器池请求数据;当用户请求www.yunjisuan.com/xx地址的时候,即不包含上述指定的目录地址路径时,代理会分配请求到默认的动态服务器池请求数据(注意:上面的xx表示任意路径)。

    image_1crrrvrm0ilb1v2mivbr6k1jvqk9.png-197.1kB

     

    根据URL方式测试反向代理服务器配置文件如下

    WEBA请求sl.yunjisuan.com/static/xx地址时,实现由静态服务器池处理请求。(192.168.200.68)

    WEBB请求sl.yunjisuan.com/upload/xx地址时,实现由upload上传服务器池处理请求。(192.168.200.69)

    WEBC其他访问请求,全都由默认的动态服务器池处理请求。(192.168.200.75)

    include方式调用的文件

     

    以location方案实现

     
    1. worker_processes 1;
    2. events {
    3. worker_connections 1024;
    4. }
    5. http {
    6. include mime.types;
    7. default_type application/octet-stream;
    8. sendfile on;
    9. keepalive_timeout 65;
    10. upstream default {
    11. server 192.168.200.75:80 weight=1;
    12. }
    13. upstream static {
    14. server 192.168.200.68:80 weight=1;
    15. }
    16. upstream upload {
    17. server 192.168.200.69:80 weight=1;
    18. }
    19. server {
    20. listen 80;
    21. server_name sl.yunjisuan.com;
    22. location / {
    23. proxy_pass http://default;
    24. include proxy.conf;
    25. }
    26. location /static/ {
    27. proxy_pass http://static;
    28. include proxy.conf;
    29. }
    30. location /upload/ {
    31. proxy_pass http://upload;
    32. include proxy.conf;
    33. }
    34. }
    35. }
     

    以if语句实现

     
    1. worker_processes 1;
    2. events {
    3. worker_connections 1024;
    4. }
    5. http {
    6. include mime.types;
    7. default_type application/octet-stream;
    8. sendfile on;
    9. keepalive_timeout 65;
    10. upstream default {
    11. server 192.168.200.75:80 weight=1;
    12. }
    13. upstream static {
    14. server 192.168.200.68:80 weight=1;
    15. }
    16. upstream upload {
    17. server 192.168.200.69:80 weight=1;
    18. }
    19. server {
    20. listen 80;
    21. server_name sl.yunjisuan.com;
    22. location / {
    23. proxy_pass http://default;
    24. include proxy.conf;
    25. if ($request_uri ~* "^/static/(.*)$"){
    26. proxy_pass http://static/$1;
    27. }
    28. if ($request_uri ~* "^/upload/(.*)$"){
    29. proxy_pass http://upload/$1;
    30. }
    31. }
    32. }
    33. }

    image_1crrvudmd1cqo9v41stst1e2a01g.png-50.9kB

     

    WEBA静态测试结果(curl方式)

    cd /usr/local/nginx/html

    cd sl

    mkdir static

    echo "static" > static/index.html

    curl http://sl.yunjisuan.com/static/

    image_1crs07hf3s9n2mo1lvl4lf74s1t.png-39.3kB

     

    WEBB上传测试结果(curl方式)

    cd /usr/local/nginx/html

    cd sl

    mkdir upload

    echo "upload" > upload/index.html

    curl http://sl.yunjisuan.com/upload/

    image_1crs0gcdf11dg1r9qdh112bhiev2a.png-40.2kB

     

    WEBC默认测试结果(curl方式)

    cd /usr/local/nginx/html

    echo "default" > default/index.html

    curl http://sl.yunjisuan.com

    image_1crs16hjl1n83ktp1j2b19tfpek34.png-35.2kB

     

    windows所有网页测试结果

    http://sl.yunjisuan.com

    http://sl.yunjisuan.com/static/index.html

    http://sl.yunjisuan.com/upload/index.html


    image_1crs1d37812ngvpu3abmpr17rg4b.png-11kB


    image_1crs1djt81aggvqmqdtg1u1atq4o.png-14.1kB


    image_1crs1eam71t7i1ombvoa1lou32555.png-13.4kB


     

     

  • 相关阅读:
    Linux的内存管理之道
    Ramdisk根文件系统启动Linux(单独的ramdisk镜像)
    详解 ARM Linux启动过程
    晒晒我的vimrc配置【不定期更新】
    [Python入门及进阶笔记]Pythondecorator装饰器小结
    谷歌10个用户体验设计原则
    用busybox创建基于Linux2.6.24内核的nfs根文件系统
    [Python入门及进阶笔记]Python魔术方法小结(方法运算符重载)
    mysql慢查询
    关于android的各种disk images(看过的讲android image比较细致的好文)
  • 原文地址:https://www.cnblogs.com/linyaonie/p/9937578.html
Copyright © 2020-2023  润新知