结构介绍

之前分享过一篇安卓UI测试,但是没有实现数据与代码分离,后期维护成本较高,所以最近抽空优化了一下。
不想看文章得可以直接去Github,欢迎拍砖
大致结构如下:

 
结构.png
  • testyaml管理用例,实现数据与代码分离,一个模块一个文件夹

  • public 存放公共文件,如读取配置文件、启动appium服务、读取Yaml文件、定义日志格式等

  • page 存放最小测试用例集,一个模块一个文件夹

  • results 存放测试报告及失败截图

     
    report.png
  • logs 存放日志

     
    logs.png
     
    logdetail.png
  • testcase 存放测试用例
  • runtest.py 运行所有测试用例

yaml格式介绍

首先看下yaml文件的格式,之前也写过一点关于yaml语法学习的文章
testcase部分是重点,其中:

  • element_info:定位元素信息

  • find_type:属性,id、xpath、text、ids

  • operate_type: click、sendkeys、back、swipe_up 为back就是返回,暂时就四种

    上面三个必填,operate_type必填!!!!!!

  • send_content:send_keys 时用到

  • index:ids时用到

  • times: 返回次数或者上滑次数

testinfo:
- id: cm001
title: 新增终端门店
execute: 1
testcase:
-
element_info: 客户
find_type: text
operate_type: click
-
element_info: com.fiberhome.waiqin365.client:id/cm_topbar_tv_right
find_type: id
operate_type: click
-
element_info: com.fiberhome.waiqin365.client:id/custview_id_singletv_inputtext
find_type: ids
operate_type: send_keys
send_content: auto0205
index: 0
-
element_info:
find_type:
operate_type: swipe_up
times: 1
-
element_info: 提交
find_type: text
operate_type: click
-
element_info:
find_type:
operate_type: back
times: 1

代码部分

公共部分

个人觉得核心的就是公共部分,相当于建房子,公共部分搞好了,后面仅仅是调用即可,建房子把架子搭好,后面就添砖加瓦吧。

读取配置文件readconfig.py
设置日志格式logs.py
获取设备GetDevices.py
这几个通用的就不做介绍了

  • 读取yaml文件 GetYaml.py
    主要用来读取yaml文件
#coding=utf-8
#author='Shichao-Dong' import sys
reload(sys)
sys.setdefaultencoding('utf8')
import yaml
import codecs class getyaml:
def __init__(self,path):
self.path = path def getYaml(self):
'''
读取yaml文件
:param path: 文件路径
:return:
'''
try:
f = open(self.path)
data =yaml.load(f)
f.close()
return data
except Exception:
print(u"未找到yaml文件") def alldata(self):
data =self.getYaml()
return data def caselen(self):
data = self.alldata()
length = len(data['testcase'])
return length def get_elementinfo(self,i):
data = self.alldata()
# print data['testcase'][i]['element_info']
return data['testcase'][i]['element_info'] def get_findtype(self,i):
data = self.alldata()
# print data['testcase'][i]['find_type']
return data['testcase'][i]['find_type'] def get_operate_type(self,i):
data = self.alldata()
# print data['testcase'][i]['operate_type']
return data['testcase'][i]['operate_type'] def get_index(self,i):
data = self.alldata()
if self.get_findtype(i)=='ids':
return data['testcase'][i]['index']
else:
pass def get_send_content(self,i):
data = self.alldata()
# print data['testcase'][i]['send_content']
if self.get_operate_type(i) == 'send_keys':
return data['testcase'][i]['send_content']
else:
pass def get_backtimes(self,i):
data = self.alldata()
if self.get_operate_type(i)=='back' or self.get_operate_type(i)=='swipe_up':
return data['testcase'][i]['times']
else:
pass def get_title(self):
data = self.alldata()
# print data['testinfo'][0]['title']
return data['testinfo'][0]['title']
  • 启动appium服务 StartAppiumServer.py
    主要是启动appium并返回端口port,这个port在下面的driver中需要
