剖析美团内部所采用的网站压力测试方案

所属分类: 网站运营 / 网站优化 阅读数: 136
收藏 0 赞 0 分享

美团内部的RPC服务大多构建在Thrift之上,在日常开发服务的过程中,需要针对这些服务进行压力测试(以下简称压测)来发现潜在问题。常用的方法有:

1.使用一些脚本语言如:Python、Ruby等,读取线上日志构建请求,用多线程模拟用户请求进行压测
2.使用开源工具进行压测
然而,无论采取哪种方法,压测都是一个十分耗时而又繁琐的过程,主要痛点有:

需要写很多代码解析日志,还原请求,对于比较复杂的请求,解析很容易出错
需要搭建脚本或者工具的运行环境,通常这一过程比较耗时
由于打压方法没有统一,导致打压的结果指标比较混乱,有的结果甚至以终端输出的方式展示,非常不直观
对一个应用的打压测试,由于环境、代码的问题,导致组内同学很难共享
针对上述问题,提供一个简单好用的压测工具是十分有必要的。

是否有必要重复造轮子
在构建压测工具之前,对于一些现有的开源工具进行了调研。现在主流的压测工具主要有以下几个:

JMeter
JMeter是一个比较老牌的压测工具,主要针对HTTP服务进行打压,该工具在以下方面并不满足美团内部的压测需求:

1.默认不支持Thrift的打压测试
2.需要本地安装,并且配置复杂
3.对于用户操作并不友好
201632393024299.png (1045×359)

twitter/iago
iago 是一个由Twitter开源的压测工具,支持对HTTP、Thrift等服务进行压测,其主要问题如下:

对每个压测应用都需要创建一个项目
压测结果并不直观
流量重放依赖本地文件
项目依赖于一个较老版本的Scala,搭建不便
相关文档比较少
除此之外,当时还考察了GatlingGrinderLocust 等一些常见的压测工具,都因为适用场景和美团的需求有些出入而排除了。

综上,针对当前压测工具的一些现状,构建一个简单易用的压测工具还是很有必要的。

目标
针对之前提到的痛点,新的压测工具主要提供以下功能:

线上流量拷贝
1.简单易用的操作界面(接入压测的时间应该控制在1小时以内)
2.清晰的图表能反映压测应用的各项指标
3.满足包括Thrift、HTTP等服务的压测需求


如何构建
抽象
目标已经明确,怎么实现呢?首先是抽象压测的过程。
一个典型的压测过程如图所示,首先在init方法里面,进行一些初始化的工作,比如连接数据库,创建客户端等。接下来,在run方法里面发出压测请求,为了保证能够对服务产生足够的压力,这里通常采用多线程并发访问,同时记录每次请求的发起时间和结束时间,这两个时间的简单相减就能够得到每次请求的响应时间,利用该结果就可以计算出TP90、平均响应时间、最大响应时间等指标,等压测结束后,通过destroy方法进行资源回收等工作。
201632393116380.png (1490×592)

以上过程可以用接口表示,无论是压测Thrift服务还是HTTP服务,本质上都是这三个方法实现的不同。考虑到压测工具的灵活性和通用性,压测工具可以将这个接口交给打压测试的同学实现,而压测工具则重点实现多线程打压,打压结果的聚合等比较耗时的工作。

复制代码
代码如下:

interface Runner {
def init(Test app) // 初始化压测
def run(Test app, String log) // 每次打压请求,传入log方便构建请求
def destroy(Test app) // 压测完毕后,回收资源
}


拷贝流量
Thrift服务打压的难点之一就是如何简单地拷贝线上真实流量用来构建打压请求。一些大型的Thrift服务数据结构非常复杂,写打压脚本的时候需要很多代码来解析日志,而且容易出错。 因此提供一个简单好用的拷贝流量方法是十分有必要的。

在这里压测工具提供了一个叫VCR(录像机)的工具来拷贝流量。VCR能够将线上的请求序列化后写到Redis里面。

考虑到用户需要查看具体请求和易用性等需求,最终选取了JSON格式作为序列化和反序列化的协议。同时需要部署在生产环境,为了降低对线上服务的影响,这里采取了单线程异步写的方式来拷贝流量。
201632393337313.png (524×402)

聚合数据
应用打压完成后,需要一些指标来评估压测结果,常见的指标有:

1.最大响应时间
2.平均响应时间
3.QPS
4.TP90
5.TP50
压测工具采用了 InfluxDB 来完成数据的聚合工作。
以TP90为例子,仅需要一行查询就能实现需求。

SQL Code复制内容到剪贴板
  1. SELECT PERCENTILE(response_time, 90) FROM test_series GROUP BY time(10s)  

架构
整体而言,整个打压过程如下:
201632393434896.jpg (1372×996)

实践
拷贝流量
美团内部的服务大多使用Java来构建,VCR以Maven Package的方式提供给用户。

对用户来说只需要2行代码可以拷贝流量。

为了不影响线上服务,通常选取单台机器进行流量拷贝工作。

Java Code复制内容到剪贴板
  1. public class TestAppRPC implements TestApp.Iface {   
  2.   
  3.     private Vcr _vcr = new Vcr("testapp"); // 指定拷贝流量的key   
  4.   
  5.     @Override  
  6.     public TestResponse echo(TestRequest req) throws TException {   
  7.         _vcr.copy(req); // 拷贝操作   
  8.         long start = System.currentTimeMillis();   
  9.         TestResponse response = new TestResponse();   
  10.         return response;   
  11.     }   
  12. }  

