java脚本接口自动化测试_接口自动化测试实践的记录

接口测试实践的记录

在敏捷开发交付的流程中,自动化测试实际上被放在一个看起来挺重要的位置,而自动化测试中,接口测试是一个投入产出比比较高的

一种自动化测试的形式,而我自己也做了一个这样的脚手架一样的东西可以方便进行自动化测试,关键是在一些现有第三包的基础上做实现,其实一个脚手架不需要几个 JAVA 类就可以完成了,至少我自己的这个在 10 个文件以内.要论行数估计也没有多少代码量,主要时间其实都是在想怎么更方便的写自动化测试,怎么使用以后的开源代码了。

下面介绍一下我自己如何完成这个自动化接口测试

脚手架设计和实现的,以及我自己实现过程中的种种发现。主要从以下几个方面来讲:

如何构建接口自动化测试的脚手架

关于接口测试参考的一些资源

关于接口测试的后续的一些想法

如何构建接口自动化测试的脚手架

接口测试本文中主要是指 HTTP 的请求,构建接口自动化测试脚手架的时候,首先先看看平常接口测试,测试人员时如何做的,我了解主要是以下几种方式:

通过操作页面/APP 来触发接口调用

使用诸如 SOAPYUI/JMETER/POSTMAN 或者其他的客户端工具来进行接口测试

我自己都使用过 SOAPUI/JMETER/POSTMAN,不能说使用的多么深入,但是常用的功能也都有用过,比如 SOAPUI 构建一个项目完整的接口自动化测试用例,大概有 200+ 以上的用例,可以支持不同的测试环境,检查点中可以检查数据库,使用 XPATH/XQUERY 来检查/获取指定的值,进行不同 API 的数据传递等等,这些工具 (指功能测试方面) 大体的逻辑我觉得是类似的,基本上都有:

发起请求的客户端,需要测试人员构建,也有通过 WSDL/WADL 自己生成的,不过数据都是需要测试人员输入的

根据表达式进行取值的 Resolver,就是可以根据 XPATH/XQUERY 语法,或者其他的语法来获取指定的值,

就是用来传递上下文数据的一种方式

外部可以参数话数据,比如环境配置

可以查看测试结果,这个其实可以理解为某种测试框架的一个功能,比如 JUNIT,TESTNG

接口自动化测试脚手架的构建

根据以上的分析如果自己需要实现的话,最主要需要实现一下其实就是请求的构建,请求构建包括了:

发起请求的客户端

请求数据的构建

对于发起请求的客户端就直接使用了 Spring RestTemplate,考虑的主要原因如下:

使用相对比较方便,模块化比较清晰

可以使用 HTTPClient 的实现

Spring RestTemplate 所在的包还有其他一些接口的支持,以后如果使用其他接口可以不需要换包也可以做

在实际的使用过程中,其实也遇到了一些问题,比如如下的内容:

HTTPS 的访问

开发接口定义不够准确的问题,造成使用 RestTemplate 时候出现了一些不在开始预期范中的问题

如何解决这些问题,在后面再详细介绍,这里说明一下使用 RestTemplate 的一个主要流程:

1. 构建请求,设置请求的 Header,URL,Accept,ContextType,Token 等等

2. 调用请求获取返回的 Response,

这个 ResponseRestTemplate 中实际上封装了一个 ResponseEntity 的类,里面包括了请求状态,Body 之类

RestTemplate 有个好处就是如果给 RestTemplate 设定了 MessageConverter 的话,他可以自动把请求的返回类型直接转换,比如你发起请求的时候设置了 JOSN 的 Message Converter,他可以帮你把类,或者字符串自己转化为 JSON 来发送,同样如果是返回值是 JSON 的话,也可以帮你自己将 JSON 转换成你指定类型的 JAVA BEAN

说完这个流程,我们就说说如何通过 RestTemplate 构建一个简单的 HTTP 请求:

Map urlVariable = new Map ();

urlVariable.put("q","test");

JavaBean javaBean = restTemplate.getForObject("http://www.baidu.com",JavaBean.class,urlVariable);