#coding=utf-8
#author='Shichao-Dong' from logs import log
import random,time
import platform
import os
from GetDevices import devices log = log()
dev = devices().get_deviceName() class Sp:
def __init__(self, device):
self.device = device def __start_driver(self, aport, bpport):
"""
:return:
"""
if platform.system() == 'Windows':
import subprocess
subprocess.Popen("appium -p %s -bp %s -U %s" %
(aport, bpport, self.device), shell=True) def start_appium(self):
"""
启动appium
p:appium port
bp:bootstrap port
:return: 返回appium端口参数
"""
aport = random.randint(4700, 4900)
bpport = random.randint(4700, 4900)
self.__start_driver(aport, bpport) log.info(
'start appium :p %s bp %s device:%s' %
(aport, bpport, self.device))
time.sleep(10)
return aport def main(self):
"""
:return: 启动appium
"""
return self.start_appium() def stop_appium(self):
'''
停止appium
:return:
'''
if platform.system() == 'Windows':
os.popen("taskkill /f /im node.exe") if __name__ == '__main__':
s = Sp(dev)
s.main()
  • 获取driver GetDriver.py
    platformName、deviceName、appPackage、appActivity这些卸载配置文件config.ini文件中,可以直接通过readconfig.py文件读取获得。
    appium_port有StartAppiumServer.py文件返回
s = Sp(deviceName)
appium_port = s.main() def mydriver():
desired_caps = {
'platformName':platformName,'deviceName':deviceName, 'platformVersion':platformVersion,
'appPackage':appPackage,'appActivity':appActivity,
'unicodeKeyboard':True,'resetKeyboard':True,'noReset':True
}
try:
driver = webdriver.Remote('http://127.0.0.1:%s/wd/hub'%appium_port,desired_caps)
time.sleep(4)
log.info('获取driver成功')
return driver
except WebDriverException:
print 'No driver' if __name__ == "__main__":
mydriver()
  • 重新封装find等命令,BaseOperate.py
    里面主要是一些上滑、返回、find等一些基础操作
#coding=utf-8
#author='Shichao-Dong' from selenium.webdriver.support.ui import WebDriverWait
from logs import log
import os
import time '''
一些基础操作:滑动、截图、点击页面元素等
''' class BaseOperate:
def __init__(self,driver):
self.driver = driver def back(self):
'''
返回键
:return:
'''
os.popen("adb shell input keyevent 4") def get_window_size(self):
'''
获取屏幕大小
:return: windowsize
'''
global windowSize
windowSize = self.driver.get_window_size()
return windowSize def swipe_up(self):
'''
向上滑动
:return:
'''
windowsSize = self.get_window_size()
width = windowsSize.get("width")
height = windowsSize.get("height")
self.driver.swipe(width/2, height*3/4, width/2, height/4, 1000) def screenshot(self):
now=time.strftime("%y%m%d-%H-%M-%S")
PATH = lambda p: os.path.abspath(
os.path.join(os.path.dirname(__file__), p)
)
screenshoot_path = PATH('../results/screenshoot/')
self.driver.get_screenshot_as_file(screenshoot_path+now+'.png') def find_id(self,id):
'''
寻找元素
:return:
'''
exsit = self.driver.find_element_by_id(id)
if exsit :
return True
else:
return False def find_name(self,name):
'''
判断页面是否存在某个元素
:param name: text
:return:
'''
findname = "//*[@text='%s']"%(name)
exsit = self.driver.find_element_by_xpath(findname)
if exsit :
return True
else:
return False def get_name(self,name):
'''
定位页面text元素
:param name:
:return:
'''
# element = driver.find_element_by_name(name)
# return element findname = "//*[@text='%s']"%(name)
try:
element = WebDriverWait(self.driver, 10).until(lambda x: x.find_element_by_xpath(findname))
# element = self.driver.find_element_by_xpath(findname)
self.driver.implicitly_wait(2)
return element
except:
self.screenshot()
log.error('未定位到元素:'+'%s')%(name) def get_id(self,id):
'''
定位页面resouce-id元素
:param id:
:return:
'''
try:
element = WebDriverWait(self.driver, 10).until(lambda x: x.find_element_by_id(id))
# element = self.driver.find_element_by_id(id)
self.driver.implicitly_wait(2)
return element
except:
self.screenshot()
log.error('未定位到元素:'+'%s')%(id) def get_xpath(self,xpath):
'''
定位页面xpath元素
:param id:
:return:
'''
try:
element = WebDriverWait(self.driver, 10).until(lambda x: x.find_element_by_xpath(xpath))
# element = self.driver.find_element_by_xpath(xpath)
self.driver.implicitly_wait(2)
return element
except:
self.screenshot()
log.error('未定位到元素:'+'%s')%(xpath) def get_ids(self,id):
'''
定位页面resouce-id元素组
:param id:
:return:列表
'''
try:
# elements = self.driver.find_elements_by_id(id)
elements = WebDriverWait(self.driver, 10).until(lambda x: x.find_elements_by_id(id))
self.driver.implicitly_wait(2)
return elements
except:
self.screenshot()
log.error('未定位到元素:'+'%s')%(id) def page(self,name):
'''
返回至指定页面
:return:
'''
i=0
while i<10:
i=i+1
try:
findname = "//*[@text='%s']"%(name)
self.driver.find_element_by_xpath(findname)
self.driver.implicitly_wait(2)
break
except :
os.popen("adb shell input keyevent 4")
try:
findname = "//*[@text='确定']"
self.driver.find_element_by_xpath(findname).click()
self.driver.implicitly_wait(2)
except:
os.popen("adb shell input keyevent 4")
try:
self.driver.find_element_by_xpath("//*[@text='工作台']")
self.driver.implicitly_wait(2)
break
except:
os.popen("adb shell input keyevent 4")
  • Operate.py
    我认为最关键的一步了,后面没有page都是调用这个文件进行测试,主要是根据读取的yaml文件,然后进行if...else...判断,根据对应的operate_type分别进行对应的click、sendkeys等操作
