TCP 百万并发 数据连接测试 python+locust
过程笔记和总结
尝试一、locust 测试百万Tcp并发
另一种方式是使用jmeter
基础环境
- 服务端
虚拟机:Centos7.2
jdk 1.8
- 客户端
虚拟机: Centos7.2
python : 3.7.3Anaconda3
locust : 0.14.5
基础知识:
- tcp协议:三次握手进行连接,四次挥手断开,稳定长连接,比Http更占用资源,比udp协议更稳定,保证数据不丢失,但速度比较慢。
- 每个tcp连接大概占用4kb内存,且断开连接后默认两分钟之后才会释放资源
- linux打开文件限制(open-files)
- linux端口限制
- tcp-socket 网络编程
- python中类继承 super 用法
- python中发送16进制数据包
Q&A
客户端调优
- 客户端端口限制导致的tcp连接不超过4000,通过修改打开文件限制来增加可使用的端口数量,注意修改后需要重启生效,具体配置参考
-- /etc/sysctl.conf
# 系统级别最大打开文件
fs.file-max = 100000
# 单用户进程最大文件打开数
fs.nr_open = 100000
# 是否重用, 快速回收time-wait状态的tcp连接
net.ipv4.tcp_tw_reuse = 1
net.ipv4.tcp_tw_recycle = 1
# 单个tcp连接最大缓存byte单位
net.core.optmem_max = 8192
# 可处理最多孤儿socket数量,超过则警告,每个孤儿socket占用64KB空间
net.ipv4.tcp_max_orphans = 10240
# 最多允许time-wait数量
net.ipv4.tcp_max_tw_buckets = 10240
# 从客户端发起的端口范围,默认是32768 61000,则只能发起2w多连接,改为一下值,可一个IP可发起差不多6.4w连接。
net.ipv4.ip_local_port_range = 1024 65535
-- /etc/security/limits.conf
# 最大不能超过fs.nr_open值, 分别为单用户进程最大文件打开数,soft指软性限制,hard指硬性限制
* soft nofile 100000
* hard nofile 100000
root soft nofile 100000
root hard nofile 100000
服务端调优
2. 服务端需要接受的链接数需要超过100w,根据需要修改虚拟机配置;可参考文章https://www.jianshu.com/p/490e2981545c
- 优化局部句柄限制
查看 ulimit-n
soft和hard为两种限制方式,其中soft表示警告的限制,hard表示真正限制,nofile表示打开的最大文件数。整体表示任何用户一个进程能够打开1000000个文件。注意语句签名有 号 表示任何用户
-- /etc/security/limits.conf
# 文末加两行
*hard nofile 1000000
soft nofile 1000000
shutdown -r now 重启并查看生效
- 突破全局句柄的限制
cat /proc/sys/fs/file-max
file-max 表示在linux 中最终所有x线程能够打开的最大文件数
sudo vi /etc/sysctl.conf
# 在文末增加
fs.file-max=1000000
让文件生效
sudo sysctl -p
Locust分布式说明
因单台node最大tcp连接6w,故测试百万连接需要对Locust做分布式处理;
Locust分布式注意点:
每台机器上都要有client文件,且必须相同
slave虚拟机环境、Locust版本、python版本必须相同,否则预期出现未知错误
slave节点较多时需要并行管理工具pdsh的使用吗,同时启动多个slave节点,减少人工启停时间成本
遇到的问题
locust 分布式执行问题
单机启动可发送60000,两台120000
集群启动发送60000 服务端ip无法超过60000 限制
已解决:
locust master 节点不作为负载机进行发送数据,slave 一个节点最多可以增加6w个长连接(linux共有端口65535, 一般1-1023 端口是系统保留的,1024-65535 是用户使用的,可用端口有6w多个)
运维管理工具 pdsh
作用:快速分发命令,启动slave节点
1、 配置ssh信任
修改主机名称 vim /etc/hostname
master 可设为node0,slave 可设为 node1~16
参考:https://blog.csdn.net/ctypyb2002/article/details/80572181
2、 安装pdsh
参考:https://www.cnblogs.com/liwanliangblog/p/9194146.html
tar -jxvf **.tag.bz2
./configure --with-ssh --with-rsh --with-mrsh--with-mqshell --with-qshell --with-dshgroups--with-machines=/etc/pdsh/machines --without-pam
make
make install
客户端代码
## locustfile.py
# coding:utf-8
import time
import random
# from socket import socket, AF_INET, SOCK_STREAM
import socket
from locust import Locust, TaskSet, events, task
class TcpSocketClient(socket.socket):
# locust tcp client
# author: ShenDu
def __init__(self, af_inet, socket_type):
super(TcpSocketClient, self).__init__(af_inet, socket_type)
def connect(self, addr):
start_time = time.time()
try:
super(TcpSocketClient, self).connect(addr)
except Exception as e:
total_time = int((time.time() - start_time) * 1000)
events.request_failure.fire(request_type="tcpsocket", name="connect", response_time=total_time, exception=e)
else:
total_time = int((time.time() - start_time) * 1000)
events.request_success.fire(request_type="tcpsocket", name="connect", response_time=total_time,
response_length=0)
def send(self, msg):
start_time = time.time()
try:
super(TcpSocketClient, self).send(msg)
except Exception as e:
total_time = int((time.time() - start_time) * 1000)
events.request_failure.fire(request_type="tcpsocket", name="send", response_time=total_time, exception=e)
else:
total_time = int((time.time() - start_time) * 1000)
events.request_success.fire(request_type="tcpsocket", name="send", response_time=total_time,
response_length=0)
def recv(self, bufsize):
recv_data = ''
start_time = time.time()
try:
recv_data = super(TcpSocketClient, self).recv(bufsize)
except Exception as e:
total_time = int((time.time() - start_time) * 1000)
events.request_failure.fire(request_type="tcpsocket", name="recv", response_time=total_time, exception=e)
else:
total_time = int((time.time() - start_time) * 1000)
events.request_success.fire(request_type="tcpsocket", name="recv", response_time=total_time,
response_length=0)
return recv_data
class TcpSocketLocust(Locust):
"""
This is the abstract Locust class which should be subclassed. It provides an TCP socket client
that can be used to make TCP socket requests that will be tracked in Locust's statistics.
"""
def __init__(self, *args, **kwargs):
super(TcpSocketLocust, self).__init__(*args, **kwargs)
self.client = TcpSocketClient(socket.AF_INET, socket.SOCK_STREAM)
ADDR = (self.host, self.port)
self.client.connect(ADDR)
class TcpTestUser(TcpSocketLocust):
host = "192.168.5.58"
port = 6667
min_wait = 100
max_wait = 1000
class task_set(TaskSet):
@task
def send_data(self):
data = "7e0200003f000004021895000b00000000000400030158ccaa06cb7" \
"9f50095000000001601051654150104000069740202000003020000" \
"2504000000002b040000000030010031010b3201467c7e"
# self.client.send(data.encode("utf-8"))
self.client.send(bytes.fromhex(data))
data = self.client.recv(2048).decode("utf-8")
print(data)
if __name__ == "__main__":
import os
os.system("locust -f sendTcp.py")
单节点启动
locust -f locustfile.py
分布式启动
# master执行
locust -f locustfile.py --master
# slave执行
locust -f locustfile.py --slave --master-host = 127.0.0.1
# pdsh 同时启动slave节点, node 可使用ip代替
pdsh -w node[1-16] "locust -f /root/loadtest/locustfile.py --slave --master-host=node0"
python编写的简易服务端
调试程序使用的服务端代码
#coding: utf-8
from __future__ import print_function
from gevent.server import StreamServer
import gevent
# sleeptime = 60
'''
python 编写的简单tcp socket server,作为服务端测试使用
'''
def handle(socket, address):
# print(address)
# data = socket.recv(1024)
# print(data)
while True:
gevent.sleep(sleeptime)
try:
socket.send("ok")
except Exception as e:
print(e)
if __name__ == "__main__":
import sys
port = 80
if len(sys.argv) > 2:
port = int(sys.argv[1])
sleeptime = int(sys.argv[2])
else:
print("需要两个参数!!")
sys.exit(1)
# default backlog is 256
server = StreamServer(('0.0.0.0', port), handle, backlog=4096)
server.serve_forever()
参考文章:
- 百万并发:
https://blog.csdn.net/jackliu16/article/details/81294741
- bytes和hex字符串之间的转换:
https://blog.csdn.net/wowocpp/article/details/79701739
- 性能测试 - Locust TCP socket client:
https://blog.csdn.net/max229max/article/details/79277295
- 深入浅出Locust:
https://juejin.im/post/58c29457128fe10060297328
- 官文:
https://docs.locust.io/en/latest/quickstart.html
- locust&jmeter比较:
https://www.jianshu.com/p/dd0fcfdfa561
TCP 百万并发 数据连接测试 python+locust的更多相关文章
- 基于管道通知的百万并发长连接server模型
0.前言 最近突然想了解怎样设计一个支持百万连接的后台server架构. 要设计一个支持百万连接的后台server,我们首先要知道会有哪些因素限制后台server的高并发连接,这里想到的因素有以下几点 ...
- MarioTCP:一个单机可日30亿的百万并发长连接服务器
原文:http://blog.csdn.net/everlastinging/article/details/10894493 注:如果用此服务器做变长data的传输,请在业务处理函数中为input ...
- 如何在mysql数据库生成百万条数据来测试页面加载速度
1.首先复制一条sql 在复制前,需要确定该记录是否有主键 若无,则代码非常简单, "; 复制的表名↑ 粘贴的表名↑ ...
- 测试开发【提测平台】分享9-DBUntils优化数据连接&实现应用搜索和分页功能
微信搜索[大奇测试开],关注这个坚持分享测试开发干货的家伙. 从本期开始知识点讲以思维导图的形式给出,内容点会按照讲解-应用-展示的形式体现,这样会更清晰些. DBUntils连接池 在项目中链接数据 ...
- 面向Web应用的并发压力测试工具——Locust实用攻略
1. 概述 该方案写作目的在于描述一个基于Locust实现的压力测试,文中详细地描述了如何利用locustfile.py文件定义期望达成的测试用例,并利用Locust对目标站点进行并发压力测试. 特别 ...
- python全栈开发day113-DBUtils(pymysql数据连接池)、Request管理上下文分析
1.DBUtils(pymysql数据连接池) import pymysql from DBUtils.PooledDB import PooledDB POOL = PooledDB( creato ...
- Linux Mysql 安装 开启远程连接 供python agent 连接测试 Mark
Linux 6.3 (1) cat /etc/redhat-release uname -a 查看yum 源: 阿里源 无源运行: echo 下载阿里云的yum源配置 wget -O / ...
- 4-STM32物联网开发WIFI(ESP8266)+GPRS(Air202)系统方案数据篇(云端电脑(Windows)安装配置数据库,使用本地Navicat for MySQL和手机APP 远程连接测试)
3-STM32物联网开发WIFI(ESP8266)+GPRS(Air202)系统方案数据篇(安装配置数据库,使用Navicat for MySQL和手机APP 连接测试) 根据前面的教程把软件复制到云 ...
- 3-STM32物联网开发WIFI(ESP8266)+GPRS(Air202)系统方案数据篇(安装配置数据库,使用Navicat for MySQL和手机APP 连接测试)
2-STM32物联网开发WIFI(ESP8266)+GPRS(Air202)系统方案数据篇(数据库简单说明) https://www.mysql.com/ 咱用安装版的 我把自己下载的放在了这里 现在 ...
随机推荐
- [Kafka][1][初识Kafka]
目录 第1章 初识Kafka 1.1 发布与订阅消息系统 1.1.1 如何开始 1.1.2 独立的队列系统 1.2 Kafka登场 1.2.1 消息和批次(Message and batch) 1.2 ...
- Camtasia绿幕素材的视频合成
随着科技和互联网的快速发展,让越来越多的人喜欢上了视频的各项制作,那么怎么让两个视频进行合成并一起播放呢?操作很简单,下面来讲解具体的操作步骤.小编选用的是Camtasia2019版本的视频编辑软件进 ...
- 精尽MyBatis源码分析 - SQL执行过程(三)之 ResultSetHandler
该系列文档是本人在学习 Mybatis 的源码过程中总结下来的,可能对读者不太友好,请结合我的源码注释(Mybatis源码分析 GitHub 地址.Mybatis-Spring 源码分析 GitHub ...
- MyBatis的一二级缓存
一级缓存 一级缓存默认是开启的,生命周期和SqlSession相同.一个会话中每次执行一个查询操作时,会先查询二级缓存,如果二级缓存没查到或者二级缓存未开启就会从一级缓存中查询,如果一级缓存也未查到就 ...
- 添加mysqli扩展
find / -name phpize 进入mysqli目录下 ./configue -prefix=/usr/local/mysqli . make && make install ...
- 系统提供的dispatch方法
h1, h2, h3, h4, h5, h6, p, blockquote { margin: 0; padding: 0 } body { font-family: "Helvetica ...
- mark一下2020已经阅读的书单专栏和源码
1.书籍 已经阅读: 深度工作 代码整洁之道 正在阅读: 敏捷软件开发 程序员修炼之道 程序员思维 计划阅读: 人月神话 数据密集型系统设计 如何阅读一本书 卓有成效的管理者 算法 第四版 sprin ...
- oracle 游标相关资料
游标 概述:游标是系统为用户开设的一个数据缓冲区,存放 SQL 语句的执行结果. 我们可以把游标理解为 PL/SQL 中的结果集,把游标当中一个集合 1:在声明区声明游标 cursor 游标名称 is ...
- 谈谈 rm -rf * 后的几点体会(年轻人得讲码德)
事情始末 平时经常开玩笑,删库跑路.删库跑路,今天我真的rm -rf *了.早上来,一个同事说要查日志,但是日志我又备份到云磁盘了,我就想着把那一天的日志wget下来看看,然后分析.本来是想放在/va ...
- RestTemplate 统一添加 Header
一.添加拦截器 public class HeaderRequestInterceptor implements ClientHttpRequestInterceptor { private fina ...