UIAutomator2 常用命令整理

https://github.com/openatx/uiautomator2

平时用的最多也最顺手的一个Android系统APP控制Python库,它不仅可以帮我在工作上实现基于Android车载娱乐系统的自动化操控和测试,也可以在双十一为你自动开抢心仪的商品。以下对常用功能进行了整理和总结。

连接ADB设备:

可以通过USB或Wifi与ADB设备进行连接,进而调用Uiautomator2框架,支持同时连接单个或多个ADB设备。

  1. USB连接:只有一个设备也可以省略参数,多个设备则需要序列号来区分
import uiautomator2 as u2


d = u2.connect("--serial-here--") 
  1. USB连接:一个设备时,可简写
d = u2.connect() 
  1. 无线连接:通过设备的IP连接(需要在同一局域网且设备上的atx-agent已经安装并启动)
d = u2.connect("10.1.2.3") 
  1. 无线连接:通过ABD wifi 等同于下面的代码
d = u2.connect_adb_wifi("10.0.0.1:5555") 
#等同于
 + Shell: adb connect 10.0.0.1:5555
 + Python: u2.connect_usb("10.0.0.1:5555")




APP操作:

用于启动或停止某个APP

  1. 获取前台应用 packageName, activity
d.app_current() 

2.1 启动应用( 默认的这种方法是先通过atx-agent解析apk包的mainActivity,然后调用am start -n $package/$activity启动)

d.app_start("com.example.app") 

2.2. 通过指定main activity的方式启动应用,等价于调用am start -n com.example.hello_world/.MainActivity

d.app_start("com.example.hello_world", ".MainActivity")

2.3. 使用 monkey -p com.example.hello_world -c android.intent.category.LAUNCHER 1 启动,这种方法有个副作用,它自动会将手机的旋转锁定给关掉

d.app_start("com.example.hello_world", use_monkey=True)
  1. 启动应用前停止此应用
d.app_start("com.example.app", stop=True) 

4.1 停止应用, 等价于am force-stop,此方法会丢失应用数据

d.app_stop("com.example.app")

4.2 停止应用, 等价于pm clear

d.app_clear('com.example.hello_world')

4.3 停止所有应用

d.app_stop_all()

4.4 停止所有应用,除了某个应用

d.app_stop_all(excludes=['com.examples.demo'])
  1. 得到APP图标
img = d.app_icon("com.examples.demo")
img.save("icon.png")
  1. 列出所有运行中的应用
d.app_list_running()
  1. 确定APP是否启动,也可以通过Session来判断
pid = d.app_wait("com.example.android") # 等待应用运行, return pid(int)
if not pid:
    print("com.example.android is not running")
else:
    print("com.example.android pid is %d" % pid)

d.app_wait("com.example.android", front=True) # 等待应用前台运行
d.app_wait("com.example.android", timeout=20.0) # 最长等待时间20s(默认)

or
d.wait_activity(".ApiDemos", timeout=10) # default timeout 10.0 seconds




Session操作

一般用于测试某个特定的APP,首先将某个APP设定为一个Session,所有的操作都基于此Session,当Session退出时,代表APP退出。

  1. 启动应用并获取session
    session的用途是操作的同时监控应用是否闪退,当闪退时操作,会抛出SessionBrokenError
sess = d.session("com.example.app") # start app
  1. 停止或重启session,即app
sess.close() # 停止app
sess.restart() # 冷启app
  1. python with 功能,开启某个APP执行某个操作后,自动退出某个session
with d.session("com.netease.cloudmusic") as sess:
    sess(text="Play").click()

4.1 当APP已运行时自动跳过启动

# launch app if not running, skip launch if already running
sess = d.session("com.netease.cloudmusic", attach=True)

4.2 当某个APP没有启动时,报错

# raise SessionBrokenError if not running
sess = d.session("com.netease.cloudmusic", attach=True, strict=True)

5.1 确定session对应的APP是否运行

# check if session is ok.
# Warning: function name may change in the future
sess.running() # True or False

5.2 确定session对应的APP是否运行,当不在运行将报错

# When app is still running
sess(text="Music").click() # operation goes normal

# If app crash or quit
sess(text="Music").click() # raise SessionBrokenError
# other function calls under session will raise SessionBrokenError too




截图与hierarchy提?。?/h2>

用于获取Android当前的截图和界面元素。

  1. 截图
# take screenshot and save to a file on the computer, require Android>=4.2.
d.screenshot("home.jpg")

# get PIL.Image formatted images. Naturally, you need pillow installed first
image = d.screenshot() # default format="pillow"
image.save("home.jpg") # or home.png. Currently, only png and jpg are supported

# get opencv formatted images. Naturally, you need numpy and cv2 installed first
import cv2
image = d.screenshot(format='opencv')
cv2.imwrite('home.jpg', image)

