有测试问题请微信联系作者,备注来意(点击此处添加)
240
一名普通的测试打工人;专注自动化测试技术研究、实践、总结、分享、交流。
用我多年的经历,给大家带来更多实用的干货。
人若有志,就不会在半坡停止。

【APP自动化基础】Appium自动化

 


Appium架构原理


Appium通过WebDriver协议与Android和iOS设备‌进行通信。(WebDriver协议支持跨平台操作,并且支持多种编程语言。)

  • Android端,appium基于WebDriver协议,利用bootstrap.jar开启服务,最后通过调⽤UiAutomator2框架,实现App的自动化操作。
  • IOS端,appium基于WebDriver协议,通过WebDriverAgent(WDA)框架,在Client端创建了一个Server,并在手机端安装了一个名为WebDriverAgentRunner的应用。这个应用会接收来自Server的指令,并连接底层的XCTest.framework,然后让XCTest.framework调用苹果的API来操作手机进行自动化。

WebDriverAgent是Facebook开发的一个iOS自动化测试工具。
IOS端早期版本是利用instruments.js运行instruments.app来注入bootstrap.js,并通过XCUItest框架(再早之前是UiAutomator框架),实现自动化操作的;ios9.3版本后就被WebDriverAgent(WDA)取代了。
Bootstrap的主要功能是在安卓目标测试机器上开启一个socket服务器,监听端口4724,将Appium从PC端发送的命令转发给UIAutomator进行处理‌。
UiAutomator2‌UiAutomator升级版,增加了对Web应用的支持和其他新功能。(Android 5.0开始引入UiAutomator2;之前版本引用UiAutomator)
UiAutomator测试框架是Android SDK自带的App UI自动化测试Java库。
XCUItest 是苹果提供的用于 iOS 应用 UI 测试的框架。
Appium在驱动iOS时,之前版本是通过UIAutomation,然后更新为XCUITest。‌
在iOS 9.2及更低版本中,苹果提供的唯一自动化技术是UIAutomation,它运行在“Instruments”中。
从iOS 10开始,苹果完全删除了UIAutomation工具,因此Appium无法按照以前的方式进行测试。为了替代UIAutomation,苹果推出了一款名为XCUITest的新型自动化技术,从iOS 9.3到iOS 10及以上版本,XCUITest成为苹果唯一支持的自动化框架‌。
Appium从1.6版本开始支持XCUITest,尽管XCUITest的功能与UIAutomation相似,但在使用过程中仍存在一些差异。例如,XCUITest为UI元素提供了不同的类名,如XCUIElementTypeButton代替了UIAButton。此外,XCUITest的页面源码输出与基于UIAutomation的结果也有显著不同‌。
为了支持XCUITest,Appium在iOS 9.3之后全面采用了WebDriverAgent方案,这是一个由Facebook开发的iOS自动化测试工具,能够支持单台Mac对应多个设备‌。
Appium 1.6.3及之后的版本添加了WebDriverAgent模块
WebDriverAgentRunner实际上集成了XCUItest的功能,使Appium能够利用XCUItest框架进行iOS设备的自动化测试

环境搭建

【自动化基础】搭建APP的UI自动化环境

获取app的Activity和Package的五种方式

方式一:通过dumpsys window获取

# 获取当前页面的Package和Activity
adb shell dumpsys window w | findstr \/ | findstr name=
# 或者:
adb shell dumpsys window | findstr mCurrentFocus
adb shell dumpsys window | grep mCu
# Mac端,获取当前页面的Package和Activity
adb shell dumpsys window w |grep / |grep name=
adb shell dumpsys window | grep mCurrentFocus
# 获取模拟器正在运行的APP的Package和Activity
adb shell dumpsys activity | find "mFocusedActivity"
adb shell dumpsys window windows | findstr mFocusedApp
# 获取真机正在运行的APP的Package和Activity---适用于Android 8.0以上
adb shell dumpsys activity | find "mResumedActivity"
#获取带端口号的Activity
adb shell dumpsys activity top | findstr ACTIVITY

方式二:列出模拟器或真机中所有软件的Package

# 获取设备的所有apk对应的包名和路径
adb shell pm list package -f
# 获取第三方apk的包
adb shell pm list package -3 -f

方式三:logcat日志抓START

adb shell logcat | grep START

方式四:logcat日志抓ActivityManager(实时查看)

adb shell logcat | grep ActivityManager
adb logcat | grep ActivityManager
adb logcat | grep -i ActivityManager.*Displayed

方式五:通过aapt工具获取

aapt dump badging C:\apk\weixin01.apk

元素定位获取工具

  • uiautomatorviewer
  • Appium Inspector
  • weditor

