python appium自动化测试框架_搭建UI自动化测试框架(Appium)

对测试人员来说,总有很多重复的手工测试工作,枯燥无味且没有成就感。要是能用机器来代替部分重复劳动,解放双手去做别的重要的事情该多好。

最接近模拟手工操作的,是UI自动化测试。虽然不如接口测试那么稳定,不如单元测试那么精准。但也能解放下双手,提高效率。

对测试人员来说,很多回归测试,并不能发现啥问题,但不去测,又不放心。万一出现bug了呢?

很多人会怀疑UI自动化测试的成果。你写了那么久,跑起来也没见找到bug。

不能单盯着bug数量的多少,对质量保障来说,没有发现bug也是个成果。

测试了,但是没有bug,不能说明没有成绩,说明质量是有保证的。

这些任务可以交给UI自动化测试去完成。跑的次数越多,节约的人工成本越多。

好了,言归正传,我们来谈谈UI automation 框架的搭建。

搭建框架前,我们先考虑几个问题:产品特性:产品是否是框架式的,几个产品是否复用一套框架?(一般内容为主的产品,都采用框架式的编码,app只是个载体,如果这样,几个产品可以共用一套测试框架)

可行性: 产品是否长期迭代?是否已经稳定?对于短期项目,写脚本没什么意义。可能你脚本还没写完,产品都退市了,一点意义都没有。

如果产品还不稳定,不停地改结构,那样脚本维护成本也很大,也没啥意义。

产品是否适合用UI automation来跑?原生比例占多大?元素是否好定位?

如果可行性不考虑清楚,后面的风险就比较大,脚本设计和维护的成本也比较高。

复用性:如果有个新的项目,你的框架小改是不是也能用在新项目上?如果一个框架都复用性很差,那么它是失败的。

组织结构:Case 如何组织? 如何展示报告?异常处理怎么处理等等,都是心里要有数的。

扩展性: 是否兼容 Android, IOS? phone, tablet? 是否可以多机一起跑?是否可以监控性能?

资源:包括时间资源,人手,公司的支持度。还有检查多少功能点?写多大规模?啥时候写?都要考虑清楚。

本人就以Appium为例,结合自己的实践,谈谈mobile的UI automation框架搭建。

先普及下基础知识

现在appium 用的是 appium desktop

设置好参数

就可以定位元素

如果元素没有id,name,class等基本定位方式的,只好用xpath(万不得已用)

现在谈谈appium里面的几个角色和关系:

Devices 和sever, driver 是一对一的关系,有几个devices,就要起几个server, driver

Build 和 driver, device 是一对多的关系, 一个build 和一个device 组合成一个driver, 可以跑在多台devices上。

Server 和 driver 是一对一关系,通过port, bp来映射关系和通行。

Device 和 case是多对多关系,为了简单,我们把case放suite 里面,组成一对一关系。

从上可以看出,Devices是关键,可以把参数都绑定在device上。

代码可以这么写:

def get_devices_version(device):

cmd = "adb -s {} shell getprop ro.build.version.release".format(device)

result = run_command_on_shell(cmd)[0]

return result

def get_devices_name(device):

cmd = "adb -s {} shell getprop ro.product.model".format(device)

result = run_command_on_shell(cmd)[0]

return result

def list_devices():

cmd = "adb devices"

result = run_command_on_shell(cmd)

print(result)

return result

def get_devices_info():

current = list_devices()

devices = []

j = 0

port= 4723

bootstrap = 5000

for i in current[1:]:

if i != "":

each_device = {}

nPos = i.index("\t")

dev = i[:nPos]

each_device["id"] = dev

each_device["version"] = get_devices_version(dev)

each_device["name"] = get_devices_name(dev)

each_device["port"]= port + j

each_device["bootstrap"] = bootstrap + j

each_device["username"]= YAML().get_users()[j]

devices.append(each_device)

j = j + 1

return devices

不管接入多少台设备,都能获取。(当然是同一平台。不能IOS,Android混插,phone,tablet混合)

看看server,可以这么写:

CMD = 'appium -a {} -p {} --bootstrap-port {} --session-override --command-timeout 600 -U {} >{} '

def close_appium_server():

kill_progress_by_name("node")

def start_appium_server(device):

host = "0.0.0.0"

port = device['port']

bootstrap_port = device['bootstrap']

udid = device['id']

appium_log = log_dir + "/" + "server.log"

cmd = CMD.format(host,port,bootstrap_port,udid,appium_log)

run_command_on_shell(cmd)

现在开始组合driver了。

def Base(device):

capabilities = YAML().get_appium_config()

if PLATFORM == 'Android':

capabilities['app'] = AppPath.get_app_filename(build_path)

capabilities['platformVersion'] = device["version"]

capabilities['deviceName'] = device["name"]

capabilities['udid'] = device["id"]

driver = webdriver.Remote('http://localhost:{}/wd/hub'.format(device['port']), capabilities)

Case这块,你想怎么组织,就怎么组织了,个人推荐用pageobject模式。

好了,大功告成,调用起来试试。

if __name__ == '__main__':

close_appium_server()

check_folder(log_dir)

jenkins = Jenkins(build_path)

jenkins.download_build()

get_devices_info()

if get_devices_info():

pool = Pool(len(get_devices_info()))

pool.map(start_appium_server,get_devices_info())

pool.close()

pool.join()

pool2 = Pool(len(get_devices_info()))

pool2.map(login,get_devices_info())

pool2.close()

pool2.join()

大体就是这样子。UI测试就是不大稳定,尤其是xpath用得比较多的时候,还有就是系统的各种弹出框的处理等。都是比较棘手的。

作者简介:

Snake, 人称安蜀黍,专职软件测试10几年,测试界的老司机。

更多精彩,请关注微信公众号:python爱好部落

你可能感兴趣的:(python,appium自动化测试框架)