# get raw jpeg data
imagebin = d.screenshot(format='raw')
open("some.jpg", "wb").write(imagebin)
  1. 获取hierarchy
# get the UI hierarchy dump content (unicoded).
xml = d.dump_hierarchy()




模拟触控操作:

用于模拟用户对手机的点击或滑动等操作

1.1 XY坐标点击

d.click(10, 20) 

1.2 XY坐标双击

d.double_click(x, y)
d.double_click(x, y, 0.1) # default duration between two click is 0.1s

1.3 长按某个坐标

d.long_click(x, y)
d.long_click(x, y, 0.5) # long click 0.5s (default)
  1. 通过元素中的Text信息来点击,程序会点击Text所在layout的中心位置,

# click on the center of the specific ui object
d(text="Settings").click()
d(Text="Settings").double_click() 
d(Text="Settings").long_click()

# wait element to appear for at most 10 seconds and then click
d(text="Settings").click(timeout=10)

# click with offset(x_offset, y_offset)
# click_x = x_offset * width + x_left_top
# click_y = y_offset * height + y_left_top
d(text="Settings").click(offset=(0.5, 0.5)) # Default center
d(text="Settings").click(offset=(0, 0)) # click left-top
d(text="Settings").click(offset=(1, 1)) # click right-bottom

# click when exists in 10s, default timeout 0s
clicked = d(text='Skip').click_exists(timeout=10.0)

# click until element gone, return bool
is_gone = d(text="Skip").click_gone(maxretry=10, interval=1.0) # maxretry default 10, interval default 1.0
  1. 滑动操作,从(10, 20)滑动到(80, 90)
d.swipe(10, 20, 80, 90) 
d.swipe(sx, sy, ex, ey, 0.5)

d(text="Settings").swipe("right")
d(text="Settings").swipe("left", steps=10)
d(text="Settings").swipe("up", steps=20) # 1 steps is about 5ms, so 20 steps is about 0.1s
d(text="Settings").swipe("down", steps=20)


# swipe from point(x0, y0) to point(x1, y1) then to point(x2, y2)
# time will speed 0.2s bwtween two points
d.swipe_points([(x0, y0), (x1, y1), (x2, y2)], 0.2))
  1. 整个屏幕右滑动
d.swipe_ext("right") 
  1. 屏幕右滑,滑动距离为屏幕宽度的90%
d.swipe_ext("right", scale=0.9) 
  1. 从一个坐标拖拽到另一个坐标
d.drag(sx, sy, ex, ey)
d.drag(sx, sy, ex, ey, 0.5) # swipe for 0.5s(default)
  1. 模拟按下后的连续操作,如九宫格解锁
d.touch.down(10, 10) # 模拟按下
time.sleep(.01) # down 和 move 之间的延迟,自己控制
d.touch.move(15, 15) # 模拟移动
d.touch.up() # 模拟抬起
  1. 模拟两指缩放操作
# notes : pinch can not be set until Android 4.3.
# from edge to center. here is "In" not "in"
d(text="Settings").pinch_in(percent=100, steps=10)
# from center to edge
d(text="Settings").pinch_out()

or

d().pinch_in(percent=100, steps=10)
d().pinch_out()




硬按键操作

用于模拟用户对手机硬按键或系统按键的操作。

  1. 模拟按 Home 或 Back 键
    目前支持以下关键字,但并非所有设备都支持:
    home
    back
    left
    right
    up
    down
    center
    menu
    search
    enter
    delete ( or del)
    recent (recent apps)
    volume_up
    volume_down
    volume_mute
    camera
    power
d.press("back") 
d.press("home") 
  1. 模拟按Android定义的硬键值
d.press(0x07, 0x02) 
# press keycode 0x07('0') with META ALT(0x02)
#具体可查询:
#https://developer.android.com/reference/android/view/KeyEvent.html
  1. 解锁屏幕
d.unlock()
# This is equivalent to
# 1. launch activity: com.github.uiautomator.ACTION_IDENTIFY
# 2. press the "home" key
  1. 模拟输入,需要光标已经在输入框中才可以
d.set_fastinput_ime(True) # 切换成FastInputIME输入法
d.send_keys("你好123abcEFG") # adb广播输入
d.clear_text() # 清除输入框所有内容(Require android-uiautomator.apk version >= 1.0.7)
d.set_fastinput_ime(False) # 切换成正常的输入法
d.send_action("search") # 模拟输入法的搜索
  1. 清空输入框
d.clear_text()




执行ADB shell命令

直接通过Python来执行ADB shell中的指令,并得到反馈。

  1. 执行shell命令,获取输出和exitCode
output, exit_code = d.shell("ps -A", timeout=60) 
  1. 仅得到输出
