从0构建一个录制UI测试工具

   很多UI自动化测试工具都具备录制UI自动化测试的能力,例如playwright,可以通过playwright vscode插件完成录制,如下图所示,当选择录制脚本时,会打开一个浏览器,在浏览器中输入被测应用url,用户在web应用上的所有操作都能被录制下来,并将录制内容存放在vscode创建的test文件中。

  那么UI类测试工具是如何实现录制功能的呢?大致原理是:通过浏览器的开发者工具协议(DevTools Protocol)与浏览器进行通信,注入和执行 JavaScript 代码,并实时捕获和处理用户操作事件,将捕获的用户操作事件转换为工具的client script。例如playwright,就会转换成playwright得locator,action脚本。

  为了透彻理解UI自动化测试录制工作原理,可以从0搭建一个具备录制能力的工具。

初始化项目代码

安装相关依赖和配置package.json文件

mkdir ui-test-tool
cd ui-test-tool
npm init -y
npm install puppeteer express socket.io
{"name": "ui-test-tool","version": "1.0.0","description": "A UI test tool to record user interactions and generate code","main": "index.js","scripts": {"start": "node index.js"},"dependencies": {"express": "^4.17.1","puppeteer": "^19.0.0","socket.io": "^4.4.1"}
}

  构建index.js文件

下面的文件中使用到了socket.io和puppeteer。Socket.IO 是一个基于事件的实时网络库,用于在浏览器和服务器之间实现实时、双向通信。Puppeteer是一个Node.js 库,它提供了一个高级API来通过开发工具协议控制Chrome/Chromium。 Express是一种保持最低程度规模的灵活 Node.js Web应用程序框架,可以快速方便地创建强大的API。

  Socket.IO 提供了一些关键的方法和函数,用于实现客户端于服务器之间的通信。
服务器端(Node.js)
require('socket.io'):引入 Socket.IO 库。
const io = require('socket.io')(server):创建 Socket.IO 服务器实例,并将其绑定到 HTTP 或 HTTPS 服务器。
io.on('connection', (socket) => { ... }):监听客户端连接事件,当有新客户端连接时执行回调函数。
socket.emit(event, data):向特定的客户端发送消息。
socket.on(event, callback):监听客户端发送的消息事件,并定义对应的处理逻辑。

客户端(浏览器)
<script src="/socket.io/socket.io.js"></script>:在 HTML 中引入 Socket.IO 客户端库。
const socket = io():创建客户端 Socket.IO 实例,连接到服务器。
socket.emit(event, data):向服务器发送消息。
socket.on(event, callback):监听服务器发送的消息事件,并定义对应的处理逻辑。

  下面的代码中利用socket.io监听客户端发送的"start-recording"message,当监听到该消息时,会通过puppeteer启动浏览器,并在浏览器中开启一个新tab页。另外,下面代码中expose了两个方法,emitClickEvent和emitHoverSelector,这两个方法被window.socket里面的代码调用。实际就是,下面的代码中调用window.socket.emit(event,data)的时候,实际调用的是:emitClickEvent和emitHoverSelector中的代码。

   在page.evaluate(()=>{...})方法中,首先创建了一个tooltip,用于显示当鼠标hover到页面element上的时候,显示element的locator,tooltip在默认情况下,display='none',只有hover的时候才显示。接着对“mouseover”添加了监听,当监听到该操作时,会获取目标element的locator,当然下面的代码是使用css来获取element的locator。如果是其他UI自动化测试工具,例如playwright,这里就会生产plawright自己语法格式的locator。除了对“mouseover”添加监听,还对"click”操作也增加了监听。