JavaBean javaBean1 = restTemplate.postForObject("http://www.baidu.com",JavaBean.class,urlVariable);

ResponseEntity e = restTemplate.getForEntity("http://www.baidu.com",JavaBean.class,urlVariable);

实际上使用 RestTemplate 还是挺简单的,不过为了让使测试更为方便一点,然后每个人的代码更统一点,自己重新封装了一下 RestTemplate 的使用,主要分为三个概念:

Service 的描述

测试数据

客户端调用

接口服务描述

Service 的描述实际上就是一个 JSON 文件,只不过自己规定了一下,格式类似于,这个文件描述了 API 的定义,当然 API 的 body 没有在这个里面,不过为了不把事情搞复杂,就暂时不放在这个里面.

{

"apiDomainName": "applicationName",

"contentType": "application/x-www-form-urlencoded",

"headers": {

"Accept": "application/json, text/javascript, */*"

},

"method": "POST",

"pathParameters": [],

"queryParameters": [

"username",

],

"resourceURL": "/application/subdomain"

}

测试数据类:

private Map queryParameters = Maps.newHashMap();

private Map pathParameters = Maps.newHashMap();

private Map headers = Maps.newHashMap();

private T body;

而如何调用客户端就变成,而且其实每一个 API 的访问其实都可以这样子来做,

ResponseEntity response = RestTemplateHelper.build(serviceDescriptionPath,requestData).call();

说明一下的是:

serviceDescriptionPath 就是接口的描述

requestData 就是需要进行测试的数据

然后实际上接口的描述是开发还没有开发好的时候就已经定了的,所以这里的变量就变成如何构建 requestData 了

构建 RequestData

构建 requestData 实际上就是设计测试用例,那么这里也是使用 Excel 的方式,将不同的值填写到 excel 里面,不过为了减少 set 值这样的操作,这个脚手架就提供了一些工具,可以直接将数据设置到 RequestData 实例,具体的操作如下:

Excel 是如下格式的:

变量名

测试用例 1

测试用例 2

data.queryParameters(username)

1

1

data.queryParameters(year)

2015

2014

data.queryParameters(month)

10

11

说明一下,通过反射的方式,可以直接生成一个 requestData 的实例,同时 queryParameters 中值已经设置好了,这样调用代码中就不需要写类似于:

RequestData data = new RequestData();

data.queryParameters.put("username","1");

data.queryParameters.put("year","2015");

这里有兴趣的同学可以参考这个包:里面其实已经有很方便的通过反射去赋值了,

org.jodd

jodd-bean

3.6.6

使用 TestNG 的 DataProvider

刚才讲述了如何发生生成数据,那么通过 Excel 的方式提供不同的数据,就可以通过 TestNG 的 DataProvider 了

所以测试数据通过,TestNG data provider 的实现在这里就不多少了,网上其实有很多内容了.

接口测试的代码看起来就是这个样子了 @DataProvider(name = "data")

public Iterator getAPITestData(Method m) throws Exception {

Map clazz = new HashMap();

clazz.put("RequestData", RequestData.class);

Iterator y = TestData.provider("testcase/api1.xls", m, clazzMap);

return y;

}

@Test(dataProvider = "data")

public void testAPITest(RequestData data) {

ResponseEntity response = RestTemplateHelper.build(serviceDescriptionPath,requestData).call();

Assert.assertEqual(response.getStatus,200); // response 的期望值实际可以通过dataprovider传入

}

而且几乎所有的代码都差不多成这个样子了,那么获取可以写个代码生成的东西,当然最后通过了 JsonPath 写了一些获取 JSON 值的工具,这个暂时也就不说了.

那么代码生成吧

当封装好这些东西之后,发现所有的接口都类似了,然后就做了代码生成的工具了,代码生成器的入口实际上个就是那个服务描述文件开始的,

所以代码生成器的参数就是服务描述文件,在实际的使用的过程中,接口描述这个文件也可以自动生成,目前总共支持以下几种:

手动编写描述文件

抓取开发 API 规格网站接口的描述,自动生成描述文件

解析 HAR 文件自动生成描述文件,解析 HAR 其实不难,就是繁琐一点字段有点多