#coding=utf-8
#author='Shichao-Dong' from GetYaml import getyaml
from BaseOperate import BaseOperate class Operate:
def __init__(self,path,driver):
self.path = path
self.driver = driver
self.yaml = getyaml(self.path)
self.baseoperate=BaseOperate(driver) def check_operate_type(self):
'''
读取yaml信息并执行
element_info:定位元素信息
find_type:属性,id、xpath、text、ids
operate_type: click、sendkeys、back、swipe_up 为back就是返回,暂时就三种 上面三个必填,operate_type必填!!!!!! send_content:send_keys 时用到
index:ids时用到
times:
:return:
''' for i in range(self.yaml.caselen()):
if self.yaml.get_operate_type(i) == 'click':
if self.yaml.get_findtype(i) == 'text':
self.baseoperate.get_name(self.yaml.get_elementinfo(i)).click()
elif self.yaml.get_findtype(i) == 'id':
self.baseoperate.get_id(self.yaml.get_elementinfo(i)).click()
elif self.yaml.get_findtype(i) == 'xpath':
self.baseoperate.get_xpath(self.yaml.get_elementinfo(i)).click()
elif self.yaml.get_findtype(i) == 'ids':
self.baseoperate.get_ids(self.yaml.get_elementinfo(i))[self.yaml.get_index(i)].click() elif self.yaml.get_operate_type(i) == 'send_keys':
if self.yaml.get_findtype(i) == 'text':
self.baseoperate.get_name(self.yaml.get_elementinfo(i)).send_keys(self.yaml.get_send_content(i))
elif self.yaml.get_findtype(i) == 'id':
self.baseoperate.get_id(self.yaml.get_elementinfo(i)).send_keys(self.yaml.get_send_content(i))
elif self.yaml.get_findtype(i) == 'xpath':
self.baseoperate.get_xpath(self.yaml.get_elementinfo(i)).send_keys(self.yaml.get_send_content(i))
elif self.yaml.get_findtype(i) == 'ids':
self.baseoperate.get_ids(self.yaml.get_elementinfo(i))[self.yaml.get_index(i)].send_keys(self.yaml.get_send_content(i)) elif self.yaml.get_operate_type(i) == 'back':
for n in range(self.yaml.get_backtimes(i)):
self.baseoperate.back() elif self.yaml.get_operate_type(i) == 'swipe_up':
for n in range(self.yaml.get_backtimes(i)):
self.baseoperate.swipe_up() def back_home(self):
'''
返回至工作台
:return:
'''
self.baseoperate.page('工作台')

