PO封装设计模式 -- Web页面端测试
一、已登录页面 -->新建PO封装的包 -- 以下源码适用于python3以上的版本

代码优化新增 Image -->对操作步骤进行截图
二、basepage 包基础类的封装如下:
from selenium.webdriver.support.wait import WebDriverWait #显示等待
from selenium.webdriver.support import expected_conditions as EC #判断元素是否定位到
#页面的基础类 -->为后续类调用
class HomePage():
'''构造方法'''
def __init__(self,url,dr):
self.url = url
self.dr = dr
#封装元素定位方式
def find_element(self,*loc):
try:
#入参本身就是元组 不需要加*
WebDriverWait(self.dr,10).until(EC.visibility_of_element_located(loc))
return self.dr.find_element(*loc)
except:
print(*loc + '元素定位在页面中无法找到')
三、page 包封装登录页面的类如下:
from selenium.webdriver.common.by import By #定位方式
from time import sleep
from TestCase.Gjs_Po.basepage.HomePage import *
#登录页面的类
class GjsLoginPage(HomePage):
#定位器
#输入账号
username_loc = (By.ID,"mobilePhone")
# 输入密码
password_loc = (By.ID,"password")
# 点击登录按钮
loginButton_loc = (By.CSS_SELECTOR,"#loginBtn")
# 登录后验证信息
loginInfo_loc = (By.CSS_SELECTOR,'#realName')
# 点击安全登录
logout_loc = (By.CSS_SELECTOR,"a.fc-blue.mr-5")
#用户名为空 --获取错误信息
userNull_loc = (By.CSS_SELECTOR, "span.error")
#密码为空 --获取错误信息
passwordNull_loc = (By.CSS_SELECTOR, "span.error")
#用户名输入非法字符--获取错误信息
usernameFeifa_loc = (By.CSS_SELECTOR, "span.error")
#打开登录页面
def openLoginPage(self):
self.dr.get(self.url)
self.dr.refresh()
self.dr.maximize_window()
sleep(0.5)
#输入用户名
def input_username(self,username):
#注意 self前面加上 *
self.find_element(*self.username_loc).send_keys(username)
#输入密码
def input_password(self,password):
self.find_element(*self.password_loc).send_keys(password)
#点击登录按钮
def click_btn(self):
self.find_element(*self.loginButton_loc).click()
##登录后验证信息
def getInfo(self):
return self.find_element(*self.loginInfo_loc).text
#点击安全退出
def logout_btn(self):
self.find_element(*self.logout_loc).click()
#用户名为空
def usernameNull(self):
return self.find_element(*self.userNull_loc).text
#密码为空
def passwordNull(self):
return self.find_element(*self.passwordNull_loc).text
#用户名输入非法字符--获取错误信息
def userFeiFa(self):
return self.find_element(*self.usernameFeifa_loc).text
四、common 登录用到的数据封装Helper类 如下:
import xlrd
import os
import logging
import time
class Helper(object):
def readExcels(self,nrow):
# 读取Excel数据 nrow是输入的行数
table = xlrd.open_workbook('D:\Project\TestCase\Gjs_Po\data\Gjsinfo.xlsx', 'r')
sheet = table.sheet_by_index(0)
return sheet.row_values(nrow)
def readusername(self,nrow):
# 读取用户名
return str(self.readExcels(nrow)[0]) # 强制转换为str
def readpassword(self,nrow):
# 读取密码
return str(self.readExcels(nrow)[1])
def readAssertText(self,nrow):
# 读取预期结果
return str(self.readExcels(nrow)[2])
def dirName(self,filename=None,filePath=None):
#定义日志文件 filename = log.md 为存储日志
return os.path.join(os.path.dirname(os.path.dirname(__file__)),filePath,filename)
def log(self,log_content):
'''日志定义级别'''
# 定义文件
logFile = logging.FileHandler(self.dirName('logInfo.md','log'), 'a',encoding='utf-8')
# log格式
fmt = logging.Formatter(fmt='%(asctime)s-%(name)s-%(levelname)s-%(module)s:%(message)s')
logFile.setFormatter(fmt)
# 定义日志
logger1 = logging.Logger('logTest', level=logging.DEBUG)
logger1.addHandler(logFile)
logger1.info(log_content)
logFile.close()
def SaveImage(self,driver,Filename):
"""用例失败截图"""
Rawpath = os.path.join(os.path.dirname(os.path.dirname(__file__)),"Image")
NewPicture = Rawpath + "\\" + time.strftime("%Y_%m_%d_%H_%M_%S") + "_" + Filename
driver.get_screenshot_as_file(NewPicture)
测试固件分离:
from selenium import webdriver
from TestCase.Gjs_Po.page.GjsLoginpage import *
import unittest
class MyunitTests(unittest.TestCase):
def setUp(self)-> None:
self.url = 'https://www.gjfax.com/toLogin'
self.dr = webdriver.Chrome()
self.dr.implicitly_wait(10)
#实例化一个对象
self.loginpage = GjsLoginPage(self.url,self.dr)
def tearDown(self)-> None:
self.dr.quit()
五、testCases 测试用例封装,继承unittest、Helper类,如下:
from TestCase.Gjs_Po.common.helper import *
from TestCase.Gjs_Po.common.ownUnit import *
import unittest
#类的继承
class GjsTestLogin(MyunitTests,Helper):
#测试正常的登录
def test_login_success(self):
#打开登录页面
self.loginpage.openLoginPage()
self.log('测试正常:自动化测试 -->打开登录页面')
#输入用户名
self.loginpage.input_username(self.readusername(1))
self.log('测试正常:自动化测试 -->输入用户名')
#输入密码
self.loginpage.input_password(self.readpassword(1))
self.log('测试正常:自动化测试 -->输入密码')
#点击登录
self.loginpage.click_btn()
self.log('测试正常:自动化测试 -->点击登录')
#断言
self.assertEqual(self.loginpage.getInfo(),self.readAssertText(1))
self.log('测试正常:自动化测试 -->断言登录后信息是否符合预期')
self.SaveImage(self.dr,"login_success.png")
#点击退出
self.loginpage.logout_btn()
self.log('测试正常:自动化测试 -->点击安全退出')
self.log('第一条用例执行结束')
#测试密码为空
def test_password_null(self):
# 打开登录页面
self.loginpage.openLoginPage()
self.log('测试密码为空:自动化测试 -->点击登录页面')
# 输入用户名
self.loginpage.input_username(self.readusername(2))
self.log('测试密码为空:自动化测试 -->输入用户名')
# 不输入密码
self.loginpage.input_password(self.readpassword(2))
self.log('测试密码为空:自动化测试 -->不输入密码')
# 点击登录
self.loginpage.click_btn()
self.log('测试密码为空:自动化测试 -->点击登录')
# 断言
self.assertEqual(self.loginpage.passwordNull(), self.readAssertText(2))
self.log('测试密码为空:自动化测试 -->断言密码为空信息是否符合预期')
self.SaveImage(self.dr,"login_passwordNull.png")
self.log('第二条用例执行结束')
#测试用户名为空
def test_username_null(self):
# 打开登录页面
self.loginpage.openLoginPage()
self.log('测试用户名为空:自动化测试 -->点击登录页面')
# 输入用户名
self.loginpage.input_username(self.readusername(3))
self.log('测试用户名为空:自动化测试 -->不输入用户名')
# 输入密码
self.loginpage.input_password(self.readpassword(3))
self.log('测试用户名为空:自动化测试 -->输入密码')
# 点击登录
self.loginpage.click_btn()
self.log('测试用户名为空:自动化测试 -->点击登录')
# 断言
self.assertEqual(self.loginpage.usernameNull(), self.readAssertText(3))
self.log('测试用户名为空:自动化测试 -->断言用户名为空信息是否符合预期')
self.SaveImage(self.dr,"login_usernameNull.png")
self.log('第三条用例执行结束')
#测试用户名输入非法字符
def test_username_Feifa(self):
# 打开登录页面
self.loginpage.openLoginPage()
self.log('测试用户名输入非法:自动化测试 -->点击登录页面')
# 输入非法用户名
self.loginpage.input_username(self.readusername(4))
self.log('测试用户名输入非法:自动化测试 -->输入非法用户名')
# 输入密码
self.loginpage.input_password(self.readpassword(4))
self.log('测试用户名输入非法:自动化测试 -->输入密码')
# 点击登录
self.loginpage.click_btn()
self.log('测试用户名输入非法:自动化测试 -->点击登录')
# 断言
self.assertEqual(self.loginpage.userFeiFa(), self.readAssertText(2))
self.log('测试用户名输入非法:自动化测试 -->断言用户名输入非法信息是否符合预期')
self.SaveImage(self.dr,"login_usernameFeifa.png")
self.log('第四条用例执行结束')
if __name__ == '__main__':
unittest.main(verbosity=2)
六、mail邮件的类封装如下:
import smtplib
import os
from email.mime.multipart import MIMEMultipart #邮件附件类
from email.mime.text import MIMEText #邮件模板类
from email.header import Header #邮件头部模板
#发送带附件的函数
def semd_mail(file_new):
with open(file_new, "rb") as fp:
mail_body = fp.read()
#基本信息
smtpserver = "smtp.163.com"
port = 25
sender = "1879781xxxx@163.com"
psw = "qqqxxx" #163邮箱密码
receiver = "1512500xxx@qq.com"
#定义邮箱主题
msg = MIMEMultipart()
msg['subject'] = Header(u"自动化测试报告","utf-8")
msg['from'] = sender
msg["to"] = receiver
#不加msg['from'] msg["to"]会报错,是因为"收件人和发件人没有定义"
#HTML 邮件正文,直接发送附件的代码
body = MIMEText(mail_body,"html","utf-8")
msg.attach(body)
# 附件
att = MIMEText(mail_body,"base64","utf-8")
att["Content-Type"] = "application/octet-stream"
att["Content-Disposition"] = 'attachment; filename="test_report.html'
msg.attach(att)
#链接邮件服务器发送邮件
try:
smtp = smtplib.SMTP()
smtp.connect(smtpserver) # 连服务器
smtp.login(sender, psw)
except:
smtp = smtplib.SMTP_SSL(smtpserver, port)
smtp.login(sender, psw) # 登录
smtp.sendmail(sender, receiver, msg.as_string()) # 发送
print("邮件发送成功")
smtp.quit()
#查找最新的邮件
def new_file(report_path):
result_path = report_path
lists = os.listdir(result_path)
lists.sort() #从小到大排序
file = [x for x in lists if x.endswith(".html")]
file_new_name = os.path.join(result_path,file[-1])
return file_new_name
七、runAllcase.py 一键执行测试用例 Main()执行
from TestCase.Gjs_Po.common.HTMLTestRunner import *
from TestCase.Gjs_Po.mail.sendmail163 import *
#实现的逻辑
#先生成测试报告
#查找最新的测试报告文件,通过new_file函数找到最新的测试报告,把返回的结果return file_new_name
#把new_file函数,file_new_name文件传入到send_mail 读取出来,然后以邮件的附件形式加载到邮件模板中,设置参数,链接服务器发送最新的测试报告#
if __name__ == '__main__':
report = os.path.dirname(__file__) #D:/Project/TestCase/Gjs_Po
test_path = os.path.join(report,"testcase") #测试用例文件夹
suite = unittest.defaultTestLoader.discover(
start_dir=test_path,pattern="Gjs_login*.py",top_level_dir=None)
report_path = os.path.join(report,"report") #测试报告文件夹
if not os.path.exists(report_path):
os.mkdir(report_path)
else:
print('存在report报告文件夹')
now = time.strftime("%Y-%m-%d %H_%M_%S") #时间戳
file_name = report_path + '\\' + now + "report_test.html" #测试报告文件名
fp = open(file_name,"wb")
runner = HTMLTestRunner(
stream = fp,
title = 'GJs自动化测试报告',
description = 'u"系统环境:Windows7 浏览器:Chrome 用例执行情况:"'
)
runner.run(suite) #执行测试用例
fp.close()
new_report = new_file(report_path) #获取最新的报告文件
semd_mail(new_report) #发送最新的测试报告
PO封装设计模式 -- Web页面端测试的更多相关文章
- PO封装设计模式 -- App移动端测试
前言: 一.App_Po 封装 (用互联网上随便一个app进行) base 存放的是页面基础类,后续的类需继承基础类 common 存放的是公共部分,测试固件分离部分,新增截图功能部分 Data 存放 ...
- Web页面速度测试工具
开发框架的时间,想测试单例和多例下对性能的影响,找了下没有特别简单易用的测试工具. 估摸着搞了一个小工具. 针对.net Framework的2.0,3.5,4.0版本. WebHttpTest2.0 ...
- [转]移动端web页面使用字体的思考
一直不知道手机端用的什么字体,只是觉得类似雅黑,直到有一次设计师问到设计移动web页面该用什么字体才严肃地想起这个问题. 前人已栽树,后人我就直接转来吧…… 回想2年前刚开始接触手机项目,接到PSD稿 ...
- web端测试和移动端测试的区别小记
转:http://qa.blog.163.com/blog/static/19014700220157128345318/ 之前一直参与web端的测试,最近一个项目加入了移动端,本人有幸参与了移动端的 ...
- 移动端web页面滚动不流畅,卡顿闪烁解决方案
移动端web页面滚动不流畅,卡顿闪烁解决方案 1.ios端的-webkit-overflow-scrolling属性可控制页面滚动效果,设置如下实现惯性滚动和弹性效果: -webkit-overf ...
- Web端测试和移动端测试
之前参加的项目有涉及Web端测试和移动端测试,简单的记录下他们之间的区别: 1.记录bug 在Web端可以通过系统自带的截图和QQ截图等方式来截取bug的图片,对于错误的地方可以用工具自带的标识来 ...
- APP端测试与web端测试的区别
想要知道APP端测试与web端测试的区别 ,那么我们就要先来了解,web和app的区别. web项目,一般都是b/s架构,基于浏览器的,而app则是c/s的,必须要有客户端.那么在系统测试测试的时候就 ...
- web端,app端,小程序端测试差异详解
前置解释:1.单纯从功能测试的层面上来讲的话,APP 测试.web 测试和H5测试在流程和功能测试上是没有区别的2.Web项目或pc项目都是在电脑上进行测试的.常见的PC项目架构有BS架构和CS架构的 ...
- 移动端WEB页面
百度前端技术学院第一阶段任务十一,关于移动端WEB页面布局,参考资料如下(都是一些网页链接): MDN:手机网页开发 MDN:在移动浏览器中使用viewport元标签控制布局 移动前端开发和 Web ...
随机推荐
- Kafka源码分析(三) - Server端 - 消息存储
系列文章目录 https://zhuanlan.zhihu.com/p/367683572 目录 系列文章目录 一. 业务模型 1.1 概念梳理 1.2 文件分析 1.2.1 数据目录 1.2.2 . ...
- APP测试的主要内容
一.功能性测试:依据需求相关的文档编写测试用例进行测试 二.兼容性测试 系统版本:Android,ios 分辨率 网络情况 可用工具:testin 三.安装,升级,卸载测试 首次安装,覆盖安装,卸载后 ...
- 一文讲全了Python 类和对象内容
摘要:这是一个关于 Python 类和对象的全部内容. 本文分享自华为云社区<从零开始学python | Python 类和对象-面向对象编程>,原文作者:Yuchuan . Pytho ...
- 向虚拟机注册钩子,实现Bean对象的初始化和销毁方法
作者:小傅哥 博客:https://bugstack.cn 沉淀.分享.成长,让自己和他人都能有所收获! 一.前言 有什么方式,能给代码留条活路? 有人说:人人都是产品经理,那你知道吗,人人也都可以是 ...
- 学习Qt Charts - 实时曲线
1.添加坐标轴 按照之前的一篇文章,先在工程中添加QChart.QChartView,代码如下: Dialog::Dialog(QWidget *parent) : QDialog(parent), ...
- 第三天编程学习Hello,World!
真正意义上迈入编程的大门--Hello,World! 新建一个文件夹(最好在桌面),方便存放代码 新建一个文件(如:Hello.txt) 改文件后缀名为.java 扩展文件得到Hello.java 编 ...
- AcWing 1141. 局域网
某个局域网内有n台计算机和m条 双向 网线,计算机的编号是1~n由于搭建局域网时工作人员的疏忽, 现在局域网内的连接形成了回路,我们知道如果局域网形成回路那么数据将不停的在回路内传输,造成网络卡的现象 ...
- UI自动化学习笔记- UnitTest单元测试框架详解
一.UnitTest基本使用 1. UnitTest框架 1.1 什么是框架 说明: 框架英文单词frame 为解决一类事情的功能集合 1.2什么是UnitTest框架 概念:UnitTest是pyt ...
- Go:go程序报错Cannot run program "C:\Users\dell\AppData\Local\Temp\___go_build_hello_go.exe" (in directory "…………"):该版本的 %1 与你运行的 Windows 版本不兼容。
问题截图 在go语言编译的时候,如果只是单单编译一个文件的话,package必须是main,意味着是可以单独编译的. 解决办法 修改为 package main 就可以 再次运行就可以啦. 文章转载至 ...
- BUU mrctf shit
吐槽:去年没写出的题,现在终于可以上手了,昂哥nb 动调发现直接卡着不动了,怀疑是反调试,果然有好几处反调试 这里选择就不先nop了,先让程序跑起来,然后attach,在输入函数下面下个断点,atta ...