后续想打通和 POSTMAN 的连接,可以接收 POSTMAN 的导出文件,然后也可以导出 POSTMANT 的,以后开 BUG 就什么也不说,直接放一个 POSTMAN 文件其实也挺帅的

至此一个接口测试的脚手架就大致完成了.总结起来就是:

封装了 RestTemplate,让他接受一个接口的描述文件,一个请求的数据

通过 Excel 传数据给请求的数据进行数据驱动

相同类似的代码进行代码生成

最后其实这样子使用下来,接口构建几个简单一点的自动化测试用例,其实也就是几分钟的事情.

一些细节

在实现过程中,实际上还有一些特殊情况,比如说需要 token,认证信息,这些通过一个公用函数的方式就可以解决,然后在代码生成的时候

直接讲这个放在实际测试的接口前面调用. 后有就是上面说到的的:

HTTPS 的访问

开发接口定义不够准确的问题,造成使用 RestTemplate 时候出现了一些不在开始预期范中的问题

HTTPS 的访问是通过如下代码解决的,创建一个略 SSL 的 httpclient 就可以了

public static RestTemplateClientHelper getHttpClientImplInstance(){

RestTemplateClientHelper client = new RestTemplateClientHelper();

HttpClient httpClient = getIgnoreSSLHttpClient();

client.setTemplate(new RestTemplate(new HttpComponentsClientHttpRequestFactory(httpClient)));

return client;

}

/**

* 获取忽略SSL的httpclient,支持https的请求

* @return

*/

private static HttpClient getIgnoreSSLHttpClient() {

CloseableHttpClient httpClient = null;

try {

httpClient = HttpClients.custom().

setHostnameVerifier(new AllowAllHostnameVerifier()).

setSslcontext(new SSLContextBuilder().loadTrustMaterial(null, new TrustStrategy() {

public boolean isTrusted(X509Certificate[] arg0, String arg1) throws CertificateException {

return true;

}

}).build()).build();

} catch (NoSuchAlgorithmException | KeyManagementException | KeyStoreException e) {

logger.error(e);

}

return httpClient;

}

还有一个就是有时开发的接口返回类型 (accept type) 不能让 RestTemplate 处理,那么其实添加自己定义个 MessageConverter 就好了:

下面是一个修改阿里自己的 FastJSON 的 MessageConverter 的例子,

其实也没改什么,就是捕捉了一个异常,主要是不知道什么原因调用时候 readInternal 就抛出和编码格式有关系的异常,然后就捕捉了一下异常反正也就把那个问题就没有了,不过这个改法应该也是有问题的.

public class ModifiedFastJsonHttpMessageConverter extends AbstractHttpMessageConverter {

........

public ModifiedFastJsonHttpMessageConverter() {

super(new MediaType("application", "json", UTF8), new MediaType("application", "*+json", UTF8));

this.charset = UTF8;

this.features = new SerializerFeature[0];

}

............

protected Object readInternal(Class> clazz, HttpInputMessage inputMessage) throws IOException, HttpMessageNotReadableException {

ByteArrayOutputStream baos = new ByteArrayOutputStream();

InputStream in = inputMessage.getBody();

byte[] buf = new byte[1024];

while(true) {

int bytes = in.read(buf);

if(bytes == -1) {

byte[] bytes1 = baos.toByteArray();

try {

return JSON.parseObject(bytes1, 0, bytes1.length, this.charset.newDecoder(), clazz);

}catch (Exception e){

return baos.toString("UTF-8");

}

}

if(bytes > 0) {

baos.write(buf, 0, bytes);

}

}

}

........

}

后续的一些想法

后续希望在这个基础上再做点其他的一些事情:

增加 POSTMAN 的代码生成的支持

探索能不能通过 API 接口描述直接生成 JMETER 的 JMX 文件,可以讲基础的 JMETER 性能测试的基础代码也生成好

整理一下放到 GITHUB 上面,其实整个脚手架自己也就是几个文件而已,:)

建立一个 MOCK SERVER,方便模拟一些 API 调用的方式

