一、已登录页面 -->新建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页面端测试的更多相关文章

  1. PO封装设计模式 -- App移动端测试

    前言: 一.App_Po 封装 (用互联网上随便一个app进行) base 存放的是页面基础类,后续的类需继承基础类 common 存放的是公共部分,测试固件分离部分,新增截图功能部分 Data 存放 ...

  2. Web页面速度测试工具

    开发框架的时间,想测试单例和多例下对性能的影响,找了下没有特别简单易用的测试工具. 估摸着搞了一个小工具. 针对.net Framework的2.0,3.5,4.0版本. WebHttpTest2.0 ...

  3. [转]移动端web页面使用字体的思考

    一直不知道手机端用的什么字体,只是觉得类似雅黑,直到有一次设计师问到设计移动web页面该用什么字体才严肃地想起这个问题. 前人已栽树,后人我就直接转来吧…… 回想2年前刚开始接触手机项目,接到PSD稿 ...

  4. web端测试和移动端测试的区别小记

    转:http://qa.blog.163.com/blog/static/19014700220157128345318/ 之前一直参与web端的测试,最近一个项目加入了移动端,本人有幸参与了移动端的 ...

  5. 移动端web页面滚动不流畅,卡顿闪烁解决方案

    移动端web页面滚动不流畅,卡顿闪烁解决方案   1.ios端的-webkit-overflow-scrolling属性可控制页面滚动效果,设置如下实现惯性滚动和弹性效果: -webkit-overf ...

  6. Web端测试和移动端测试

    之前参加的项目有涉及Web端测试和移动端测试,简单的记录下他们之间的区别:   1.记录bug 在Web端可以通过系统自带的截图和QQ截图等方式来截取bug的图片,对于错误的地方可以用工具自带的标识来 ...

  7. APP端测试与web端测试的区别

    想要知道APP端测试与web端测试的区别 ,那么我们就要先来了解,web和app的区别. web项目,一般都是b/s架构,基于浏览器的,而app则是c/s的,必须要有客户端.那么在系统测试测试的时候就 ...

  8. web端,app端,小程序端测试差异详解

    前置解释:1.单纯从功能测试的层面上来讲的话,APP 测试.web 测试和H5测试在流程和功能测试上是没有区别的2.Web项目或pc项目都是在电脑上进行测试的.常见的PC项目架构有BS架构和CS架构的 ...

  9. 移动端WEB页面

    百度前端技术学院第一阶段任务十一,关于移动端WEB页面布局,参考资料如下(都是一些网页链接): MDN:手机网页开发 MDN:在移动浏览器中使用viewport元标签控制布局 移动前端开发和 Web ...

随机推荐

  1. CUDA Pro:通过向量化内存访问提高性能

    CUDA Pro:通过向量化内存访问提高性能 许多CUDA内核受带宽限制,而新硬件中触发器与带宽的比率不断提高,导致带宽受限制的内核更多.这使得采取措施减轻代码中的带宽瓶颈非常重要.本文将展示如何在C ...

  2. MEMS传感器作为变革的驱动力

    MEMS sensors as drivers for change 物联网(IoT)正在改变与周围世界互动的方式.每个人,每件事,都是相互联系的,很快就会相互联系.微机电系统(MEMS)设备和传感器 ...

  3. pycham_编码格式设置,处理打印log乱码,处理读取配置文件报错

    一.打印日志乱码,处理设置如下: 二.配置文件读取方法因为gbk编码配置后需要同步修改 原报错信息:

  4. springmvc——CharacterEncodingFilter过滤器要放在所有过滤器前面

    CharacterEncodingFilter的拦截顺序必须是第一个,否则还是会出现乱码问题.这是因为 request对象的parameter并不是一开始就解析的,它是等你第一次调用getParame ...

  5. 2021年Wordpress博客搭建

    2021年WordPress博客搭建教程 这是一篇关于2021最新版的WP个人博客搭建教程.整篇文章会事无巨细的一步步讲述搭建博客的每一步. 0.前言 随着互联网和移动互联网的飞速发展,博客这一功能恍 ...

  6. noip2006 总结

    T1 能量项链 原题 在Mars星球上,每个Mars人都随身佩带着一串能量项链.在项链上有N颗能量珠.能量珠是一颗有头标记与尾标记的珠子,这些标记对应着某个正整数.并且,对于相邻的两颗珠子,前一颗珠子 ...

  7. NOIP模拟测试18「引子·可爱宝贝精灵·相互再归的鹅妈妈」

    待补 引子 题解 大模拟,注意细节 代码1 #include<bits/stdc++.h> using namespace std; int n,m;char a[1005][1005]; ...

  8. SprignBoot是如何访问工程目录下的静态资源?

    ​ 目录 1.牛刀小试 1.1 图片静态资源的访问 1.2 为静态资源添加访问前缀 1.3  WelCome Page 的奇妙跳转 2.那么,SpringBoot是如何做到的呢? ​​​​​​​ 1. ...

  9. linux常用命令及一些静态动态库相关知识

    1 查找然后grep,最后在复制到特定目录 find . -depth -name *.java | xargs grep -i lijiangtao | awk -F ":" ' ...

  10. Eureka中读写锁的奇思妙想,学废了吗?

    前言 很抱歉 好久没有更新文章了,最近的一篇原创还是在去年十月份,这个号确实荒废了好久,感激那些没有把我取消关注的小伙伴. 有读者朋友经常私信问我: "你号卖了?" "文 ...