通过使用appium-desktop录制脚本,编写app自动化脚本的过程中,会使用到一些AW,下面就这些AW的使用方法做详细的介绍。通过实践可以看到这几个AW可以完成测试工作。
AWOpenGivePage
1、功能描述
路由打开给定的页面
2、字段描述
字段名称 |
是否必须 |
字段类型 |
描述 |
pageRouter |
是 |
String |
需要打开页面的pg或者js地址。框架中目前配置了一些页面的PG,录脚本时会自动填充,如果没有配置过,脚本录制后需要手动修改下脚本。或者联系我在框架中添加(需要提供页面的activity和PG) |
3、举例
{ "className": "AWOpenGivePage", "remark": "打开指定页面", "pageRouter": "https://xxxxx/xxxx/xxx/xxxx.weex.js" } { "className": "AWOpenGivePage", "remark": "打开指定页面", "pageRouter": "en51cc://xxx/xxxx/xxxx" } |
AWAppiumDriver
1、功能描述
driver的初始化、关闭
2、字段描述
字段名称 |
是否必须 |
字段类型 |
描述 |
action |
是 |
String |
动作 open:初始化driver close:关闭driver |
url |
是 |
String |
请求appium-server地址;默认:http://127.0.0.1:4723/wd/hub |
allRecord |
否 | Boolean | 是否都是录制脚本,默认false;true时不再加载数据库中保存的控件元素(非录制脚本,控件元素是通过接口落地到数据库中,执行脚本时需要从数据库中读取出来) |
appActivity |
否 | String | 要启动app的首页activity,默认管家 |
appPackage |
否 | String | 要启动app的包名,默认管家 |
3、举例
{ "className": "AWAppiumDriver", "url": "http://127.0.0.1:4723/wd/hub", "action": "open", "allRecord": "true", "appActivity":"com.xxxx", "appPackage":"com.xxxx" } |
AWAppiumRecordAction
1、功能描述
提供多种查找元素的方式,以及页面操作执行。该AW内容一般都是录制出来的,特殊点见字段描述
2、字段描述
字段名称 |
是否必须 |
字段类型 |
描述 |
elementInfoList
|
是 |
List<List<String>> |
具体操作步骤:["美团外卖店主","click", "accessibility id","美团外卖店主"] 1、支持的操作click、sendKeys、tap、swipe、tapSendKey。 2、支持在上面的click/sendKeys操作前面加check,默认false。比如:checkclick:true、checkclick、checksendKeys:false。作用是在操作之前检查元素是否存在,true标示一次执行期间每次都检查,否则只检查一次。 3、1中支持的所有操作,除了tapSendKey,其它都是通过录制脚本录制。tapSendKey使用于有些页面输入框元素定位不到,只能通过定位坐标的方式进行输入。录制时可以先录制为tap操作,录制成功后(["坐标点击","tap", "","0.4361111111111111","0.2875"]),手动修改脚本,tap修改为tapSendKey,数组后面添加需要输入的内容(["坐标点击","tapSendKey", "","0.4361111111111111","0.2875","15267081011"]) 4、每种操作的操作步骤都可以在数组的最后面添加等待时间,默认10秒钟。比如:["美团外卖店主","click", "accessibility id","美团外卖店主","15"] 5、元素定位方式,脚本录制中主要使用id、accessibility id、xpath、h5classname,其中xpath有时候录制出来的是全路径,为了提高执行效率可以手动修改为相对路径;当录制页面是webView页面时,定位输入框使用h5classname方式,目的是兼容不同型号、不同系统的手机对该页面解析可能不一致的问题。除了上面提到的三种方式,常用还有classnames方式 6、classnames使用方式,场景:需要定位的元素存在多个,除了xpath以外,其它一样。录制完成后(["点击相册","click", "xpath","//android.widget.TextView[@text='相册']"])修改xpath修改为classnames,后面路径修改为classname:instance格式(android.widget.TextView:11) |
checkElementInfo
|
否 |
List<List<String>> |
具体操作步骤:["是否有更新提示","click","xpath","//android.widget.TextView[@text='检查到测试包有更新,是否安装?']","5"], 1、检查该字段第一个元素是否存在,存在就执行该字段所有的操作步骤,否则不执行,继续执行elementInfoList中的操作步骤 2、功能与上个字段的check功能类似,两个可以相互切换。不同点是有多个步骤需要检查时使用check可以一个AW完成,使用checkElementInfo需要多个AW分开写。
|
3、举例
纯粹演示AW使用方法,内容步骤无意义
{ "className": "AWAppiumRecordAction", "remark": "用户登录", "checkElementInfo":[ ["是否有更新提示","click","xpath","//android.widget.TextView[@text='检查到测试包有更新,是否安装?']","5"], ["点击下次更新","click","xpath","//android.widget.CheckBox[@text='不再提醒']","1"] ], "elementInfoList": ["点击不再提示","checkclick:true","xpath","//android.widget.TextView[@text='下次再说']","1"], ["账号密码登录","click", "id","com.zhangdan.app:id/TextView_Go_UserName"], ["重试","checkclick", "accessibility id","重试"] ["11位手机号码","sendKeys", "id", "com.zhangdan.app:id/EditText_Passwd","${passwd}"], ["11位手机号码","sendKeys", "h5classname", "android.widget.EditText:1","${passwd}"] ["如果有密码键盘的情况,需要滑屏","swipe", "","0.47685185185185186","0.3758278145695364","0.48055555555555557","0.2052980132450331","5"], ["坐标点击","tapSendKey", "","0.3537037037037037","0.3859375","111111"] ["坐标点击","tap", "","0.4861111111111111","0.5083333333333333"] ["招行","click""classnames", "android.widget.TextView:11"] ] } |
AWRecordAssert
1、功能描述
结果校验,支持校验元素是否存在、元素各个属性的值以及控件颜色等
2、字段描述
字段名称 |
是否必须 |
字段类型 |
描述 |
checkElementInfo |
是 |
List<List<String>> |
需要检验的各元素["手动输入花呗账单","accessibility id", "3","手动输入花呗账单"] 1、元素定位方式除了AWAppiumRecordAction提到,还支持toast、toastLike方式。 2、每个校验项数组后面支持增加等待时间,默认10秒。 3、数组中第三个元素标示校验项,3标示检验元素存在。还支持元素其它属性值校验,替换掉3即可,比如:text、checkable、clickable等。需要注意除了3以外,填写其它校验项时需要在数组后面添加一个期望的值 4、支持颜色校验["手动输入花呗账单","accessibility id", "color","手动输入花呗账单","#FFFFF"],由于期望的颜色是十六进制颜色码,所以使用时,期望值可以先随便填写一个,运行脚本后查看日志打印的实际的十六进制颜色码是多少。再修改脚本。 |
checkExit |
否 |
Boolean |
检验元素是否存在,默认true,检验元素存在,否则检验元素不存在 |
3、举例
{ "className":"AWRecordAssert", "checkElementInfo":[ ["", "id","3","com.ali.user.mobile.security.ui:id/viewContainers"] ["所有待还 (元)","id", "text","com.zhangdan.app:id/tv_line1","所有待还 (元)"], ["招行 网银37 1001","classnames", "text","android.widget.TextView:18","6666"] ["淘宝店主", "accessibility id","3","淘宝店主"] ["toast校验","toastLike", "3","请输入有效密码"] ["手动输入花呗账单","accessibility id", "color","手动输入花呗账单","#FFFFF"] ] } |
AWKeyBoardAction
1、功能描述
纯粹键盘操作,不依赖元素。
2、字段描述
字段名称 |
是否必须 |
字段类型 |
描述 |
actionName |
否 |
String |
操作名称 默认back,返回到上一级。 hideKeyboard 隐藏键盘 switchToNative 切换到native |
3、举例
{ "className":"AWKeyBoardAction" } { "className":"AWKeyBoardAction", "actionName":"hideKeyboard" } |
AWSleep
1、功能描述
线程等待时间
2、字段描述
3、举例
{ "className": "AWSleep", "seconds": "1" } |
AWHttpClient
1 功能描述
发送HTTP请求到服务端,并按照一定规则校验服务端返回的响应消息。
2 字段描述
字段名称 | 是否必须 | 字段类型 | 描述 |
---|---|---|---|
request |
是 |
HttpClientRequest |
Http request相关信息,具体请参见下表2.1 |
reponseExpected |
是 |
HttpClientResponse |
Http reponseExpected 相关信息,具体请参见下表2.2 |
config |
否 |
MapComparedConfig |
MapComparedConfig相关信息,具体请参见1.1.1 |
retryTime | 否 | int | 请求重试次数(不包含原请求),默认为0,如值为3,则表示最多请求4次 |
retrySecond | 否 | int | 重试的间隔时间,单位为秒,默认立即重试 |
retryStateCodeList | 否 | List<Integer> | 匹配的响应状态码集合,如[500, 501]表示如果响应状态码为500或501时则进行重试,不设置时根据reponseExpected的stateCode与实际响应状态码比对,不一致则进行重试 |
重试策略说明:
1.若响应返回为null则直接进行重试
2.如果未设置retryStateCodeList或retryStateCodeList为空集合,则判断reponseExpected.getStateCode()是否与实际的状态码是否一致,不一致则进行请求重试
3.如果retryStateCodeList不为空集合,则判断retryStateCodeList集合是否包含实际状态码,若包含则进行重试
2.1 HttpClientRequest
字段名称 | 是否必须 | 字段类型 | 描述 |
---|---|---|---|
type |
是 |
String |
请求类型,1:GET 2:POST 3:PUT 4:DELETE 5:PATCH |
url | 是 | String |
request链接地址,如:"url":"http://xxxxx/xxxx/api/v1/xxxx" 或者"url":"${global.data.gjj.addr}${global.gjj.url.newaccounts}", |
query | 否 |
Map<String, String> |
query map,URL问号后面的键值对 |
path | 否 |
Map<String, String> |
path map,URL中path部分的键值对 |
headers | 否 |
Map<String, String> |
请求头 |
requestBody |
否 |
Object |
post/put请求的body,可以直接复制 swagger的请求 |
2.2 HttpClientResponse
字段名称 | 是否必须 | 字段类型 | 描述 |
---|---|---|---|
stateCode |
是 | String | 响应的状态码,如 "stateCode":"200" |
responseBody |
否 | Object | 响应的body,可以直接复制swagger返回的消息 |
headers |
否 |
Map<String, String> |
响应头 |
3 举例
{ "remark": "调用address接口", "className": "AWHttpClient", "config": { "primaryKeys": [ "orderNo" ], "keyNameAndPrimarykeys": { "items": [ "itemUrl" ] } }, "request": { "type": 1, "url": "${global.data.ecommerce.addr}/xxxx/api/v1/xxxx/{ecommerce_id}/xxxx/xxxx", "path": { "ecommerce_id": "${global.taobao.ecommerceid}" }, "headers": { "content-type": "application/json", "userId": "${global.auth.userid}", "Authorization": "${global.auth.token}" } }, "reponseExpected": { "stateCode": "200", "responseBody": [ { "dealDate": "2017-04-17T13:16:36.000+08", "orderNo": "3234925410420512", "amount": 78.35, "address": "中华人民共和国", "consignee": "张三", "phoneNumber": "1816", "items": [ { "itemUrl": "//trade.taobao.com/trade/detail/tradeSnap.htm?tradeID=3234925410430512", "itemName": "十二生肖传说精装图画书海豚绘本花园适合3岁以上亲子阅读正版童书", "price": 17.4, "quantity": 1 } ] } ] } |
AWTemplateActuator
1、AW变为模版
正常编写完AW,在测试用例中作为一个步骤,格式如下:
举例:
{ "className": "AWAppiumRecordAction", "remark": "", "mobile":"15210001001", "password":"qazxsw" } |
1.1、增加下面两个参数
"awIsTemplate":true,
"templateName":"youname"
1.2、参数化
把变量参数化,比如上面的例子中,需要把mobile和password的值参数化
上面测试步骤变成的模版如下:
{ "className": "AWAppiumRecordAction", "remark": "", "awIsTemplate":true, "mobile":"${mobile}", "password":"${password}" "templateName":"loginTemplate" } |
原有AW只要是继承了ActionWord类都支持该功能。
2、模版引用
使用框架中的 AWTemplateActuator引用模版。该AW唯一参数citationTemplates是一个对象的list。对象属性包括
templateName–引用的模版名称
templateParamValue–模版中参数值
assertTemplateName–校验结果的模版名称
举例:
比如引用上面的模版的使用方法如下:
{ "className": "AWTemplateActuator", "citationTemplates": [ { "templateName": "loginTemplate", "templateParamValue": { "mobile": "15267", "password": "2222" }, "assertTemplateName": "loginSuccess"----这是把校验结果作为一个模版 } ] } |
当然如果你参数值是固定的,可以不用参数,模版引用中只需要不传递templateParamValue参数即可。
2.1、扩展
对于输入我们一般会做很多非法性校验,使用模版执行器只需要配置参数值即可,比如上面的例子:
{ "className": "AWTemplateActuator", "citationTemplates": [ { "templateName": "loginTemplate", "templateParamValue": { "mobile": "15267",--手机号位数不够"password": "2222" }, "assertTemplateName": "loginSuccess"----这是把校验结果作为一个模版 }, { "templateName": "loginTemplate", "templateParamValue": { "mobile": "152671111111111",---手机号过长"password": "x@#@" }, "assertTemplateName": "loginSuccess"----这是把校验结果作为一个模版 } ] } |
3、模版路径
3.1、模版和非模版混合使用
一个用例文件中可以包含AW模版和非AW模版两种类型,但是对于模版记得使用AWTemplateActuator执行模版
3.2、模版路径方法
为了模版的重复使用,可以把模版放到一个文件夹里面,用例文件只负责使用AWTemplateActuator调用模版
3.2.1、新建模版文件夹
新建文件夹,把所有模版都放到该文件夹里面
3.2.2、增加配置
application.propertites中增加template.path:./xx/xx/xxx/xxx。
其它不变。
更多文章请关注公众号