使用Python的Treq on Twisted来进行HTTP压力测试


Posted in Python onApril 16, 2015

从事API相关的工作很有挑战性,在高峰期保持系统的稳定及健壮性就是其中之一,这也是我们在Mailgun做很多压力测试的原因。

这么久以来,我们已经尝试了很多种方法,从简单的ApacheBench到复杂些的自定义测试套。但是本贴讲述的,是一种使用python进行“快速粗糙”却非常灵活的压力测试的方法。
使用python写HTTP客户端的时候,我们都很喜欢用 Requests library。这也是我们向我们的API用户们推荐的。Requests 很强大,但有一个缺点,它是一个模块化的每线程一个调用的东西,很难或者说不可能用它来快速的产生成千上万级别的请求。
Treq on Twisted简介

为解决这个问题我们引入了Treq (Github库)。Treq是一个HTTP客户端库,受Requests影响,但是它运行在Twisted上,具有Twisted典型的强大能力:处理网络I/O时它是异步且高度并发的方式。

Treq并不仅仅限于压力测试:它是写高并发HTTP客户端的好工具,比如网页抓取。Treq很优雅、易于使用且强大。这是一个例子:

>>> from treq import get
  
 >>> def done(response):
 ...   print response.code
 ...   reactor.stop()
  
 >>> get("http://www.github.com").addCallback(done)
  
 >>> from twisted.internet import reactor
 200

简单的测试脚本
如下是一个使用Treq的简单脚本,用最大可能量的请求来对单一URL进行轰炸。

#!/usr/bin/env python
 from twisted.internet import epollreactor
 epollreactor.install()
  
 from twisted.internet import reactor, task
 from twisted.web.client import HTTPConnectionPool
 import treq
 import random
 from datetime import datetime
  
 req_generated = 0
 req_made = 0
 req_done = 0
  
 cooperator = task.Cooperator()
  
 pool = HTTPConnectionPool(reactor)
  
 def counter():
   '''This function gets called once a second and prints the progress at one
   second intervals.
   '''
   print("Requests: {} generated; {} made; {} done".format(
       req_generated, req_made, req_done))
   # reset the counters and reschedule ourselves
   req_generated = req_made = req_done = 0
   reactor.callLater(1, counter)
  
 def body_received(body):
   global req_done
   req_done += 1
  
 def request_done(response):
   global req_made
   deferred = treq.json_content(response)
   req_made += 1
   deferred.addCallback(body_received)
   deferred.addErrback(lambda x: None) # ignore errors
   return deferred
  
 def request():
   deferred = treq.post('http://api.host/v2/loadtest/messages',
              auth=('api', 'api-key'),
              data={'from': 'Loadtest <test@example.com>',
                 'to': 'to@example.org',
                'subject': "test"},
             pool=pool)
   deferred.addCallback(request_done)
   return deferred
  
 def requests_generator():
   global req_generated
   while True:
     deferred = request()
     req_generated += 1
     # do not yield deferred here so cooperator won't pause until
     # response is received
     yield None
  
 if __name__ == '__main__':
   # make cooperator work on spawning requests
   cooperator.cooperate(requests_generator())
  
   # run the counter that will be reporting sending speed once a second
   reactor.callLater(1, counter)
  
   # run the reactor
   reactor.run()

输出结果:

2013-04-25 09:30 Requests: 327 generated; 153 sent; 153 received
 2013-04-25 09:30 Requests: 306 generated; 156 sent; 156 received
 2013-04-25 09:30 Requests: 318 generated; 184 sent; 154 received

“Generated”类的数字代表被Twisted反应器准备好但是还没有发送的请求。这个脚本为了简洁性忽略了所有错误处理。为它添加超时状态的信息就留给读者作为一个练习。

这个脚本可以当做是一个起始点,你可以通过拓展改进它来自定义特定应用下的处理逻辑。建议你在改进的时候用collections.Counter 来替代丑陋的全局变量。这个脚本运行在单线程上,想通过一台机器压榨出最大量的请求的话,你可以用类似 mulitprocessing 的技术手段。

