修改了 BeautifulReport源码中的一些内容<附:BeautifulReport--适用于unittest自动化测试的可视化报告>;

1.生成截图时,原来的img文件夹必须在项目根目录下,做了调整;

2.生成的截图无法在html报告上展示,老是会报路径错误,检查发现源码中在做判断时少加了system()方法才导致错误的出现;

源码github截图

修改后截图

简单的登录功能实现,懂车帝app为例;

1.页面元素封装;

page_car.py
from common.basics import Crazy

class CarLoginPage(Crazy):
"""元素定位""" # 跳过广告
skip_loc = ('id', 'com.ss.android.auto:id/c8r') def click_skip(self):
self.click(self.skip_loc) tag_loc = ("id", "com.ss.android.auto:id/bqp") # 首页文本
def text_home(self):
return self.get_texts(self.tag_loc, 0) # 是否未 登录
def text_landed(self):
return self.get_texts(self.tag_loc, 3) # 进入 未登录(我的)页面
def click_landed(self):
self.clicks(self.tag_loc, 3) # # 判断是否进入未登录(我的)页 设置 编辑资料
common_loc = ('id', 'com.ss.android.auto:id/dd') def text_common(self):
return self.get_text(self.common_loc) # 点击手机图标
phone_loc = ('id', 'com.ss.android.auto:id/brg') def click_phone(self):
self.click(self.phone_loc) # 判断是否进入手机登录页面
phone_login_loc = ('id', 'com.ss.android.auto:id/jv') def text_phone_login(self):
return self.get_text(self.phone_login_loc) # 输入手机号
_phone_loc = ('id', 'com.ss.android.auto:id/jy') def input_phone(self, phone):
self.send_keys(self._phone_loc, phone) # 点击获取验证码
verification_code_loc = ('id', 'com.ss.android.auto:id/jz') def click_verification_code(self):
self.click(self.verification_code_loc) # 输入验证码
input_verification_code_loc = ('id', 'com.ss.android.auto:id/k3') def input_verification(self, code):
self.send_keys(self.input_verification_code_loc, code) # 点击 进入app 按钮
into_btn_loc = ('id', 'com.ss.android.auto:id/k7') def click_into_btn(self):
self.click(self.into_btn_loc) # QQ登录
qq_loc = ('id', 'com.ss.android.auto:id/brh') def click_qq(self):
self.click(self.qq_loc) # 授权按钮
# grant_btn_loc = ('id', 'com.tencent.mobileqq:id/name') [36,1418][1044,1538]
# grant_btn_loc = [(36, 1301), (1044, 1421)]
grant_btn_loc = [(36, 1418), (1044, 1538)] def click_grant_btn(self):
self.click_coordinate(self.grant_btn_loc) # 判断登录成功
personal_center_loc = ('id', 'com.ss.android.auto:id/br3') def text_personal_center(self):
return self.get_text(self.personal_center_loc) # 微信登录
wechat_loc = ('id', 'com.ss.android.auto:id/bri') def click_wechat(self):
self.click(self.wechat_loc) # 确认登录 [42,1176][1038,1320] com.tencent.mm:id/ch6 [39,1079][1041,1211] [42,1176][1038,1320]
confirm_wechat_loc = [(42, 1176), (1038, 1320)] def click_confirm_wechat(self):
self.click_coordinate(self.confirm_wechat_loc) # 退出操作
set_up_loc = ('id', 'com.ss.android.auto:id/bvb') def click_set_up(self):
self.click(self.set_up_loc) # 判断进入设置页 text_common # 退出按钮
logout_loc = ('id', 'com.ss.android.auto:id/c7o') def click_logout(self):
self.click(self.logout_loc) # 是否存在退出按钮
def element_logout(self):
return self.find_element(self.logout_loc) # 退出确认
logout_confirm_loc = ('id', 'com.ss.android.auto:id/i3') def text_logout_confirm(self):
return self.get_text(self.logout_confirm_loc) # 确认退出
confirm_logout_loc = ('id', 'com.ss.android.auto:id/c9_') def click_confirm_logout(self):
self.click(self.confirm_logout_loc) # 修改资料
update_user_loc = ('id', 'com.ss.android.auto:id/br5') def click_update_user(self):
self.click(self.update_user_loc) # 用户名
user_name_loc = ('id', 'com.ss.android.auto:id/j_') def click_user_name(self):
self.click(self.user_name_loc) # 请输入用户名 title
enter_user_loc = ('id', 'com.ss.android.auto:id/ali') def text_enter_user(self):
return self.get_text(self.enter_user_loc) # 修改用户名
input_user_loc = ('id', 'com.ss.android.auto:id/alj') def input_user(self, name):
self.send_keys(self.input_user_loc, name) # 确定按钮
enter_btn_loc = ('id', 'com.ss.android.auto:id/alm') def click_enter_btn(self):
self.click(self.enter_btn_loc) # 获取用户名称
get_user_name_loc = ('id', 'com.ss.android.auto:id/jc') def text_get_user_name(self):
return self.get_text(self.get_user_name_loc)
basics.py 是用显示等待封装的一些常用方法<附:基于显示等待封装的一些常用方法>;在做授权操作时,需要使用坐标定位的方法实现点击操作;剩下的就是页面主要元素的函数;

