App测试小工具

news2025/4/16 9:41:23

前言

最近app测试比较多,每次都得手动输入日志tag,手动安装,测完又去卸载,太麻烦。就搞了小工具使用。

效果预览

在这里插入图片描述每次测试完成,点击退出本次测试,就直接卸载了,usb插下一个手机又可以继续测了。

使用了uiautomator2 ,PySimpleGUI和 Pyinstaller打包,没什么太难的东西,,只简单使用了这些,没深入复杂的使用。

源代码分享

import subprocess
import PySimpleGUI as sg
import time
import uiautomator2 as u2


class DeviceManager:
    """设备管理器"""

    def __init__(self):
        self.device = None
        self.connect_device()

    def connect_device(self):
        """连接设备,如果失败会重试"""
        try:
            self.device = u2.connect()
            self.device.implicitly_wait(5)
            return True
        except Exception as e:
            print(f"设备连接失败: {e}")

    def ensure_connected(self):
        """确保设备已连接"""
        try:
            # 尝试执行一个简单的命令来检查连接状态
            self.device.info
            return True
        except:
            return self.connect_device()

    def install_app(self, file_path):
        """安装应用并返回包名"""
        new_package = ''
        if not self.ensure_connected():
            return None

        try:
            installed_before = set(self.device.app_list())
            self.device.app_install(file_path)
            installed_after = set(self.device.app_list())
            new_package = list(installed_after - installed_before)
            return new_package[0] if new_package else None
        except Exception as e:
            print(f'安装应用失败: {e}')
            return None
        finally:
            window.TKroot.title(f'{new_package}测试中...')

    def uninstall_app(self, package_name):
        """卸载应用"""
        if not package_name:
            return

        if not self.ensure_connected():
            return

        try:
            if package_name in self.device.app_list():
                self.device.app_uninstall(package_name)
                print(f"成功卸载应用: {package_name}")
            else:
                print(f"应用 {package_name} 未安装")
        except Exception as e:
            print(f"卸载应用失败: {e}")


class LogcatManager:
    """日志管理器"""

    def __init__(self):
        self.processes = []
        self.window_ids = []
        self.is_running = True

    def start_logcat(self, tag):
        """启动单个logcat进程"""
        while self.is_running:
            try:
                apple_script = f'''
                tell application "Terminal"
                    set newWindow to do script "adb logcat -s {tag}"
                    activate
                    id of window 1
                end tell
                '''
                window_id = subprocess.check_output(['osascript', '-e', apple_script]).decode().strip()
                self.window_ids.append(window_id)

                # 检查logcat进程是否存在
                while self.is_running:
                    result = subprocess.run(['pgrep', '-f', f'adb logcat -s {tag}'], capture_output=True)
                    if result.returncode != 0:
                        print(f"日志进程 {tag} 已断开,正在重连...")
                        break
                    time.sleep(5)

            except Exception as e:
                print(f"启动日志进程失败: {e}, 正在重试...")

            if not self.is_running:
                break
            time.sleep(3)

    def stop_all(self):
        """停止所有日志进程"""
        self.is_running = False
        # 关闭所有Terminal窗口
        for window_id in self.window_ids:
            close_script = f'''
            tell application "Terminal"
                repeat with w in windows
                    if id of w is {window_id} then
                        close w
                    end if
                end repeat
            end tell
            '''
            subprocess.run(['osascript', '-e', close_script])
        self.window_ids = []


d = DeviceManager()
logcat_mgr = LogcatManager()

# 定义布局
layout = [
    [sg.Text('选择apk文件')],
    [sg.InputText(key='-FILE-'), sg.FileBrowse(button_text='选择apk文件')],
    [sg.Text('需要监控的日志,以英文逗号分割(tag1,tag2)')],
    [sg.InputText(key='-TAG-')],
    [sg.Button('安装并监控'), sg.Button('退出本次测试'), sg.Text("如需退出程序,点击左上角'x'")],
]

with open('./logo.bin', 'rb') as f:
    data = f.read()
# 创建窗口
window = sg.Window('外广安装apk', layout, icon=data)

# 安装包名
package_name = ''

terminal_window_ids = []


def install_app(file_path):
    window.TKroot.title('apk安装中...')
    window.refresh()
    return d.install_app(file_path)


