nginx sticky实现基于cookie负载均衡示例详解


Posted in Servers onDecember 24, 2022

前言

 

sticky 是一个nginx的第三方模块 它不在nginx发行版中 需要额外编译这个模块的, 它的思想就是不依靠后端生成cookie , 而是sticky在nginx这里生成cookie ,然后下发到客户端, 客户端收到cookie后 以后的请求带着这个cookie 就会通过这个cookie 进行hash 被一直定位到后端的某一台服务器了

优点:

  • 它比纯 ip hash 负载有个优点就是 纯 ip hash 像局域网内的访问ip 访问会导致ip倾斜
  • 它比 hash $cookie_jsessionid的优点就是 它不依赖后端 不用后端生成 session 从而减少后端的 资源

思考

想想为什么要用这个 sticky 来把用户尽量一直定位到一台服务器呢? 在多台后台服务器的环境下,我们为了确保一个客户只和一台服务器通信,我们势必使用长连接。使用什么方式来实现这种连接呢,常见的有使用nginx自带的ip_hash来做,我想这绝对不是一个好的办法,如果前端是CDN,或者说一个局域网的客户同时访问服务器,导致出现服务器分配不均衡,以及不能保证每次访问都粘滞在同一台服务器。

如果基于cookie会是一种什么情形,想想看, 每台电脑都会有不同的cookie,在保持长连接的同时还保证了服务器的压力均衡,nginx sticky值得推荐。

如果浏览器不支持cookie,那么sticky不生效,毕竟整个模块是给予cookie实现的.

1.cookie_jsessionid 负载均衡

在说sticky 之前先来看看 nginx 通过 cookie_jessionid 的负载均衡方式

1.1 后端准备

@Autowired
lateinit var env: Environment
@GetMapping("/server")
fun server(request:HttpServletRequest):String {
  //获取当前服务的端口
  val port = env.getProperty("local.server.port")
  println("now port: $port")
  //调用了request.getSession(true) 则会没有session的时候创建session
  val session = request.getSession(true)
  val name = session.getAttribute("name")
  println("name: $name")
  if (name == null){
    session.setAttribute("name","johnny")
  }
  return "success"
}

1.2 hash $cookie_jsessionid;配置

在upstream 里面配置 hash 的方式 使用 cookie_jsessionid 去做hash

#user  nobody;
worker_processes  1;
#error_log  logs/error.log;
#error_log  logs/error.log  notice;
#error_log  logs/error.log  info;
#pid        logs/nginx.pid;
events {
    worker_connections  1024;
}
http {
    include       mime.types;
    default_type  application/octet-stream;
    sendfile        on;
    #tcp_nopush     on;
    #keepalive_timeout  0;
    keepalive_timeout  65;
    #gzip  on;
    upstream backend {
        # 指定hash 方式是 cookie_jessionid nginx自带的方式
        hash $cookie_jsessionid;
        server 172.16.225.1:8081;
			  server 172.16.225.1:8080;
    }
    server {
        listen       80;
        server_name  localhost;
        #charset koi8-r;
        #access_log  logs/host.access.log  main;
        location / {
            # 指定负载到后端upstream
	    			proxy_pass http://backend;  
        }
        error_page   500 502 503 504  /50x.html;
        location = /50x.html {
            root   html;
        }
    }
}

可以看到 服务器下发了 cookie JSESSIONID 并且多次请求这个 都不会改变 因为nginx 根据 JSESSIONID 它进行hash 每次都负载到同一台后端服务器, 因为这个后端服务器已经存在了 这个session 所以不会再次创建

nginx sticky实现基于cookie负载均衡示例详解

可以看到 多次请求 都打到这个 8081 的后端服务了

nginx sticky实现基于cookie负载均衡示例详解

2.nginx sticky 负载均衡

2.1 下载 sticky

Bitbucket

bitbucket.org/nginx-goodi…

nginx sticky实现基于cookie负载均衡示例详解

2.2 重新编译升级nginx

1)下载完成,放入服务器解压,记住解压的位置,后面要用

2)进入到nginx的安装文件

3)配置nginx

tar -xvf nginx-goodies-nginx-sticky-module-ng-c78b7dd79d0d.tar.gz
mv nginx-goodies-nginx-sticky-module-ng-c78b7dd79d0d nginx-sticky
# 添加sticky 模块
./configure \
--prefix=/usr/local/nginx \
--add-module=/opt/nginx-sticky

make 编译的时候有可能会报错

找到sticky刚刚的解压目录,进入修改文件 vim ngx_http_sticky_misc.c ,加入下面的头文件

#include <openssl/sha.h> 
#include <openssl/md5.h>

nginx sticky实现基于cookie负载均衡示例详解

再次make , 当然后面如果还报错的话,openssl检查是否安装

apt-get install -y openssl

2.3 upstream 配置 sticky

配置好后重启nginx

upstream backend {
        #hash $cookie_jsessionid;
        sticky; #指定使用 sticky 进行负载均衡
        server 172.16.225.1:8081;
				server 172.16.225.1:8080;
 }

nginx sticky实现基于cookie负载均衡示例详解

2.4 修改后端不再创建session

此时后端不会创建session 也不会下发cookie jsessionid 了

@Autowired
lateinit var env: Environment
@GetMapping("/server")
fun server(request:HttpServletRequest):String {
    val port = env.getProperty("local.server.port")
    println("now port: $port")
    return "success"
}

2.5 再次 多次请求

可以看到stick 帮我们下发了 route 这个cookie , 并且这个不会变 默认关闭浏览器就会失效

nginx sticky实现基于cookie负载均衡示例详解

可以看到请求还是只会落在一台服务器上