2.测试用例;

test_car.py

import unittest
import time
from BeautifulReport import BeautifulReport
from page.page_car import CarLoginPage
from common.logger import Log
from common.basics import open_app
from common import readConfig class TestCar(unittest.TestCase):
@classmethod
def setUpClass(cls):
cls.driver = open_app()
cls.log = Log()
cls.car = CarLoginPage(cls.driver)
# cls.img_path = os.path.join(os.path.dirname(os.path.dirname(os.path.realpath(__file__))), 'report\img')
cls.img_path = readConfig.img_path # 必须是这个路径 @classmethod
def tearDownClass(cls):
cls.driver.close_app() def save_img(self, img_name):
self.driver.get_screenshot_as_file('{}/{}.png'.format(self.img_path, img_name)) @BeautifulReport.add_test_img('test_1app')
def test_1app(self):
"""打开app测试用例"""
car = self.car
# car.click_skip() # 跳过广告
self.assertEqual('首页', car.text_home(), '进入app出错!')
self.log.info('进入app成功!') @BeautifulReport.add_test_img('QQ登录')
def test_2qq(self):
"""QQ登录测试用例"""
car = self.car
if car.text_landed() == '我的':
car.click_landed()
self.sign_out()
if car.text_landed() == '未登录':
self.log.info('用户未登录,准备 QQ 用户登录中...')
car.click_landed()
self.assertEqual(car.text_common(), '常用功能', '未进入 未登录 页面!')
self.log.info('选择 QQ 用户登录.')
car.click_qq()
time.sleep(2)
car.click_grant_btn()
time.sleep(2)
self.assertEqual(car.text_personal_center(), '个人主页', 'QQ 用户登录失败!')
self.log.info('QQ 用户登录成功!')
self.save_img('QQ登录') @BeautifulReport.add_test_img('微信登录')
def test_2wechat(self):
"""微信登录测试用例"""
car = self.car
if car.text_landed() == '我的':
car.click_landed()
self.sign_out()
if car.text_landed() == '未登录':
self.log.info('用户未登录,准备 微信 用户登录中...')
car.click_landed()
self.assertEqual(car.text_common(), '常用功能', '未进入 未登录<我的> 页面!')
self.log.info('选择 微信 用户登录.')
car.click_wechat()
time.sleep(2)
if car.text_personal_center() == '个人主页':
self.log.info('微信 用户登录成功!')
else:
car.click_confirm_wechat()
time.sleep(2)
self.assertEqual(car.text_personal_center(), '个人主页', '微信 用户登录失败!')
self.log.info('微信 用户登录成功1!')
self.save_img('微信登录') @BeautifulReport.add_test_img('编辑资料')
def test_4update_user_info(self):
"""编辑用户资料"""
car = self.car
if car.text_landed() == '未登录':
self.log.warning('未检测到用户登录,不能进行编辑资料操作!')
elif car.text_landed() == '我的':
self.log.info('用户已登录,准备进行 编辑资料 操作.')
car.click_landed()
car.click_update_user()
self.assertEqual(car.text_common(), '编辑资料', '出现错误,没有进入 编辑资料 页面!')
start_user = car.text_get_user_name()
car.click_user_name()
car.input_user('')
car.click_enter_btn()
end_user = car.text_get_user_name()
self.save_img('编辑资料')
time.sleep(2)
if start_user == end_user:
self.log.info('用户名称未做修改.')
else:
self.log.info('修改用户名称成功!修改后的名称:{}'.format(end_user)) # def test_3phone(self):
# """手机登录"""
# car = self.car
# if car.text_landed() == '我的':
# car.click_landed()
# self.sign_out()
# if car.text_landed() == '未登录':
# self.log.info('用户未登录,准备 手机 用户登录中...')
# car.click_landed()
# self.assertEqual(car.text_common(), '常用功能', '未进入 未登录<我的> 页面!')
# self.log.info('选择 手机 用户登录.')
# car.click_phone()
# self.assertEqual(car.text_phone_login(), '手机快捷登录', '进入 手机快捷登录 页面失败!')
# car.input_phone('xxxxxxxxxxx')
# time.sleep(1)
# car.click_verification_code()
# time.sleep(2)
# car.input_verification('5556')
# car.click_into_btn()
# self.assertEqual(car.text_personal_center(), '个人主页', '手机 用户登录失败!')
# self.log.info('手机 用户登录成功!') @BeautifulReport.add_test_img('退出登录')
def sign_out(self):
"""用户退出"""
car = self.car
if car.text_common() == '常用功能':
self.assertEqual(car.text_personal_center(), '个人主页', '没有用户登录,无法退出!')
self.log.info('正在执行用户退出登录操作.')
car.click_set_up()
self.assertEqual(car.text_common(), '设置', '进入设置页面失败!')
self.log.info('进入设置页成功!')
car.click_logout()
self.assertEqual(car.text_logout_confirm(), '退出确认', '出现错误,没有发现退出确认弹框!')
car.click_confirm_logout()
if not car.element_logout():
self.log.info('退出登录成功!')
self.save_img('退出登录')
car.back()
else:
self.log.info('当前不在 我的 页面,无法进行退出登录操作!') if __name__ == '__main__':
unittest.main()
CarLoginPage,元素类;Log方法,打印日志<附:python logging模块 输出日志和过期清理>;open_app方法,连接appium,启动被测app;readConfig方法,配置文件;
手机登录的就比较蛋疼了,试了根据adb logcat获取手机日志,提取验证码没有成功...