# 事件循环
while True:
    event, values = window.read()
    if event == '退出本次测试':
        window.TKroot.title('外广测试apk')
        # 清空文件输入框和标签输入框
        # window['-FILE-'].update('')
        # window['-TAG-'].update('')
        # 只关闭本次打开的Terminal窗口
        for window_id in terminal_window_ids:
            close_script = f'''
                   tell application "Terminal"
                       try
                           repeat with w in windows
                               if id of w is {window_id} then
                                   do script "exit" in w
                                   close w
                               end if
                           end repeat
                       end try
                   end tell
                   '''
            try:
                subprocess.run(['osascript', '-e', close_script], check=True)
            except subprocess.CalledProcessError as e:
                print(f"关闭终端窗口失败: {e}")

        # 确保所有相关的 adb logcat 进程都被终止
        try:
            subprocess.run(['pkill', '-f', 'adb logcat'], check=False)
        except Exception as e:
            print(f"终止 adb logcat 进程失败: {e}")

        # 卸载安装的应用
        if package_name:
            d.uninstall_app(package_name)
        # break

    elif event == '安装并监控':
        file_path = values['-FILE-']
        if file_path:
            # 安装应用并获取包名
            package_name = install_app(file_path)
            print('安装成功时间:', time.strftime('%H:%M:%S', time.localtime()))

            # 启动日志监控
            tags = values['-TAG-'].split(',')
            for tag in tags:
                # 打开新终端窗口,设置标题并执行 adb logcat 命令
                apple_script = f'''
                                tell application "Terminal"
                                    set newWindow to do script "adb logcat -s {tag}"
                                    set current settings of newWindow to settings set "Basic"
                                    set custom title of newWindow to "Log Monitor - {tag}"
                                    activate
                                    id of (first window whose selected tab is newWindow)
                                end tell
                                '''
                window_id = subprocess.check_output(['osascript', '-e', apple_script]).decode().strip()
                terminal_window_ids.append(window_id)
        else:
            print('请选择apk文件')

    elif event == sg.WINDOW_CLOSED:
        break
# 关闭窗口
window.close()

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2335867.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

数智读书笔记系列029 《代数大脑:揭秘智能背后的逻辑》

《代数大脑:揭秘智能背后的逻辑》书籍简介 作者简介 加里F. 马库斯(Gary F. Marcus)是纽约大学心理学荣休教授、人工智能企业家,曾创立Geometric Intelligence(后被Uber收购)和Robust.AI公司。他在神经科学、语言学和人工智能领域发表了大量论文,并著有《重启AI》等多部…

Apache Kafka UI :一款功能丰富且美观的 Kafka 开源管理平台!!

Apache Kafka UI 是一个免费的开源 Web UI,用于监控和管理 Apache Kafka 集群,可方便地查看 Kafka Brokers、Topics、消息、Consumer 等情况,支持多集群管理、性能监控、访问控制等功能。 1 特征 多集群管理: 在一个地方监控和管理…

临床协调简历模板

模板信息 简历范文名称:临床协调简历模板,所属行业:其他 | 职位,模板编号:C1S3WO 专业的个人简历模板,逻辑清晰,排版简洁美观,让你的个人简历显得更专业,找到好工作。希…

【第45节】windows程序的其他反调试手段上篇

目录 引言 一、通过窗口类名和窗口名判断 二、检测调试器进程 三、父进程是否是Explorer 四、RDTSC/GetTickCount时间敏感程序段 五、StartupInfo结构的使用 六、使用BeingDebugged字段 七、 PEB.NtGlobalFlag,Heap.HeapFlags,Heap.ForceFlags 八、DebugPort:CheckRem…

Linux常见指令介绍上(入门级)

1. ls指令 功能:显示出当前目录下的所有子目录与文件。 PS:注意显示的是当前文件下的子目录与文件。 以下这些是可以和ls配套使用的一些选项 -a 列出目录下的所有文件,包括以 . 开头的隐含文件。 -d 将目录象文件一样显示,而不是显示其下…

【版本控制】git命令使用大全

大家好,我是jstart千语。今天来总结一下git的使用命令,上文会先将git命令都列出来,便于快速寻找,然后还会对部分常用命令图文讲解,适合新手,让你快速地理解。最后还会总结在idea中使用git。如果有缺失的&am…

手阳明大肠经之合谷穴

个人主页:云纳星辰怀自在 座右铭:“所谓坚持,就是觉得还有希望!” 又名:1.《针灸甲乙经》:虎口。 所属经络:手阳明大肠经 定位 在手背,第1、2掌骨间,当第2掌骨桡侧的中点…

【VUE3】练习项目——大事件后台管理

目录 0 前言 1 准备工作 1.1 安装pnpm 1.2 创建vue项目 1.3 Eslint & Prettier的配置 1.4 husky 提交代码检查 1.5 目录调整 1.6 VueRouter4 1.6.1 基础配置 1.6.2 路由跳转 1.7 引入 Element Plus 组件库 1.8 Pinia 1.8.1 优化 1.9 封装请求工具 1.9.1 安…

万字长篇————C语言指针学习汇总

经过一段时间的学习,我们已经接触到了C语言的很多知识了。不过目前我们接下来我们要接触C语言中一个最大的“门槛”:指针。 什么是指针? 在介绍指针之前,我们首先要明白变量与地址之间的关系。 举一个生活中的案例:一…