做一个简单点获取 JSON 中指定字段,然后传递给下一个 API 使用的工具

一些资源

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/528700.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

java 安卓视频播放器_java - 学习做一个安卓视频播放器,有一些小问题!忘大家请教...

小葫芦2017-04-18 10:51:293楼首先解决你的一点疑惑,如何获取本地所有视频路径。Android系统在存储视频,音频,图片等资源的时候,会自动把其相关信息存储在数据库中,信息包括名字,大小,存储路径等…

java作业四_Java第四次作业

1.(二次方程式)为二次方程式ax2bxc0设计一个名为QuadraticEquation的类。这类包括:(1)代表三个系数的私有数据域a、b、c(2)一个参数为a、b和c的构造方法(3)a、b、c的三个get方法(4)一个名为getDiscriminant()的方法返回判别式,b2-4ac(5)一个名为getRoot1…

java collectiongroup 类_Java中的collection集合类型总结

java集合是java提供的工具包,包含了常用的数据结构:集合、链表、队列、栈、数组、映射等。java集合工具包位置是java.util.*java集合主要可以划分为4个部分:list列表、set集合、map映射、工具类(iterator迭代器、enumeration枚举类、arrays和…

python time localtimeq获取准确时间_python的内置模块time和datetime的方法详解以及使用(python内的time和datetime时间格式)...

time内置模块的方法1、time() 时间戳time() -> floating point number 浮点数Return the current time in seconds since the Epoch.Fractions of a second may be present if the system clock provides them.importtimeprint(time.time())C:\python35\python3.exe D:/pyp…

java权限控制最小访问原则_[Java教程]Java之路(五) 访问权限控制

[Java教程]Java之路(五) 访问权限控制0 2016-12-09 01:00:14在Java中,所有事物都具有某种形式的访问权限控制。访问权限的控制等级从最大到最小依次为:public,protected,包访问权限(无关键词)和private。public,protec…

java jndi tcp_spring配置下通过tomcat的jndi服务连接数据库

在开发OA系统因为采用了springhibernate架构,其中数据源很多,采用jdbc注入到spring中,结果时间长了,数据库容易断开,改成了jndi,jndi由spring去管理1、配置jndi文件文件位置%TOMCAT_HOME% \conf\Catalina\l…

php 下拉菜单 多个值,PHP,而foreach下拉菜单在所有下拉菜单中都具有相同的选定值...

3个带值的下拉菜单$options1 array( 1> Equals, ≠ Does not Equal, > Is greater than, ≥ Is greater than or equal to, < Is less than, ≤ Is less than or equal, ? Contains, ? Does not contain);$counter_maximum 3;while ($counter < $counter_maxim…

php 安装pdo odbc,PHP PDO ODBC连接

我们正在尝试通过PHP中的ODBC创建与SQL数据库的连接.这是我们目前的脚本&#xff1a;$cnx new PDO("odbc:Driver{EFR};Serverlocalhost;Port:7004;DatabaseEFR;UidLcLfVJFLTKTCEHRO;Pwd*********;");该驱动程序正在Qlikview中工作,该Qlikview也连接到此数据库.这个驱…

php 鼠标点击图片放大,css3如何实现鼠标放上图片放大?(附代码)

在css3的学习中&#xff0c;我们会经常做一些小的动画效果&#xff0c;这感觉非常有趣&#xff0c;所以今天的这篇文章将给大家来介绍关于css3实现图片放大的一个效果&#xff0c;有兴趣的小伙伴可以看一下。我们都知道css3中增加了一个transform属性应用于元素的2D或3D转换&am…

php 文件保存函数,php 写入和读取文件函数

//读取文件函数function readfromfile($file_name) {if (file_exists($file_name)) {$filenumfopen($file_name,"r");flock($filenum,lock_ex);$file_datafread($filenum, filesize($file_name));rewind($filenum); //osphp.com.cnfclose($filenum);return $file_dat…

java入职华为,通过这9个Java面试题,就可以入职华为啦

该楼层疑似违规已被系统折叠 隐藏此楼查看此楼1&#xff0e;Java程序的种类有(BCD )A&#xff0e;类(Class) B&#xff0e;Applet C&#xff0e;Application D&#xff0e;Servlet2&#xff0e;下列说法正确的有( BCD)A&#xff0e; 环境变量可在编译source code时指定B&#x…

java hive查询,hive查询报错

我在每次添加where条件&#xff0c;或者使用子查询时&#xff0c;都会报这个错误。以下是报错详情java.lang.Exception: java.sql.SQLException: Error while processing statement: FAILED: Execution Error, return code 1 from org.apache.hadoop.hive.ql.exec.tez.TezTaskj…

java死锁怎么用jvm调试,线程死锁演示,线程锁演示,模拟JVM的线程次序调度

线程死锁演示&#xff0c;线程锁演示,模拟JVM的线程次序调度模拟JVM的线程次序调度注释A&#xff0c;不注释B&#xff0c;一般不死锁注释B&#xff0c;不注释A&#xff0c;死锁都不注释&#xff0c;随机package org.he.bin;/** * author BenHe * email qing878gmail.com * for …

脉位调制解调 matlab,通信原理与matlab仿真v2 第五章 DBPSK调制解调器(1)

在毕业设计的相关课程中&#xff0c;已经提及了这个DPSK调制方式。不过如果把它放在理论教学课程中来讲&#xff0c;就需要补充一些基本知识。当然还会给同学们看一下实际在卫星通信中使用的DBPSK程序。之前讲到了BPSK&#xff0c;回顾一下发送滤波器的知识吧。由于信号要在信道…

matlab scatter3函数,MATLAB3维散点图scatter3plote.pptx

MATLAB3维散点图scatter3plote第三章 Matlab图形绘制;主要内容;3.1 二维曲线的绘制;yrand(100,1);plot(y);x0:0.01*pi:2*pi;y[sin(x’),cos(x’)];plot([x’,x’],y);;Matlab绘图命令中的各种选项;2、特殊的二维图形函数;②极坐标系函数polar&#xff0c;调用形式为&#xff1a…

php首页遍历出商品详情页,ECSHOP首页/分类页/详情页各页面调用显示销量

1、在首页新品、热卖、精品等调用商品销售打开 includes/lib_goods.php在文件尾部?>前添加以下代码&#xff1a;function get_buy_sum($goods_id){$sql "select sum(goods_number) from " . $GLOBALS[ecs]->table(order_goods) . " AS g ,".$GLOBA…

matlab地址数据类型uns,使用matlab生成sine波mif文件

使用matlab生成sine波mif文件作者&#xff1a;lee神在使用altera 的FPGA中的rom過程中常常會使用到.mif文件或.hex文件。對於初學者&#xff0c;無論mif還是hex都是很令人疑惑的東西&#xff0c;這里就對mif文件的格式及其創建做一點簡單的說明。Mif是memory initialization fi…

matlab计算频域动态性能指标,基于MATLAB自动控制系统时域频域分析与仿真.doc

word文档 可编辑复制word文档 可编辑复制基于MATLAB的自动控制系统时域频域分析与仿真摘 要自动控制系统就是在无人直接操作或干预的条件下&#xff0c;通过控制装置使控制对象自动的按照给定的规律运行&#xff0c;使被控量按照给定的规律去变化的系统。在现代工业生产中&…

php历史上的今天源码,代码获取历史上的今天发生的事_基础知识

//http://history.sturgeon.mopaas.com //主页//http://history.sturgeon.mopaas.com/jsonp?callback? //jsonp接口//http://history.sturgeon.mopaas.com/jsonp //json接口//http://history.sturgeon.mopaas.com/jsonp/11 //历史上的1月1日//http://history.sturgeon.mopaas…

OpenAI科学家Hyung Won Chung演讲精华版

文章目录 第一个观点&#xff1a;涌现第二个观点&#xff1a;如何扩大规模1、标记化2、嵌入3、计算4、评估&#xff08;损失函数&#xff09;5、反向传播 最近从Google跳槽到OpenAI的AI科学家 Hyung Won Chung 比较拗口&#xff0c;我就简称尚哥了 他最近做了一个技术演讲 …