准备工作一:
准备3个文件:3个文件全部放在同级目录下
register.py #要测试的代码文件
test_register.py #测试文件,用于编写测试用例
report_result.py #加载并执行测试用例,生成测试报告
准备excel文件 存放测试用例
准备工作二-测试代码及需求:
准备工作三-测试用例读取与数据驱动:
将测试数据根据上图整理至excel表格中 ,并放在测试代码的同级目录下
准备工作四-创建配置文件名为log.ini
[log] name=handler level=INFO filename=file_name
参考如下图:
实操:
第一步:读取测试用例+ddt数据驱动
import unittest #导入unitest框架 用于加载测试用例及执行测试 from test01.register import register #导入注册功能模块 import ddt # 数据驱动 from openpyxl import load_workbook # 导入openpyxl模块下的 操作excel文件的对象用于打开excel文件读取测试用例 import os.path wb = load_workbook('测试用例.xlsx') # 加载测试用例excel文件,返回工作簿对象 # print(list(wb)) sh = wb['register'] # 获取其中一个的对象 例如register这个表的对象 list_dates = list(sh.values) # 获取到该表里所有的数据,并以列表的形式返回,values:以行的形式返回多行数据 key = list_dates[0] # 列表的第一行数据是key 用于后面组合字典 list_data = [] # 定义空字典,用于存放组合好的字典 for value in list_dates[1:]: # 遍历从第二行开始往后的数据,每行数据都是value data_case = dict(zip(key, value)) # 使用zip函数将 key(首行)中的数据与Value(其他行) 中的每个值一一对应形成一个键值对形式的字典 list_data.append(data_case) @ddt.ddt #加载数据驱动 class TestRegister(unittest.TestCase): @ddt.data(*list_data) # 解包列表中的测试用例数据,传递给 测试函数中的第二个参数,解包的每个用例数据都是用一个字典对象存储的 def test_register(self, data): res = register(str(data['user']), str(data['password1']), str(data['password2']))#int类型没有len,所以没有长度概念,如果不强转成str类型则会抛出指定异常 expected = data['expected'] self.assertEqual(eval(expected), res, '该条用例运行失败了~') # 注意:这里引用eval方法,即将遍历出来的首行key对应的value值脱去一层字符串的外衣,使其变成字典类型与res实际结果进行比对,否则数据类型不一致也会断言失败 if __name__ == '__main__':#测试代码,用于测试执行当前的测试代码,避免其他已导入该模块的文件执行此条代码 unittest.main() pass
第二步:封装好的加载excel数据加载类
#===========================封装好的日志提取器========================== import os.path #导入os.path模块,用于文件操作 import openpyxl #导入openpyxl,用户加载并读取excel文件数据 from openpyxl import load_workbook #导入openpyxl里面的load_workbook,用于加载excel中数据 from logging日志的配置和应用.handlerlog import HandlerLog from logging日志的配置和应用.handlerlog import logger #定义一个满足用户使用的类,测试用例数据获取类 #1、初始化excel路径,根据路径判断文件是否存在,后缀名是否xlsx,存在的话打开并加载该工作簿,不存在则抛出异常 #2、实现方法:读取表单数据,返回一个列表,列表里面都是字典类型 #如果一个变量的值,在类的多个方法当中都要使用得到,那么就要在__init__方法中设置为实例属性为self. #类与上面的代码?注释之间隔2行。 class HandelExcel: #定义一个类 #类内部方法之间,用一行隔开。 def __init__(self,excel_path:str): # 判断是否是字符串类型 #if not isinstance(excel_path,str) self.sh = None self.wb = None logger.info('加载个文件,文件名{}'.format(excel_path)) try: self.wb = load_workbook(excel_path) #实例化工作簿对象 并加载excel_path except: logger.exception('文件加载失败') raise def select_sheets(self,sheet_name): logger.info('获取到当前页面{}'.format(sheet_name)) self.sh = self.wb[sheet_name] def get_values(self): list_dates = list(self.sh.values) # 获取到该表里所有的数据,并以列表的形式返回,values:以行的形式返回多行数据 logger.info('获取到表里所有的数据{}'.format(list_dates)) key = list_dates[0] # 列表的第一行数据是key 用于后面组合字典 logger.info('获取到keys值为{}'.format(key)) list_data = [] # 定义空字典,用于存放组合好的字典 for value in list_dates[1:]: # 遍历从第二行开始往后的数据,每行数据都是value data_case = dict(zip(key, value)) # 使用zip函数将 key(首行)中的数据与Value(其他行) 中的每个值一一对应形成一个键值对形式的字典 list_data.append(data_case) logger.info('获取到测试用例数据{}'.format(list_data)) return list_data if __name__ == '__main__': #得到excel文件的路径 file_dir = os.path.dirname(os.path.abspath(__file__)) excel_path = os.path.join(file_dir,'测试用例.xlsx') #实例化handleExcel类 he = HandelExcel(excel_path) he.select_sheets('register') list_data = he.get_values() print(list_data)
第三步:封装好日志生成器:
#==========================封装好的日志收集器============================ import configparser import logging from configparser import ConfigParser #创建类,继承logging.Logger class HandlerLog(logging.Logger): def __init__(self, conf_path): #初始化,conf_path是配置文件的绝对路径,日志收集器name和设定的日志级别,产生的日志级别将大于等于设定的级别 self.conf = ConfigParser() #实例化ConfigParser这个类的属性 self.conf.read(conf_path,encoding='utf-8') #读取配置文件的内容,如果有中文则编码格式必须带有utf-8 super().__init__(self.conf.get('log','name'), self.conf.get('log','level')) #通过init属性里面传入的配置文件conf_path,再调用父类属性,通过.get的方式获取配置文件的name和level属性值 def handle_stream(self): formatter = logging.Formatter('%(asctime)s %(levelname)s %(filename)s %(lineno)d %(message)s') #将日志格式化 handler_stream = logging.StreamHandler() #将格式化输出格式作为参数传给输出渠道 handler_stream.setFormatter(formatter) self.addHandler(handler_stream) #将输出渠道绑定到日志收集器上 def handle_file(self): formatter = logging.Formatter('%(asctime)s %(levelname)s %(filename)s %(lineno)d %(message)s') # 将日志格式化 handler_file = logging.FileHandler(self.conf.get('log','filename'), encoding='utf-8') #创建日志渠道类,将日志输出到filename对应的值里 也就是读取配置文件log.ini的filename值 handler_file.setFormatter(formatter) #设定日志格式 self.addHandler(handler_file) #将输出渠道绑定到日志收集器上 logger = HandlerLog('log.ini') logger.handle_file() logger.handle_stream() if __name__ == '__main__': logger = HandlerLog('log.ini') logger.handle_file() logger.handle_stream()
第四步:创建main提取测试用例、日志收集器、excel数据加载与读取的操作类
导入写好的测试用例读取+ddt数据驱动模块及类、日志收集器的logger实例、excelhandel类实例及数据读取方法,使它们关联到一起并输出日志
import os import os.path from test01.test_new_register import TestRegister from test01.HandelExcel import HandelExcel #把封装好的HandelExcel类导过来使用 from logging日志的配置和应用.handlerlog import logger #把封装好的日志收集器类导过来使用 #得到excel文件的路径 file_dir = os.path.dirname(os.path.abspath(__file__)) excel_path = os.path.join(file_dir,'测试用例.xlsx') #实例化handleExcel类 he = HandelExcel(excel_path) he.select_sheets('register') list_data = he.get_values()
第五步:生成测试报告
有三种测试报告,这里仅写1种常用的:BeautifulReport
可以去pypi寻找并下载,使用pip下载:
pip install HTMLTestReport
pip install BeautifulReport
如果下载过程中报time out,建议去百度搜索pip国内镜像源下载
BeautifulReport测试报告生成:
import unittest #导入unittest框架,用于加载及执行测试用例 from BeautifulReport.BeautifulReport import BeautifulReport #导入BeautifulReport测试模板 s = unittest.TestLoader().discover(r'C:/Users/cfs/PycharmProjects/pythonProject4/test01') #加载该目录下的测试用例文件,默认去找test*_开头的文件,并加载到套件中 runner = BeautifulReport(s) #创建BeautifulReport的对象,并将套件传入这个对象里 runner.report(description='描述', filename='beautiful_report.html', report_dir='') #引用report方法创建测试报告 filename如果只填写文件名称+后缀名,即生成报告在当前路径,如果想要生成在其它路径则需要填写绝对路径+文件名+后缀名 见到控制台下面输出说明生成成功, 点击进入到生成的测试报告文件,自定义选择浏览器并打开 并输出测试日志到控制台与log文件中