output = d.shell("pwd").output 
  1. 仅得到Exitcode
exit_code = d.shell("pwd").exit_code
  1. 推送文件到ADB设备中
# push to a folder
d.push("foo.txt", "/sdcard/")
# push and rename
d.push("foo.txt", "/sdcard/bar.txt")
# push fileobj
with open("foo.txt", 'rb') as f:
   d.push(f, "/sdcard/")
# push and change file access mode
d.push("foo.sh", "/data/local/tmp/", mode=0o755)
  1. 获取文件到本地
d.pull("/sdcard/tmp.txt", "tmp.txt")

# FileNotFoundError will raise if the file is not found on the device
d.pull("/sdcard/some-file-not-exists.txt", "tmp.txt")




元素操作或Selector

这是Uiautomator2最为关键的核心功能,测试者可以根据界面中的元素来判断当前画面是否符合预期或基于界面元素进行点按滑动等操作。

目前Uiautomator2支持以下种类的关键字参数:
text, textContains, textMatches, textStartsWith
className, classNameMatches
description, descriptionContains, descriptionMatches, descriptionStartsWith
checkable, checked, clickable, longClickable
scrollable, enabled,focusable, focused, selected
packageName, packageNameMatches
resourceId, resourceIdMatches
index, instance

举个例子,测试者可以通过以上关键字的组合,来实现特定界面元素的定位,如下面这段代码是要求UT2去点击界面中,元素text信息为clock,className为'android.widget.TextView'的元素:

# Select the object with text 'Clock' and its className is 'android.widget.TextView'
d(text='Clock', className='android.widget.TextView').click()

除了,可以使用关键字的组合来限定特定UI元素,UT2也支持通过子节点或兄弟节点来限定特定UI元素。如下面这几段代码分别是通过某个元素,获取其子元素或同胞元素中的信息或进行后续操作。

# children
# get the children or grandchildren
d(className="android.widget.ListView").child(text="Bluetooth")
# get the children or grandchildren
d(className="android.widget.ListView").child(text="Bluetooth")
# siblings
# get siblings
d(text="Google").sibling(className="android.widget.ImageView")

也可以根据子节点的Text或 Description或Instance来定位元素, 特别提下下面代码中的这个allow_scroll_search功能,它调用UT2自动滚动直到找到对应元素:

# get the child matching the condition className="android.widget.LinearLayout"
# and also its children or grandchildren with text "Bluetooth"
d(className="android.widget.ListView", resourceId="android:id/list") \
.child_by_text("Bluetooth", className="android.widget.LinearLayout")

# get children by allowing scroll search
d(className="android.widget.ListView", resourceId="android:id/list") \
.child_by_text(
   "Bluetooth",
   allow_scroll_search=True,
   className="android.widget.LinearLayout"
 )

下面有另一个实例来展示UT2的定位,以下为Android的系统设置界面及它的hierarchy:
系统设置界面:


Setting.png

hierarchy:

<node index="0" text="" resource-id="android:id/list" class="android.widget.ListView" ...>
 <node index="0" text="WIRELESS & NETWORKS" resource-id="" class="android.widget.TextView" .../>
 <node index="1" text="" resource-id="" class="android.widget.LinearLayout" ...>
   <node index="1" text="" resource-id="" class="android.widget.RelativeLayout" ...>
     <node index="0" text="Wi?Fi" resource-id="android:id/title" class="android.widget.TextView" .../>
   </node>
   <node index="2" text="ON" resource-id="com.android.settings:id/switchWidget" class="android.widget.Switch" .../>
 </node>
 ...
</node>

通过child_by_text + child组合后可以定位到WIFI的开关。

d(className="android.widget.ListView", resourceId="android:id/list") \
 .child_by_text("Wi?Fi", className="android.widget.LinearLayout") \
 .child(className="android.widget.Switch") \
 .click()

也可以通过,位置关系来定位元素:
d(A).left(B), selects B on the left side of A.
d(A).right(B), selects B on the right side of A.
d(A).up(B), selects B above A.
d(A).down(B), selects B under A.

## select "switch" on the right side of "Wi?Fi"
d(text="Wi?Fi").right(className="android.widget.Switch").click()

还有可以通过元素的instances来定位,比如一个界面中有多个switch,我们可以通过下面的形式来定位是第一个还是第二个

d(className="android.widget.Switch", instance=0)
d(className="android.widget.Switch")[0]

or


# get the count of views with text "Add new" on current screen
d(text="Add new").count

# same as count property
len(d(text="Add new"))

# get the instance via index
d(text="Add new")[0]
d(text="Add new")[1]
...

# iterator
for view in d(text="Add new"):
   view.info  # ...
  1. 等待某个元素出现
d(text="Settings").exists # True if exists, else False
d.exists(text="Settings") # alias of above property.