【APP自动化测试】APP元素定位工具

ADB启动app

adb shell am start com.tencent.tmgp.WePop/.PermissionActivity
adb shell am start -W -n com.tencent.tmgp.WePop/com.tencent.tmgp.WePop.PermissionActivity -S
-W:等待启动完成。
-n component 指定带有软件包名称前缀的组件名称以创建显式 intent,如 com.example.app/.ExampleActivity。
-S:在启动 Activity 前,强行停止目标应用。

具体am参数解释参考:https://blog.csdn.net/weixin_44380181/article/details/129698064

UI自动化项目Demo

import time
from appium import webdriver
from selenium.webdriver.support.wait import WebDriverWait
from selenium.webdriver.common.by import By
from appium.webdriver.common.touch_action import TouchAction
class Base:
"""基类公共方法封装"""
def __init__(self, driver):
"""初始化"""
self.driver = driver
def base_find(self, loc, timeout=5, poll_frequency=0.5):
"""获取查找元素"""
# 显示等待
return WebDriverWait(self.driver,
timeout=timeout,
poll_frequency=poll_frequency) \
.until(lambda x: x.find_element(*loc))
def base_input(self, loc, value):
"""输入操作"""
# 获取元素
el = self.base_find(loc)
# 清空(防止缓存数据)
el.clear()
# 输入
el.send_keys(value)
def base_click(self, loc):
"""点击操作"""
self.base_find(loc).click()
def base_get(self, loc):
"""获取文本"""
return self.base_find(loc).text
def base_xy_click(self, x, y):
"""根据xy轴位置进行点击"""
return TouchAction(self.driver).press(x=x, y=y).release().perform()
def base_switch_handle(self):
"""窗口句柄切换"""
handle = self.driver.current_window_handle # 当前窗体句柄
print("当前窗体句柄:", handle)
handles = self.driver.window_handles # 所有窗体句柄
print("窗体句柄", handles) # 获取浏览器所有窗体的句柄
self.driver.switch_to.window(handles[-1]) # 切换至最新窗口句柄
class PaoPao:
# 1.初始化脚本
def __init__(self):
desired_caps = {
'platformName': 'Android', # 平台系统
'deviceName': 'emulator-5554', # 设备名称
'platformVersion': '7.1.2', # 系统版本
'appPackage': 'com.xueqiu.android', # 被测应用包
'appActivity': '.view.WelcomeActivityAlias', # 要打开的应用活动
# 'udid': '750BBKL22GDN' # 苹果配置
# 'app': 'xueqiu.apk', # 安装应用包
# 'autoLaunch': False, #是否让Appium自动安装和启动应用,默认为True
# 'automationName': "uiautomator1", # 自动化测试框架 (1.4以上的appium不用写)
'autoGrantPermissions': "true", # 默认允许app获取相关权限
'noReset': 'True', # 每次非首次启动(冷启动)可以记住登录
'unicodeKeyboard': 'true', # 此两行是为了解决字符输入不正确的问题
'resetKeyboard': 'true', # 运行完成后重置软键盘的状态
}
self.driver = webdriver.Remote("http://localhost:4723/wd/hub", desired_caps)
self.base = Base(self.driver)
# 2.执行脚本编写
def run(self):
# 判断登录状态
self.base.base_click((By.ID, "dialog_confirm_btn"))
print("tag")
time.sleep(2)
self.base.base_click((By.TAG_NAME, "同意"))
if __name__ == '__main__':
PaoPao().run()

POM设计模式

POM模式介绍

POM(page object model)页面对象模型,将每个待测页面都封装成一个page类。然后将那些繁琐的元素定位和元素操作都封装到这个类里,是一种封装思想。在自动化测试中引入POM设计模式,可以实现页面元素和测试用例的分离,能使测试代码的可读性、维护性和复用性变得更好。

POM设计思路

POM设计模式一般分为三层(分层设计模式)

  • 第一层:对selenium进行二次封装,定义一个所有页面类都继承的BasePage类,封装selenium的常用方法,如元素的定位、输入、点击等,用那些封装那些。
  • 第二层:将每个待测页面封装成一个page类,这个类中包含三个小层:
    • 表现层,页面的可见元素。
    • 操作层,对页面元素进行的操作,如输入、点击等。
    • 业务层,对页面元素操作后实现的功能,如登录、注册、支付等。
  • 第三层:使用单元测试框架对业务逻辑进行测试,并实现数据驱动(参数化)。

项目实战

【APP自动化进阶】APP自动化项目框架实战

posted @   三叔测试笔记  阅读(285)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· 25岁的心里话
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
· 按钮权限的设计及实现
返回顶部 跳转底部
点击右上角即可分享
微信分享提示