愿你乐在压力测试!

Python 相关文章推荐
python魔法方法-自定义序列详解
Jul 21 Python
浅谈pyhton学习中出现的各种问题(新手必看)
May 17 Python
Python 实现数据库更新脚本的生成方法
Jul 09 Python
python 随机数使用方法,推导以及字符串,双色球小程序实例
Sep 12 Python
Python爬虫框架Scrapy常用命令总结
Jul 26 Python
python使用scrapy发送post请求的坑
Sep 04 Python
selenium+python自动化测试之使用webdriver操作浏览器的方法
Jan 23 Python
django model通过字典更新数据实例
Apr 01 Python
为什么说python适合写爬虫
Jun 11 Python
python logging 重复写日志问题解决办法详解
Aug 04 Python
python数据分析之用sklearn预测糖尿病
Apr 22 Python
再谈python_tkinter弹出对话框创建
Mar 20 Python
Python3中多线程编程的队列运作示例
Apr 16 #Python
使用Python脚本操作MongoDB的教程
Apr 16 #Python
使用Python中的greenlet包实现并发编程的入门教程
Apr 16 #Python
利用Python的Twisted框架实现webshell密码扫描器的教程
Apr 16 #Python
使用Python的Twisted框架实现一个简单的服务器
Apr 16 #Python
使用Python的Twisted框架编写简单的网络客户端
Apr 16 #Python
从Python的源码浅要剖析Python的内存管理
Apr 16 #Python
You might like
再说下636单管机
2021/03/02 无线电
探讨php中header的用法详解
2013/06/07 PHP
PHP调用API接口实现天气查询功能的示例
2017/09/21 PHP
php微信公众号开发之欢迎老朋友
2018/10/20 PHP
Javascript YUI 读码日记之 YAHOO.util.Dom - Part.3
2008/03/22 Javascript
jQuery的三种$()
2009/12/30 Javascript
jquery $.getJSON()跨域请求
2011/12/21 Javascript
jquery ajax提交整个表单元素的快捷办法
2013/03/27 Javascript
js中浮点型运算BUG的解决方法说明
2014/01/06 Javascript
微信小程序本作用域下调用全局JS详解及实例
2017/02/22 Javascript
详解vue前后台数据交互vue-resource文档
2017/07/19 Javascript
JavaScript实现音乐自动切换和轮播
2017/11/05 Javascript
angularjs数组判断是否含有某个元素的实例
2018/02/27 Javascript
在vue中使用jointjs的方法
2018/03/24 Javascript
Vue-router的使用和出现空白页,路由对象属性详解
2018/09/03 Javascript
微信小程序上线发布流程图文详解
2019/05/06 Javascript
JS事件流与事件处理程序实例分析
2019/08/16 Javascript
[02:22]完美世界DOTA2联赛PWL S3 集锦第一期
2020/12/15 DOTA
Python 序列化 pickle/cPickle模块使用介绍
2014/11/30 Python
python实现输入数字的连续加减方法
2018/06/22 Python
python实现合并多个list及合并多个django QuerySet的方法示例
2019/06/11 Python
Pandas库之DataFrame使用的学习笔记
2019/06/21 Python
python中的反斜杠问题深入讲解
2019/08/12 Python
Python PO设计模式的具体使用
2019/08/16 Python
python 实现简单的FTP程序
2019/12/27 Python
Python QTimer实现多线程及QSS应用过程解析
2020/07/11 Python
python音频处理的示例详解
2020/12/23 Python
草莓网美国官网:Strawberrynet USA
2016/12/11 全球购物
说出一些常用的类,包,接口
2014/09/22 面试题
进修护士自我鉴定
2013/10/14 职场文书
分公司经理任命书
2014/06/05 职场文书
农村党支部书记司法四风问题对照检查材料
2014/09/26 职场文书
法律意见书范文
2015/05/20 职场文书
领导干部学习三严三实心得体会
2016/01/05 职场文书
教你使用VS Code的MySQL扩展管理数据库的方法
2022/01/22 MySQL
Go语言 详解net的tcp服务
2022/04/14 Golang