# advanced usage
d(text="Settings").exists(timeout=3) # wait Settings appear in 3s, same as .wait(3)

d.xpath("立即开户").wait() # 等待元素,最长等10s(默认)
d.xpath("立即开户").wait(timeout=10) # 修改默认等待时间
  1. xpath操作
    具体可以参考:https://github.com/openatx/uiautomator2/blob/master/XPATH.md
# xpath操作
d.xpath("立即开户").click() # 包含查找等待+点击操作,匹配text或者description等于立即开户的按钮
d.xpath("http://*[@text='私人FM']/../android.widget.ImageView").click()

d.xpath('//*[@text="私人FM"]').get().info # 获取控件信息

for el in d.xpath('//android.widget.EditText').all():
   print("rect:", el.rect) # output tuple: (left_x, top_y, width, height)
   print("bounds:", el.bounds) # output tuple: (left, top, right, bottom)
   print("center:", el.center())
   el.click() # click operation
   print(el.elem) # 输出lxml解析出来的Node

3.输入框的操作

d(text="Settings").get_text()  # get widget text
d(text="Settings").set_text("My text...")  # set the text
d(text="Settings").clear_text()  # clear the text
  1. 等待某个元素出现或消失
# wait until the ui object appears
d(text="Settings").wait(timeout=3.0) # return bool
# wait until the ui object gone
d(text="Settings").wait_gone(timeout=1.0)




Setting

  1. 默认控件等待时间(原生操作,xpath插件的等待时间)
d.settings['wait_timeout'] = 20.0 
or
d.implicitly_wait(20.0)
  1. 点击的等待延时
d.click_post_delay = 1.5 # default no delay
  1. 配置accessibility服务的最大空闲时间,超时将自动释放。默认3分钟。(如果两个步骤需要等待较长时间,且不希望下一次发送指令时重启UT2,则可以将此时间加大)
d.set_new_command_timeout(300)




守护

用于处理非预期的弹出框,如崩溃窗口,一些确定或取消弹出框。

  1. 监控弹窗(在线程中监控)
# 常用写法,注册匿名监控
d.watcher.when("安装").click()

# 注册名为ANR的监控,当出现ANR和Force Close时,点击Force Close
d.watcher("ANR").when(xpath="ANR").when("Force Close").click()

# 其他回调例子
d.watcher.when("抢红包").press("back")
d.watcher.when("http://*[@text = 'Out of memory']").call(lambda d: d.shell('am force-stop com.im.qq'))

# 移除ANR的监控
d.watcher.remove("ANR")

# 移除所有的监控
d.watcher.remove()

# 开始后台监控
d.watcher.start()
d.watcher.start(2.0) # 默认监控间隔2.0s

# 强制运行所有监控
d.watcher.run()

# 停止监控
d.watcher.stop()

# 停止并移除所有的监控,常用于初始化
d.watcher.reset()




插件

Performance 性能采集(记录CPU,RAM等数据)
https://github.com/openatx/uiautomator2/blob/6e0d75d778a86c626df778e0432c8e339e3d9be4/uiautomator2/ext/perf/README.md

Aircv 图像比对插件(有较多限制,推荐自己单独截图后调用Aircv来实现图像比对或点击等功能,或用ATX来实现)
https://github.com/openatx/uiautomator2/blob/436404119fafce303ad8f3a07811c044d101b9eb/uiautomator2/ext-archived/aircv/README.md

Htmlreport插件(将操作生成HTML文件)
https://github.com/openatx/uiautomator2/tree/6e0d75d778a86c626df778e0432c8e339e3d9be4/uiautomator2/ext/htmlreport

最后编辑于
?著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,992评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,212评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,535评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,197评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,310评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,383评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,409评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,191评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,621评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,910评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,084评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,763评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,403评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,083评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,318评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,946评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,967评论 2 351

推荐阅读更多精彩内容

  • 在平时的工作中,会经常用到adb命令,在这里稍微整理了一下。 一.概要 1.什么是adb? adb全称为Andro...
    坚固的浮沙阅读 783评论 0 3
  • adb 即 Android Debug Bridge 安卓调试桥,adb 是一个C/S架构的命令行工具,主要由 3...
    坚持未来阅读 2,722评论 0 2
  • Tcp/IP通信也是不安全的,在传输的时候也可能出现漏洞 查看正在运行的进程 adb shell ps -A |g...
    远远飘着云阅读 4,393评论 0 0
  • Android 调试桥 Android 调试桥 (adb) 是一个通用命令行工具,其允许您与模拟器实例或连接的 A...
    guanjm阅读 1,461评论 0 1
  • 1、日志相关: adb logcat //显示全部日志 adb logcat > c:\test.log //...
    学习不断阅读 59,065评论 0 11