生成的截图和html报告:
1.每条用例都会生成一张截图,错误截图另算;

2.html报告;

错误case截图

正常case截图

												

BeautifulReport 实现app UI自动化测试的更多相关文章

  1. APP UI自动化测试思路总结

    python+appium自动化测试系列就要告一段落了,本篇博客咱们做个小结. 首先想要说明一下,APP自动化测试可能很多公司不用,但也是大部分自动化测试工程师.高级测试工程师岗位招聘信息上要求的,所 ...

  2. Appium App UI 自动化测试理论知识

    (一)App自动化测试背景 随着移动终端的普及,手机应用越来越多,也越来越重要.App的回归测试用例数量越来越多,全量回归也越来越消耗时间.另外移动端碎片化严重(碎片化:兼容性测试,手机品牌多样.An ...

  3. ios app UI自动化测试用到的命令

    ios测试的app测试包,真机设备需要开发者证书并且将测试机的udid加入到pp文件文件,configruation 要求为debug模式的ipa包, 1.苹果手机的UDID, a.通过 xcode- ...

  4. APP的UI自动化测试框架及平台化探索

    顾铮,10年+测试及测试开发相关经验,2014年加入京东,曾主导设计开发UI测试框架,参与CI测试平台建设,现负责iOS侧的工具,框架建设.在UI自动化,性能测试,单元测试方面有较深入研究,在App, ...

  5. UI自动化测试:App的WebView页面中,当搜索栏无搜索按钮时处理方法

    一.遇到的问题 在做移动端的UI自动化测试时,经常会遇到上图所示的搜索框,这里有个麻烦就是搜索框没有"搜索"按钮,UI自动化测试时不能确认搜索. 要解决这个问题,我们可以通过 dr ...

  6. 从0到1搭建移动App功能自动化测试平台(2):操作iOS应用的控件

    转自:http://debugtalk.com/post/build-app-automated-test-platform-from-0-to-1-Appium-interrogate-iOS-UI ...

  7. UI自动化测试的那些事

    互联网产品的迭代速度远高于传统软件,尤其是移动APP不但更新频繁,还需要在不同硬件.系统版本的环境下进行大量兼容测试,这就给传统测试方法和测试工具带来了巨大挑战.为满足产品敏捷开发.快速迭代的需求,自 ...

  8. 从0到1搭建移动App功能自动化测试平台(0):背景介绍和平台规划

    本文作者: 伯乐在线 - debugtalk .未经作者许可,禁止转载!欢迎加入伯乐在线 专栏作者. 转载地址:http://blog.jobbole.com/101221/ 背景 最近新加入DJI的 ...

  9. <自动化测试方案_7>第七章、PC端UI自动化测试

    第七章.PC端UI自动化测试 UI自动化测试又分为:Web自动化测试,App自动化测试.微信小程序.微信公众号UI层的自动化测试工具非常多,比较主流的是UFT(QTP),Robot Framework ...

随机推荐

  1. vue中,对象数组多层嵌套时,更新数据更新页面

    vue中的对象和数组的元素直接赋值修改时,是不能响应到view中去的 1.对象更新 this.a={title:'列表1’}; this.a.title='列表2’; <h1>{{a.ti ...

  2. 百度分享到修改url

    <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...

  3. JavaScript 常见错误

    1. 严格缩进 JavaScript 会自动添加句末的分号,导致一些难以察觉的错误 return { key: value }; // 相当于 return; { key: value }; 2. 括 ...

  4. 微信小程序底部tabbar

    在 app.json    文件里面 : { "pages":[ "pages/index/index", "pages/logs/logs" ...

  5. 梦里寻她千百度,Bug却在隔壁老张处

    程序员与 Bug 是一对矛盾的存在,程序员既要在解决 Bug 中获得成就感,同时也讨厌 Bug 本身的存在.“程序不息,Bug 不止”,程序员在与 Bug 的斗争中,也有很多有趣的事情发生,我们整理了 ...

  6. Netty入门(一):零基础“HelloWorld”详细图文步骤

    因为接下来的项目要用到netty,所以就了解一下这个程序,奈何网上的教程都是稍微有点基础的,所以,就写一篇对于netty零基础的,顺便也记录一下. 先扔几个参考学习的网页: netty 官方API:  ...

  7. Cassandra最小化安装

    Cassandra最小化安装 环境 CentOS 7.2 64位 IP_address:172.27.0.8 安装包装备 [root@master ~]# ll /usr/local/src tota ...

  8. in与exists和not in 与 not exists的区别

    1.in 与 exists: 外表大,用IN:内表大,用EXISTS: 原理: 用in:外表使用了索引,直接作hash连接: 用exists:内表使用了索引,外表作loop循环再进行匹配: 2.not ...

  9. java程序如何正确的打日志

    什么是日志 简单的说,日志就是记录程序的运行轨迹,方便查找关键信息,也方便快速定位解决问题. 我们 Java 程序员在开发项目时都是依赖 Eclipse/ Idea 等开发工具的 Debug 调试功能 ...

  10. Redis的数据结构之Set

    存储Set 和List类型不同的是,Set集合中不允许出现重复的元素 Set可包含的最大元素数量是4294967295 存储set常用命令: 添加/删除元素 获取集合中的元素 集合中的差集运算 集合中 ...