import express from 'express';
import http from 'http';
import { Server } from 'socket.io';
import puppeteer from 'puppeteer';const app = express();
const server = http.createServer(app);
const io = new Server(server);app.use(express.static('public'));io.on('connection', (socket) => {socket.on('start-recording', async () => {const browser = await puppeteer.launch({ headless: false });const page = await browser.newPage();await page.exposeFunction('emitClickEvent', (selector) => {socket.emit('record-action', `await page.click('${selector}');\n`);});await page.exposeFunction('emitHoverSelector', (selector) => {socket.emit('record-selector', selector);});await page.evaluateOnNewDocument(() => {window.socket = {emit: (event, data) => {if (event === 'click'){window.emitClickEvent(data);} else if (event === 'hover') {window.emitHoverSelector(data);}}};});await page.goto('https://angularjs.realworld.io/#/login');await page.evaluate(() => {let path = [];const tooltip = document.createElement('div');tooltip.style.position = 'absolute';tooltip.style.background = 'rgba(0, 0, 0, 0.7)';tooltip.style.color = '#fff';tooltip.style.padding = '5px';tooltip.style.borderRadius = '3px';tooltip.style.zIndex = '9999';tooltip.style.display = 'none';document.body.appendChild(tooltip);document.addEventListener('mouseover', (event) => {const path = [];let el = event.target;while (el) {let selector = el.nodeName.toLowerCase();if (el.id) {selector += `#${el.id}`;} else if (el.className) {selector += `.${Array.from(el.classList).join('.')}`;} else {let sib = el, nth = 1;while (sib = sib.previousElementSibling) {if (sib.nodeName.toLowerCase() === selector) nth++;}selector += `:nth-of-type(${nth})`;}path.unshift(selector);el = el.parentNode;}const selectorPath = path.join(' > ');tooltip.innerText = selectorPath;tooltip.style.top = `${event.pageY}px`;tooltip.style.left = `${event.pageX}px`;tooltip.style.display = 'block';window.socket.emit('hover', selectorPath);});// Hide tooltip on mouseoutdocument.addEventListener('mouseout', () => {tooltip.style.display = 'none';});document.addEventListener('click', (event) => {let el = event.target;while (el) {let selector = el.nodeName.toLowerCase();if (el.id) {selector += `#${el.id}`;} else if (el.className) {selector += `.${Array.from(el.classList).join('.')}`;} else {let sib = el, nth = 1;while (sib = sib.previousElementSibling) {if (sib.nodeName.toLowerCase() === selector) nth++;}selector += `:nth-of-type(${nth})`;}path.unshift(selector);el = el.parentNode;}window.socket.emit('click', path.join(' > '));}, true);});socket.on('disconnect', async () => {await browser.close();console.log('Client disconnected');});});
});server.listen(3001, () => {console.log('Listening on port 3001');
});

 构建index.html文件

  index.html文件内容,在html内容中,引入了socket.io.js。当用户点击“start recording”按钮时,客户端脚本向服务端发送“start-recording”message,trigger服务端执行打开浏览器,开启新页面的操作。当在打开的页面上点击某个元素时,会将脚本显示在textare中。当鼠标hover到页面上的任意元素上时,会在旁边显示该元素的selector。

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>UI Test Tool</title><script src="/socket.io/socket.io.js" defer></script><script type="module" defer>document.addEventListener('DOMContentLoaded', () => {const socket = io();document.getElementById('start').addEventListener('click', () => {socket.emit('start-recording');});socket.on('record-action', (action) => {document.getElementById('code').innerText += action;});socket.on('record-selector', (action) => {document.getElementById('code').innerText += action;});});</script>
</head>
<body>
<div><button id="start">Start Recording</button></div>
<div>
<label for="code">Selector:</label><br>
<textarea id="code" rows="10" cols="50"></textarea>
</div>
</body>
</html>

 验证工具效果

  通过node index.js命令,启动服务。在浏览器中输入启动的服务地址,点击Start Recroding按钮,就会新开启chrome浏览器,并在浏览器中open new page,访问angularjs的这个应用,当鼠标hover到任意元素时,会显示该元素的selector。

  以上就是从0搭建UI录制工具的过程。如果是playwright vscode插件的录制具体又是如何实现的呢?实际,实现原理和上面相同,只是细节更加复杂,在下一篇博客中将通过阅读源码介绍vscode extension如何实现录制功能的。

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

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

相关文章

C++:enum枚举共用体union

enum枚举 C继承C的枚举用法 (1)典型枚举类型定义&#xff0c;枚举变量定义和使用 (2)枚举类型中的枚举值常量不能和其他外部常量名称冲突&#xff1a; 举例1宏定义&#xff0c;举例2另一个枚举 // 定义一个名为Color的枚举类型 enum Color {RED, // 红色&#xff0c;默认值…

昇思25天学习打卡营第11天|SSD目标检测

1. 学习内容复盘 模型简介 SSD&#xff0c;全称Single Shot MultiBox Detector&#xff0c;是Wei Liu在ECCV 2016上提出的一种目标检测算法。使用Nvidia Titan X在VOC 2007测试集上&#xff0c;SSD对于输入尺寸300x300的网络&#xff0c;达到74.3%mAP(mean Average Precision)…

JAVA毕业设计145—基于Java+Springboot+vue+uniapp的驾校预约小程序(源代码+数据库+15000字论文)

毕设所有选题&#xff1a; https://blog.csdn.net/2303_76227485/article/details/131104075 基于JavaSpringbootvueuniapp的驾校预约小程序(源代码数据库15000字论文)145 一、系统介绍 本项目前后端分离&#xff0c;分为用户、教练、管理员三种角色 1、用户&#xff1a; …

Redis数据结构——跳跃表(Skiplist)

Redis数据结构——跳跃表&#xff08;Skiplist&#xff09; Redis作为一个高性能的键值存储系统&#xff0c;提供了多种数据结构供开发者选择。其中&#xff0c;跳跃表&#xff08;Skiplist&#xff09;是一种特殊的数据结构&#xff0c;它在Redis中主要用于实现有序集合&…

ModuleNotFoundError: No module named ‘_sysconfigdata_x86_64_conda_linux_gnu‘

ModuleNotFoundError: No module named _sysconfigdata_x86_64_conda_linux_gnu 1.软件环境⚙️2.问题描述&#x1f50d;3.解决方法&#x1f421;4.结果预览&#x1f914; 1.软件环境⚙️ Ubuntu 20.04 Python 3.7.0 2.问题描述&#x1f50d; 今天发现更新conda之后&#xff0…

Redisson(分布式锁、限流)

注意Redisson是基于Redis的&#xff0c;所以必须先引入Redis配置&#xff08;参考SpringBoot集成Redis文章&#xff09; 1. 集成Redisson 引入依赖 <!-- 二选一,区别是第一个自动配置&#xff0c;第二个还需要手动配置也就是第二步自定义配置&#xff0c;注意版本号&…

Windows怎么实现虚拟IP

在做高可用架构时&#xff0c;往往需要用到虚拟IP&#xff0c;在linux上面有keepalived来实现虚拟ip的设置。在windows上面该怎么弄&#xff0c;keepalived好像也没有windows版本&#xff0c;我推荐一款浮动IP软件PanguVip&#xff0c;它可以实现windows上面虚拟ip的漂移。设置…

UE5材质之HLSL:深度

UE4/5的Custom节点&#xff1a;在VScode使用HLSL&#xff08;新手入门用&#xff09;_vscode写hlsl-CSDN博客 效果&#xff1a; 材质节点&#xff1a; 自定义节点代码&#xff1a; float3 rayStepViewDir*-1; float4 inputTexTexture2DSample(TexObject,TexObjectSampler,uv)…

字体属性(笔记)

字体属性: 含义属性参数说明例子字体样式font-family字体样式的名称可以是使用多个字体样式&#xff0c;后面作为备用字体&#xff0c;font-family&#xff1a;“微软雅黑”&#xff0c;“宋体”;字体大小font-size像素:px,em,remfont-size:20px&#xff1b;字体颜色color颜色…

如何用 PHP 实现一个自定义爬虫框架

随着互联网的不断发展&#xff0c;信息量爆炸式增长&#xff0c;获取有价值的信息已经成为了许多人的需求。在这样的大环境下&#xff0c;爬虫技术逐渐兴起&#xff0c;成为了大数据时代的重要工具之一。爬虫技术的应用十分广泛&#xff0c;其可以用于网络舆情监测、数据分析、…

docker中mysql突然无法远程连接设置

docker登陆到docker.hub docker login -u 用户名 回车密码 将容器打包成自己的镜像 docker commit -a "用户名" -m "redis" 533d6f1402ca 用户名/myredis:v1.2 将镜像发布到平台上 docker push用户名/myredis:v1.2 删除本地镜像 docker rm image …

JavaSE主要内容(全套超完整)

一、为什么选择Java&#xff08;Java的优势&#xff09; 1、应用面广&#xff1a; 相较于其他语言&#xff0c;Java的应用面可谓是非常广&#xff0c;这得益于他的跨平台性和其性能的稳定性。他在服务器后端&#xff0c;Android应用开发&#xff0c;大数据开发&#xf…

MATLAB中添加 Git 子模块

目录 更新子模块 对子模块使用提取和合并 使用推送将更改发送到子模块存储库 要重用其他存储库中的代码&#xff0c;可以指定 Git™ 子模块。 要将外部 Git 存储库克隆为子模块&#xff0c;请执行以下操作&#xff1a; 在 MATLAB 当前文件夹浏览器中点击右键&#xff0c;然…

Jmeter性能场景设计

为什么会有性能场景设计呢&#xff1f; 相信有部分同学对场景设计优点模糊&#xff0c;前面博文提到的是场景提取 场景设计&#xff1a;在压测的过程中怎么设置线程数、Ramp-Up时间(秒)、循环次数等等 一、 性能场景分类 场景的概念&#xff1a; a. 单场景 b. 混合场景 c. 容…

Python学习笔记17 -- 猜数字小游戏2

目录 一、功能函数 1、说明函数 -- 对游戏玩法及设置进行说明 2、答案函数 -- 生成答案 3、猜测函数 -- 让玩家进行猜测 4、对照函数 -- 将答案和猜测进行对照 4.1 A函数 4.2 B函数 5、结果函数 -- 判断得到结果或继续猜测 6、时间函数 -- 判断一局游戏所用时间 7、打…

开源项目-商城管理系统

哈喽,大家好,今天主要给大家带来一个开源项目-商城管理系统 商城管理系统分前后端两部分。前端主要有商品展示,我的订单,个人中心等内容;后端的主要功能包括产品管理,门店管理,会员管理,订单管理等模块 移动端页面

深入理解深度神经网络(DNN)

深入理解深度神经网络&#xff08;DNN&#xff09; 大家好&#xff0c;我是免费搭建查券返利机器人省钱赚佣金就用微赚淘客系统3.0的小编&#xff0c;也是冬天不穿秋裤&#xff0c;天冷也要风度的程序猿&#xff01; 什么是深度神经网络&#xff08;DNN&#xff09;&#xff…

力扣第216题“组合总和 III”

在本篇文章中&#xff0c;我们将详细解读力扣第216题“组合总和 III”。通过学习本篇文章&#xff0c;读者将掌握如何使用回溯法来解决这一问题&#xff0c;并了解相关的复杂度分析和模拟面试问答。每种方法都将配以详细的解释&#xff0c;以便于理解。 问题描述 力扣第216题…

[OtterCTF 2018]Name Game

Name Game 题目描述&#xff1a;我们知道这个帐号登录到了一个名为Lunar-3的频道。账户名是什么&#xff1f;猜想&#xff1a;既然登陆了游戏&#xff0c;我们尝试直接搜索镜像中的字符串 Lunar-3 。 直接搜索 Lunar-3 先把字符串 重定向到 txt文件里面去然后里面查找 Lunar-3…

什么是机器学习,机器学习与人工智能的区别是什么(一)?

人工智能和计算机游戏领域的先驱阿瑟塞缪尔&#xff08;Arthur Samuel&#xff09;创造了 "机器学习"一词。他将机器学习定义为 “一个让计算机无需明确编程即可学习的研究领域” 。通俗地说&#xff0c;机器学习&#xff08;ML&#xff09;可以解释为根据计算机的经…