公共部分的代码就介绍这么多,在编写这个框架的时候,大部分精力都花在这部分,所以个人觉得还是值得好好研究的

Page部分

page部分是最小用例集,一个模块一个文件夹,以客户为例,
目前写了两个用例,一个新增,一个排序,文件如下:

 
file.png

代码如下,非常的简洁,

import sys
reload(sys)
sys.setdefaultencoding('utf8')
import codecs,os
from public.Operate import Operate
from public.GetYaml import getyaml PATH = lambda p: os.path.abspath(
os.path.join(os.path.dirname(__file__), p)
)
yamlpath = PATH("../../testyaml/cm/cm-001addcm.yaml") class AddcmPage: def __init__(self,driver):
self.path = yamlpath
self.driver = driver
self.operate = Operate(self.path,self.driver) def operateap(self):
self.operate.check_operate_type() def home(self):
self.operate.back_home()

运行用例

这部分用了unittest,运行所有测试用例和生成报告。
一个模块一个用例,以客户为例:CmTest.py

from page.cm.CmAddcmPage import AddcmPage
from page.cm.CmSortcmPage import SortcmPage from public.GetDriver import mydriver
driver = mydriver() import unittest,time
class Cm(unittest.TestCase): def test_001addcm(self):
'''
新增客户
:return:
'''
add = AddcmPage(driver)
add.operateap()
add.home()
def test_002sortcm(self):
'''
客户排序
:return:
'''
sort = SortcmPage(driver)
sort.sortlist()
sort.home() def test_999close(self):
driver.quit()
time.sleep(10) if __name__ == "__main__":
unittest.main()

首先从page层将需要运行的用例都import进来,然后用unittest运行即可。
如果想要运行所有的测试用例,需要用到runtest.py

import time,os
import unittest
import HTMLTestRunner
from testcase.CmTest import Cm def testsuit():
suite = unittest.TestSuite()
suite.addTests([unittest.defaultTestLoader.loadTestsFromTestCase(Cm), ]) # runner = unittest.TextTestRunner(verbosity=2)
# runner.run(suite) now=time.strftime("%y-%m-%d-%H-%M-%S")
PATH = lambda p: os.path.abspath(
os.path.join(os.path.dirname(__file__), p)
)
dirpath = PATH("./results/waiqin365-") filename=dirpath + now +'result.html'
fp=open(filename,'wb')
runner=HTMLTestRunner.HTMLTestRunner(stream=fp,title='waiqin365 6.0.6beta test result',description=u'result:') runner.run(suite)
fp.close() if __name__ =="__main__":
testsuit()

这边的思路差不多,也是先导入再装入suite即可

总结

就目前而言,暂时算是实现了数据与用例的分离,但是yaml的编写要求较高,不能格式上出错。
同时也有一些其他可以优化的地方,如:

  • 对弹窗的判断
  • 断开后重连机制
  • 失败后重跑机制
    等等,后续可以根据需求进行优化
    最后再贴一下开源地址Github,有兴趣的小伙伴可以去看一下,欢迎拍砖
    备注:完成过程中参考了Louis-me 和 auto 这两个开源项目,感谢!!!

作者:迈阿密小白
链接:https://www.jianshu.com/p/00aff8435a92
來源:简书
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处