nginx sticky实现基于cookie负载均衡示例详解

3.sticky 其他用法

sticky [name=route] [domain=.foo.bar] [path=/] [expires=1h] [hash=index|md5|sha1] [no_fallback];

name: 可以为任何的string字符,默认是route

domain:哪些域名下可以使用这个cookie

path:哪些路径对启用sticky,例如path/test,那么只有test这个目录才会使用sticky做负载均衡

expires:cookie过期时间,默认浏览器关闭就过期,也就是会话方式。

no_fallbackup:如果设置了这个,cookie对应的服务器宕机了,那么将会返回502(bad gateway 或者 proxy error),建议不启用

总结

本篇主要介绍了 nginx sticky 负载均衡,它不需要后端去生成session 下发jsessionid 而是nginx的sticky模块帮我们去下发一个 route 的 cookie , nginx 使用这个cookie 进行hash 负载, 从而实现了 客户每次访问都粘滞在同一台服务器

以上就是nginx sticky实现基于cookie负载均衡示例详解的详细内容,更多关于nginx sticky cookie负载均衡的资料请关注三水点靠木其它相关文章!

Servers 相关文章推荐
Nginx 根据URL带的参数转发的实现
Apr 01 Servers
详解Apache SkyWalking 告警配置指南
Apr 22 Servers
Nginx性能优化之Gzip压缩设置详解(最大程度提高页面打开速度)
Feb 12 Servers
Nginx的基本概念和原理
Mar 21 Servers
Nginx设置HTTPS的方法步骤 443证书配置方法
Mar 21 Servers
Docker下安装Oracle19c
Apr 13 Servers
IIS服务器中设置HTTP重定向访问HTTPS
Apr 29 Servers
阿里云服务器Ubuntu 20.04上安装Odoo 15
May 20 Servers
winserver2019安装软件一直卡在应用程序正在为首次使用做准备
Jun 10 Servers
Windows Server 2008配置防火墙策略详解
Jun 28 Servers
Nginx报404错误的详细解决方法
Jul 23 Servers
ubuntu端向日葵键盘输入卡顿问题及解决
Dec 24 Servers
解决ubuntu安装软件时,status-code=409报错的问题
Dec 24 #Servers
ubuntu开机后ROS程序自启动问题
Dec 24 #Servers
ubuntu端向日葵键盘输入卡顿问题及解决
Dec 24 #Servers
码云(gitee)通过git自动同步到阿里云服务器
Dec 24 #Servers
Valheim服务器 Mod修改安装教程 【ValheimPlus】
Dec 24 #Servers
keepalived + nginx 实现高可用方案
Dec 24 #Servers
ubuntu20.04虚拟机无法上网的问题及解决
Dec 24 #Servers
You might like
php不用正则验证真假身份证
2013/11/06 PHP
PHP输出缓冲控制Output Control系列函数详解
2015/07/02 PHP
php使用FFmpeg接口获取视频的播放时长、码率、缩略图以及创建时间
2016/11/07 PHP
php生成毫秒时间戳的实例讲解
2017/09/22 PHP
阿里云Win2016安装Apache和PHP环境图文教程
2018/03/11 PHP
php解决约瑟夫环算法实例分析
2019/09/30 PHP
PHP中迭代器的简单实现及Yii框架中的迭代器实现方法示例
2020/04/26 PHP
js 全兼容可高亮二级缓冲折叠菜单
2010/06/04 Javascript
缓动函数requestAnimationFrame 更好的实现浏览器经动画
2012/12/07 Javascript
js去字符串前后空格5种实现方法及比较
2013/04/03 Javascript
jQuery :first选择器使用介绍
2013/08/09 Javascript
JS对话框_JS模态对话框showModalDialog用法总结
2014/01/11 Javascript
jQuery中siblings()方法用法实例
2015/01/08 Javascript
jquery在ie7下选择器的问题导致append失效的解决方法
2016/01/10 Javascript
JavaScript中日期函数的相关操作知识
2016/08/03 Javascript
jQuery通过ajax方法获取json数据不执行success的原因及解决方法
2016/10/15 Javascript
从零开始最小实现react服务器渲染详解
2018/01/26 Javascript
Vue商品控件与购物车联动效果的实例代码
2019/07/21 Javascript
[01:02:47]EG vs Secret 2019国际邀请赛淘汰赛 胜者组 BO3 第一场 8.21.mp4
2020/07/19 DOTA
python使用ctypes模块调用windowsapi获取系统版本示例
2014/04/17 Python
Python 遍历子文件和所有子文件夹的代码实例
2016/12/21 Python
python虚拟环境virtualenv的安装与使用
2017/09/21 Python
使用Python进行AES加密和解密的示例代码
2018/02/02 Python
python 通过SSHTunnelForwarder隧道连接redis的方法
2019/02/19 Python
学习python分支结构
2019/05/17 Python
使用Windows批处理和WMI设置Python的环境变量方法
2019/08/14 Python
python os.path.isfile 的使用误区详解
2019/11/29 Python
Python操作Elasticsearch处理timeout超时
2020/07/17 Python
html5 input输入实时检测以及延时优化
2018/07/18 HTML / CSS
澳大利亚领先的在线礼品网站:Gifts Australia
2020/08/15 全球购物
一道Delphi面试题
2016/10/28 面试题
《天安门广场》教学反思
2014/04/23 职场文书
怎么写工作检讨书
2014/11/16 职场文书
2015大学生暑期实习报告
2015/07/13 职场文书
幼儿园开学报名通知
2015/07/16 职场文书
在CSS中映射鼠标位置并实现通过鼠标移动控制页面元素效果(实例代码)
2021/04/22 HTML / CSS