技术标签: 测试面试 自动化测试 selenium 1024程序员节 软件测试 单元测试 职场和发展 面试找工作
自动化测试是软件测试中非常重要的一部分,可以提高测试效率和测试覆盖率。在UI自动化测试中,Selenium是非常流行的工具。本文将介绍如何使用Python和Selenium搭建UI自动化测试框架。
在开始搭建UI自动化测试框架之前,需要先安装Python和Selenium。可以从Python官网下载Python安装包,并使用pip命令安装Selenium。
在搭建UI自动化测试框架之前,需要先规划好框架的目录结构。以下是一个比较完整的目录结构:
├── config
│ ├── __init__.py
│ ├── config.ini
│ └── logger.ini
├── data
│ ├── __init__.py
│ ├── test_data.xlsx
│ └── test_data.json
├── logs
│ ├── __init__.py
│ └── test.log
├── page
│ ├── __init__.py
│ └── login_page.py
├── report
│ ├── __init__.py
│ └── test_report.html
│ └── screenshots
│ ├── test_login_failure_2022-05-01_08-30-00.png
│ └── test_login_success_2022-05-01_08-30-00.png
├── testcase
│ ├── __init__.py
│ └── test_login.py
├── utils
│ ├── __init__.py
│ ├── driver.py
│ ├── logger.py
│ └── read_config.py
│ └── take_screenshot.py
├── .gitignore
├── README.md
├── requirements.txt
└── run.py
config:存放配置文件,包括config.ini和logger.ini。
data:存放测试数据,包括Excel和JSON格式的数据。
logs:存放日志文件,包括test.log。
page:存放页面对象,每个页面对应一个.py文件。
report:存放测试报告,包括test_report.html。
screenshots:用于存放测试过程中的截图。
testcase:存放测试用例,每个用例对应一个.py文件。
utils:存放工具类,包括driver.py、logger.py和read_config.py。
.gitignore:Git忽略文件列表。
README.md:项目说明文件。
requirements.txt:Python依赖包列表。
run.py:测试执行入口。
mkdir config data logs page report report/screenshots testcase utils
touch .gitignore README.md requirements.txt run.py
cd config && touch __init__.py config.ini logger.ini && cd ..
cd data && touch __init__.py test_data.xlsx test_data.json && cd ..
cd logs && touch __init__.py test.log && cd ..
cd page && touch __init__.py login_page.py && cd ..
cd report && touch __init__.py test_report.html && cd ..
cd testcase && touch __init__.py test_login.py && cd ..
cd utils && touch __init__.py driver.py logger.py read_config.py && cd ..
pip install selenium openpyxl configparser loguru
[Browser]
browser_name = chrome
[URL]
base_url = https://www.example.com
[loggers]
keys=root
[handlers]
keys=consoleHandler,fileHandler
[formatters]
keys=formatter
[logger_root]
level=DEBUG
handlers=consoleHandler,fileHandler
[handler_consoleHandler]
class=StreamHandler
level=DEBUG
formatter=formatter
args=(sys.stdout,)
[handler_fileHandler]
class=handlers.TimedRotatingFileHandler
level=DEBUG
formatter=formatter
args=('logs/test.log', 'D', 1, 30)
[formatter_formatter]
format=%(asctime)s - %(name)s - %(levelname)s - %(message)s
datefmt=%Y-%m-%d %H:%M:%S
import os
from selenium import webdriver
from configparser import ConfigParser
def get_driver():
config = ConfigParser()
config.read(os.path.join(os.path.dirname(__file__), '../config/config.ini'))
browser_name = config.get('Browser', 'browser_name')
if browser_name.lower() == 'chrome':
return webdriver.Chrome()
elif browser_name.lower() == 'firefox':
return webdriver.Firefox()
elif browser_name.lower() == 'edge':
return webdriver.Edge()
else:
raise ValueError('Unsupported browser: {}'.format(browser_name))
import os
from loguru import logger
from configparser import ConfigParser
config = ConfigParser()
config.read(os.path.join(os.path.dirname(__file__), '../config/logger.ini'))
logger.add(config.get('handler_fileHandler', 'class'), level=config.get('logger_root', 'level'), rotation=config.getint('handler_fileHandler', 'args')[2], retention=config.getint('handler_fileHandler', 'args')[3])
def get_logger(name):
return logger.bind(name=name)
import os
from configparser import ConfigParser
config = ConfigParser()
config.read(os.path.join(os.path.dirname(__file__), '../config/config.ini'))
def get_base_url():
return config.get('URL', 'base_url')
import os
from datetime import datetime
from utils.driver import get_driver
def take_screenshot(name):
driver = get_driver()
screenshot_dir = os.path.join(os.path.dirname(__file__), '../report/screenshots')
os.makedirs(screenshot_dir, exist_ok=True)
screenshot_file = os.path.join(screenshot_dir, '{}_{}.png'.format(name, datetime.now().strftime('%Y-%m-%d_%H-%M-%S')))
driver.save_screenshot(screenshot_file)
from selenium.webdriver.common.by import By
from utils.driver import get_driver
class LoginPage:
url = '/login.html'
username_input = (By.ID, 'username')
password_input = (By.ID, 'password')
login_button = (By.ID, 'login-button')
def __init__(self):
self.driver = get_driver()
def open(self):
self.driver.get(get_base_url() + self.url)
def close(self):
self.driver.quit()
def login(self, username, password):
self.driver.find_element(*self.username_input).send_keys(username)
self.driver.find_element(*self.password_input).send_keys(password)
self.driver.find_element(*self.login_button).click()
import unittest
from page.login_page import LoginPage
from utils.take_screenshot import take_screenshot
class TestLogin(unittest.TestCase):
def setUp(self):
self.page = LoginPage()
def tearDown(self):
self.page.close()
def test_login_success(self):
self.page.open()
self.page.login('admin', 'admin123')
self.assertIn('Welcome', self.page.driver.title)
def test_login_failure(self):
self.page.open()
self.page.login('admin', 'wrong_password')
self.assertIn('Login failed', self.page.driver.page_source)
take_screenshot('test_login_failure')
import unittest
from datetime import datetime
from utils.logger import get_logger
from report import HTMLTestRunner
logger = get_logger(__name__)
if __name__ == '__main__':
logger.info('Start testing...')
suite = unittest.defaultTestLoader.discover('testcase')
report_file = 'report/test_report_{}.html'.format(datetime.now().strftime('%Y-%m-%d_%H-%M-%S'))
with open(report_file, 'wb') as f:
runner = HTMLTestRunner.HTMLTestRunner(stream=f, title='Test Report', description='Test Result')
runner.run(suite)
logger.info('Testing finished. Report file: {}'.format(report_file))
# 将截图嵌入测试报告
with open(report_file, 'r+', encoding='utf-8') as f:
content = f.read()
for root, dirs, files in os.walk('report/screenshots'):
for file in files:
screenshot_file = os.path.join(root, file)
if 'test_report' not in screenshot_file:
screenshot_name = os.path.splitext(os.path.basename(screenshot_file))[0]
screenshot_time = datetime.strptime(screenshot_name.split('_')[-2], '%Y-%m-%d')
screenshot_url = os.path.join(get_base_url(), screenshot_file)
content = content.replace('{}"'.format(screenshot_name), '{}" width="50%"'.format(screenshot_url))
f.seek(0)
f.write(content)
python run.py
测试完成后,测试报告将保存在report目录下的test_report.html文件中。
持续集成是软件开发过程中非常重要的一环,可以帮助开发团队实现快速迭代和快速交付。Jenkins是一个开源的持续集成工具,可以帮助团队实现自动化构建、自动化测试和自动化部署。本文将介绍如何使用Jenkins进行UI自动化测试的持续集成。
Jenkins可以从官网下载安装包,根据操作系统选择相应的安装包进行安装。安装完成后,启动Jenkins服务。
Jenkins需要安装一些插件来支持UI自动化测试,包括:
Selenium Plugin:用于集成Selenium和Jenkins。
HTML Publisher Plugin:用于生成测试报告。
在Jenkins的插件管理页面中,搜索并安装这两个插件。
在Jenkins的系统管理页面中,配置全局环境变量和全局工具配置。
配置全局环境变量:设置PYTHONPATH变量,指向Python的安装目录。
配置全局工具:添加Python和浏览器驱动的安装路径。
在Jenkins的首页中,点击“新建任务”,选择“自由风格软件项目”,输入任务名称,点击“确定”。
在任务配置页面中,选择“Git”作为源码管理方式,填写代码仓库地址和分支信息。
在任务配置页面中,选择“构建触发器”,配置定时构建或者代码提交构建。
在任务配置页面中,选择“构建环境”,配置Python环境和浏览器驱动。
在任务配置页面中,选择“构建步骤”,配置构建脚本。
构建脚本如下:
#!/bin/bash
# 安装依赖包
pip install -r requirements.txt
# 执行测试
python run.py
# 生成测试报告
cp report/test_report.html $WORKSPACE
# 发布测试报告
echo '<h2>UI自动化测试报告</h2>' > report.html
echo '<iframe src="test_report.html" width="100%" height="600"></iframe>' >> report.html
cp report.html $WORKSPACE
在任务配置页面中,选择“构建后操作”,配置测试报告的发布方式。
发布方式如下:
发布HTML测试报告:选择“HTML Publisher Plugin”,设置测试报告路径为$WORKSPACE/report.html。
在Jenkins的任务页面中,点击“立即构建”,Jenkins将自动拉取代码,执行测试,并生成测试报告。测试报告将发布到Jenkins的测试报告页面中。
本文主要介绍了如何使用Python Selenium框架搭建UI自动化测试框架,并给出了一个完整的目录结构示例进行参考。通过UI自动化测试框架,可以帮助团队快速构建自动化测试框架,提高测试效率和测试质量。希望本文能对你有所帮助。
最后: 下方这份完整的软件测试视频教程已经整理上传完成,需要的朋友们可以自行领取【保证100%免费】
我们学习必然是为了找到高薪的工作,下面这些面试题是来自阿里、腾讯、字节等一线互联网大厂最新的面试资料,并且有字节大佬给出了权威的解答,刷完这一套面试资料相信大家都能找到满意的工作。
文章浏览阅读645次。这个肯定是末尾的IDAT了,因为IDAT必须要满了才会开始一下个IDAT,这个明显就是末尾的IDAT了。,对应下面的create_head()代码。,对应下面的create_tail()代码。不要考虑爆破,我已经试了一下,太多情况了。题目来源:UNCTF。_攻防世界困难模式攻略图文
文章浏览阅读2.9k次,点赞3次,收藏10次。偶尔会用到,记录、分享。1. 数据库导出1.1 切换到dmdba用户su - dmdba1.2 进入达梦数据库安装路径的bin目录,执行导库操作 导出语句:./dexp cwy_init/[email protected]:5236 file=cwy_init.dmp log=cwy_init_exp.log 注释: cwy_init/init_123..._达梦数据库导入导出
文章浏览阅读1.9k次。1. 在官网上下载KindEditor文件,可以删掉不需要要到的jsp,asp,asp.net和php文件夹。接着把文件夹放到项目文件目录下。2. 修改html文件,在页面引入js文件:<script type="text/javascript" src="./kindeditor/kindeditor-all.js"></script><script type="text/javascript" src="./kindeditor/lang/zh-CN.js"_kindeditor.js
文章浏览阅读2.3k次,点赞6次,收藏14次。SPI的详情简介不必赘述。假设我们通过SPI发送0xAA,我们的数据线就会变为10101010,通过修改不同的内容,即可修改SPI中0和1的持续时间。比如0xF0即为前半周期为高电平,后半周期为低电平的状态。在SPI的通信模式中,CPHA配置会影响该实验,下图展示了不同采样位置的SPI时序图[1]。CPOL = 0,CPHA = 1:CLK空闲状态 = 低电平,数据在下降沿采样,并在上升沿移出CPOL = 0,CPHA = 0:CLK空闲状态 = 低电平,数据在上升沿采样,并在下降沿移出。_stm32g431cbu6
文章浏览阅读1.2k次,点赞2次,收藏8次。数据链路层习题自测问题1.数据链路(即逻辑链路)与链路(即物理链路)有何区别?“电路接通了”与”数据链路接通了”的区别何在?2.数据链路层中的链路控制包括哪些功能?试讨论数据链路层做成可靠的链路层有哪些优点和缺点。3.网络适配器的作用是什么?网络适配器工作在哪一层?4.数据链路层的三个基本问题(帧定界、透明传输和差错检测)为什么都必须加以解决?5.如果在数据链路层不进行帧定界,会发生什么问题?6.PPP协议的主要特点是什么?为什么PPP不使用帧的编号?PPP适用于什么情况?为什么PPP协议不_接收方收到链路层数据后,使用crc检验后,余数为0,说明链路层的传输时可靠传输
文章浏览阅读587次。软件测试工程师移民加拿大 无证移民,未受过软件工程师的教育(第1部分) (Undocumented Immigrant With No Education to Software Engineer(Part 1))Before I start, I want you to please bear with me on the way I write, I have very little gen...
文章浏览阅读304次。Thinkpad X250笔记本电脑,装的是FreeBSD,进入BIOS修改虚拟化配置(其后可能是误设置了安全开机),保存退出后系统无法启动,显示:secure boot failed ,把自己惊出一身冷汗,因为这台笔记本刚好还没开始做备份.....根据错误提示,到bios里面去找相关配置,在Security里面找到了Secure Boot选项,发现果然被设置为Enabled,将其修改为Disabled ,再开机,终于正常启动了。_安装完系统提示secureboot failure
文章浏览阅读10w+次,点赞93次,收藏352次。1、用strtok函数进行字符串分割原型: char *strtok(char *str, const char *delim);功能:分解字符串为一组字符串。参数说明:str为要分解的字符串,delim为分隔符字符串。返回值:从str开头开始的一个个被分割的串。当没有被分割的串时则返回NULL。其它:strtok函数线程不安全,可以使用strtok_r替代。示例://借助strtok实现split#include <string.h>#include <stdio.h&_c++ 字符串分割
文章浏览阅读2.3k次。1 .高斯日记 大数学家高斯有个好习惯:无论如何都要记日记。他的日记有个与众不同的地方,他从不注明年月日,而是用一个整数代替,比如:4210后来人们知道,那个整数就是日期,它表示那一天是高斯出生后的第几天。这或许也是个好习惯,它时时刻刻提醒着主人:日子又过去一天,还有多少时光可以用于浪费呢?高斯出生于:1777年4月30日。在高斯发现的一个重要定理的日记_2013年第四届c a组蓝桥杯省赛真题解答
文章浏览阅读851次,点赞17次,收藏22次。摘要:本文利用供需算法对核极限学习机(KELM)进行优化,并用于分类。
文章浏览阅读1.1k次。一、系统弱密码登录1、在kali上执行命令行telnet 192.168.26.1292、Login和password都输入msfadmin3、登录成功,进入系统4、测试如下:二、MySQL弱密码登录:1、在kali上执行mysql –h 192.168.26.129 –u root2、登录成功,进入MySQL系统3、测试效果:三、PostgreSQL弱密码登录1、在Kali上执行psql -h 192.168.26.129 –U post..._metasploitable2怎么进入
文章浏览阅读257次。本文将为初学者提供Python学习的详细指南,从Python的历史、基础语法和数据类型到面向对象编程、模块和库的使用。通过本文,您将能够掌握Python编程的核心概念,为今后的编程学习和实践打下坚实基础。_python人工智能开发从入门到精通pdf