基于python+appium+yaml安卓UI自动化测试分享的更多相关文章

  1. python+appium+yaml安卓UI自动化测试分享

    一.实现数据与代码分离,维护成本较低,先看看自动化结构,大体如下: testyaml管理用例,实现数据与代码分离,一个模块一个文件夹 public 存放公共文件,如读取配置文件.启动appium服务. ...

  2. Python 基于python实现的http接口自动化测试框架(含源码)

    基于python实现的http+json协议接口自动化测试框架(含源码) by:授客 QQ:1033553122      欢迎加入软件性能测试交流 QQ群:7156436  由于篇幅问题,采用百度网 ...

  3. appium+python 【Mac】UI自动化测试封装框架流程简介 <一>

    为了多人之间更方便的协作,那么框架本身的结构和编写方式将变得很重要,因此每个团队都有适合自己的框架.如下本人对APP的UI自动化测试的框架进行进行了简单的汇总.主要目的是为了让团队中的其余人员接手写脚 ...

  4. ATOMac - 基于Python的Mac应用Ui自动化库

    ATOMacTest 一.缘 起 近期工作需要对一款Mac端应用实现常用功能的自动化操作,同事推荐ATOMac这款工具,这几天简单研究了下,同时也发现现网介绍ATOMac的资料非常有限,故在此记录下A ...

  5. appium+python+unittest+HTMLRunner编写UI自动化测试集

    简介 获取AppPackage和AppActivity 定位UI控件的工具 脚本结构 PageObject分层管理 HTMLTestRunner生成测试报告 启动appium server服务 以py ...

  6. 如何快速搭建基于python+appium的自动化测试环境

    首先申明本文是基本于Python与Android来快速搭建Appium自动化测试环境: 主要分为以下几个步骤: 前提条件: 1)安装与配置python环境,打开 Python官网,找到“Downloa ...

  7. appium+python 【Mac】UI自动化测试封装框架介绍 <五>---脚本编写(多设备)

    目的: 通过添加设备号,则自动给添加的设备分配端口,启动对应的appium服务.注意:为了方便,将共用一个配置文件. 1.公共的配置文件名称:desired_caps.yaml platformVer ...

  8. appium+python 【Mac】UI自动化测试封装框架介绍 <二>---脚本编写(单设备)

    1.单设备的执行很简单,平时可多见的是直接在config中进行配置并进行运行即可.如下: # coding=UTF- ''' Created on // @author: SYW ''' from T ...

  9. appium+python 【Mac】UI自动化测试封装框架介绍 <四>---脚本的调试

    优秀的脚本调试定位问题具备的特点: 1.方便调试. 2.运行报错后容易定位出现的问题. 3.日志的记录清晰 4.日志可被存储,一般测试结果的分析在测试之后会进行,那么日志的存储将会为后期的分析问题带来 ...

随机推荐

  1. python note 01 计算机基础与变量

    1.计算机基础. 2.python历史. 宏观上:python2 与 python3 区别: python2 源码不标准,混乱,重复代码太多, python3 统一 标准,去除重复代码. 3.pyth ...

  2. 震惊!90%的程序员不知道的Java知识!

    震惊!90%的程序员不知道的Java知识! 初学Java的时候都会接触的代码 public static void main(String[] args){ ... } 当时就像背公式一样把这行代码给 ...

  3. BZOJ2212或洛谷3521 [POI2011]ROT-Tree Rotations

    BZOJ原题链接 洛谷原题链接 线段树合并裸题. 因为交换子树只会对子树内部的逆序对产生影响,所以我们计算交换前的逆序对个数和交换后的个数,取\(\min\)即可. 对每个叶子节点建一棵动态开点线段树 ...

  4. spring rabbitmq集成

    pom.xml加 <dependency> <groupId>com.rabbitmq</groupId> <artifactId>amqp-clien ...

  5. Hbase 性能改进

    第一种性能改进方式:

  6. idea配置servlet记录,tmocat当服务器,学习

    没整理图片,将就看吧, Mac10.11.6 idea2018.1.3 servlet+tmocat9 遇到问题: 端口错误 java.rmi.server.ExportException: Port ...

  7. ABP框架系列之二十六:(EventBus-Domain-Events-领域事件)

    In C#, a class can define own events and other classes can register it to be notified when something ...

  8. poj 3279(暴力)

    题意:有一个n*m的格子,每个格子都有黑白两面(0表示白色,1表示黑色).我们需要把所有的格子都反转成黑色,每反转一个格子,它上下左右的格子都会跟着反转.请求出用最小步数完成反转时每个格子反转的次数. ...

  9. 下划线字符串camel

    const camel = (str) => { let slices = str.split('_'); let result = []; for(let i = 1, len = slice ...

  10. a标签使用href=”javascript:void(0); 在火狐浏览器跟chrome 不兼容

    使用如下方式的链接.在Chrome中点击后行为符合预期,但在IE下会新开标签卡(根据参考资料,Firefox中有相同问题).<a href=”javascript:void(0);” targe ...