一旦流量拷贝完成后,通过Web界面,用户能够查看日志的收集情况和单条日志的详情。
201632393521398.jpg (2782×1310)

压测逻辑实现
压测工具采用Groovy来进行编写。对每个应用来说,只需要实现runner接口就可以实现对应用的打压。

复制代码
代码如下:

interface Runner {
def init(Test app)
def run(Test app, String log)
def destroy(Test app)
}

以Thrift服务为例:

Java Code复制内容到剪贴板
  1. class TestServiceRunner implements Runner {   
  2.   
  3.     RPCService.Client _client   
  4.     TTransport _transport;   
  5.   
  6.     @Override  
  7.     def init(Test app) {   
  8.         def conf = app.config // 读取应用配置   
  9.         _transport = new TFramedTransport(new TSocket(conf.get("thrift_service_host") as String, conf.get("thrift_service_port") as int))   
  10.         TProtocol protocol = new TBinaryProtocol(_transport)   
  11.         _client = new RPCService.Client(protocol)   
  12.         _transport.open()   
  13.     }   
  14.   
  15.     @Override  
  16.     def run(Test app, String log) {   
  17.         TestRequest req = Vcr.deSerialize(log, TestRequest.class// 将拷贝流量反序列化   
  18.         _client.echo(req) // 发送请求   
  19.     }   
  20.   
  21.     @Override  
  22.     def destroy(Test app) {   
  23.         _transport.close() // 关闭服务   
  24.     }   
  25. }  

创建应用
实现以上接口后,就可以对应用进行打压了。

用户可以通过Web界面创建应用,除了必填配置以外,用户可以按照应用灵活配置。
201632393558352.jpg (2746×1034)

性能指标
用户可以通过直观的图表来查看应用的各种性能指标。
201632393619103.jpg (2880×2174)

结束语
压测工具上线以来,已经接入了20多个应用,完成数百次打压实验,现在应用的接入时间仅需要15~30分钟。保证了美团服务的稳定和节省了开发同学的时间,使大家告别了以往繁琐冗长的打压测试。

欢迎对这方面有兴趣的同学一起讨论。

更多精彩内容其他人还在看

如何优化核心关键词?优化核心关键词的方法

如何优化核心关键词?这是一个对网站优化来说比较重要的事,下面小编就给大家带来优化核心关键词的方法,一起来看看吧
收藏 0 赞 0 分享

网站怎么更新优质的内容?

从网站优化刚兴起的时代,互联网中就一直流传着“内容为王,外链为皇”,也就是在网站搭建完成之后,只要做好内容与外链的建设,就能让网站关键词获得非常不错的排名。
收藏 0 赞 0 分享

网站结构的优化注意哪些方面?SEO五要素之网站结构优化

网站结构注意事项,怎么优化才合理。那么,相信各位站长都知道,如果网站没有一个合理的结构,从百度搜索引擎的角度去看的话,不合理的网站结构将会严重的影响搜索引擎对其抓取、索引以及排名和权重。那么网站结构的优化注意哪些方面?下面我们一起来了解一下吧
收藏 0 赞 0 分享

电商类网站SEO优化策略的原则有哪些?

对于一个合格的SEOer来说,大型电子商务平台的SEO要怎么做呢?本文就给大家详细介绍下电商类网站SEO优化策略的原则,一起来看看吧
收藏 0 赞 0 分享

长尾关键词如何挖掘?新手学会这八个方法也能轻松挖掘长尾关键词

做SEO的你应该知道,长尾关键词能给网站带来源源不断的流量,是网站流量的基础,如何高效的长尾关键词挖掘,是站长朋友们时刻关心的问题。本文小编为大家分享八种比较有效的长尾关键词挖掘方法,希望对大家有所帮助和启发
收藏 0 赞 0 分享

为了排名网站采取大量原创内容是否是最佳的优化路径?

为了排名网站采取大量原创内容是否是最佳的优化路径?现在百度很支持原创,有些网站为了更好的排名,都大批量的发原创,这是否是最佳的优化路径?下面我们就来看看详细的内容,需要的朋友可以参考下
收藏 0 赞 0 分享

如何从网站中挖掘更多有价值的关键词?

关键词就如打仗时的子弹,没有子弹,你无法保障战斗的顺利进行,同样,没有关键词,你难以保障流量的提升、订单收益,那么如何从网站中挖掘更多有价值的关键词?下面小编就为大家介绍一下,来看看吧
收藏 0 赞 0 分享

网站高质量友链和外链怎么发?具体方法详解

后期网站的友链和外链的作用将越来越少,也就是单靠换友情链接和发外链是无法提升网站的优化排名的。那问题来了,我们应该如果操作网站高质量友情以及如何发外链呢?具体的方法是哪些?来看看吧
收藏 0 赞 0 分享

网站为什么会出现被K?企业网站优化几个要点分析

说企业网站越来越难做了,怎么做都没有好的排名。其实你有没有想过,你负责的网站,到底有没有按照百度搜索引擎优化指南的要求去做,出个分析我们看看网站为什么会出现被K的情况发生,来看看吧
收藏 0 赞 0 分享

百度文库中的文章的排名是怎么优化的?

百度文库中的文章的排名是怎么优化的?大家都知道,我们喜欢在百度里发文章,原因只有一个,流量很高,但是百度文库中的文章想要做好的排名也很难,该怎么优化呢?下面我们就来看看详细的教程,需要的朋友可以参考下
收藏 0 赞 0 分享
查看更多