实验一 HDFS的Shell操作

一、实验目的 熟悉HDFS Shell的语法格式,完成对HDFS上文件和目录的一系列操作 二、实验要求 2.1 练习dfs常用的子命令 ① -ls ② -du ③ -mv ④ -cp ⑤ -rm ⑥ -put ⑦ -cat ⑧ -help ⑨ -mkdir ⑩ -get 2.2通过Shell脚本定时采集数据到HDFS 三、实…

复变函数摘记3

复变函数摘记3 5. 留数5.1 可去奇点、极点、本性奇点5.2 零点与极点的关系5.3 在无穷远点处的情形5.4 留数 5. 留数 \quad 如果函数 f ( z ) f(z) f(z) 在 z 0 z_0 z0​ 及 z 0 z_0 z0​ 的邻域内处处可导,那么称 f ( z ) f(z) f(z) 在点 z 0 z_0 z0​ 处解析。…

【行测】判断推理:类比推理 and 定义判断

> 作者:დ旧言~ > 座右铭:读不在三更五鼓,功只怕一曝十寒。 > 目标:掌握 类比推理 and 定义判断 基本题型,并能运用到例题中。 > 毒鸡汤:有些事情,总是不明白,所以我不会…

Java使用WebSocket视频拆帧进度处理与拆帧图片推送,结合Apipost进行调试

<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-websocket</artifactId></dependency>Configuration public class WebSocketConfig {/*** 启动 WebSocket 服务器*/Beanpublic ServerEndpointE…

计算斐波那契数列

1.对于斐波那契数列的理解&#xff1a; (1)常规 该数列由兔子繁衍的想法产生&#xff0c;故又“兔子数列” 其数值为&#xff1a;1、1、2、3、5、8、13、21、34 从数学定义上走&#xff1a;F(0)1&#xff0c;F(1)1, F(n)F(n - 1)F(n - 2)&#xff08;n ≥ 2&#xff0c;n ∈…

JAVA如何操作文件?(超级详细)

目录 一、认识文件和相关知识 1.认识文件 2.⽬录 3.⽂件路径&#xff08;Path&#xff09; 4.文本文件和二进制文件的区分 二、File类操作文件 1.构造方法 2.方法 2.1 方法表 2.2 get相关的方法和构造方法 2.2.1 “.” 和 “..” 2.3 is相关的方法 2.4 删除相关…

(2)VTK C++开发示例 --- 绘制多面锥体

文章目录 1. 概述2. CMake链接VTK3. main.cpp文件4. 演示效果 更多精彩内容&#x1f449;内容导航 &#x1f448;&#x1f449;VTK开发 &#x1f448; 1. 概述 VTK C开发示例程序&#xff1b; 使用C 和VTK绘制一个多面锥体。 环境说明系统ubuntu22.04、windows11cmake3.22、3.2…

(2025-04-12)向老主机箱中安装新买的显卡及固态硬盘

目录 1 引言2 显卡及其驱动的安装3 固态硬盘的安装及C盘扩容3.1 固态硬盘正确连接到主板上后&#xff0c;操作系统上面仍然不显示对应盘符怎么办&#xff1f;3.2 如何对C盘扩容&#xff1f;3.3 新问题&#xff1a;原有D盘程序不能运行 4 总结 1 引言 今天安装昨天买的新固态硬…

rk3588 驱动开发(一)字符设备开发

3.字符设备驱动开发 3.1 什么是字符设备驱动 字符设备&#xff1a;就是一个个字节&#xff0c;按照字节流进行读写操作的设备&#xff0c;读写是按照先后顺序的。 举例子&#xff1a;IIC 按键 LED SPI LCD 等 Linux 应用程序调用驱动程序流程&#xff1a; Linux中驱动加载成功…

AbMole| 体内相互作用筛选揭示肝脏对癌症转移的限制

癌症转移&#xff0c;作为导致患者死亡的主要原因之一&#xff0c;其复杂机制一直是医学界研究的热点。肝脏&#xff0c;作为癌症转移的常见靶器官&#xff0c;其微环境对癌症细胞的生长和转移具有重要影响。然而&#xff0c;肝脏如何限制癌症转移的具体机制尚不完全清楚。 来…

STM32移植文件系统FATFS——片外SPI FLASH

一、电路连接 主控芯片选型为&#xff1a;STM32F407ZGT6&#xff0c;SPI FLASH选型为&#xff1a;W25Q256JV。 采用了两片32MB的片外SPI FLASH&#xff0c;电路如图所示。 SPI FLASH与主控芯片的连接方式如表所示。 STM32F407GT6W25Q256JVPB3SPI1_SCKPB4SPI1_MISOPB5SPI1_MOSI…