python+unittest+logging+configparser配置读取+ddt数据驱动单元测试-注册案例

准备工作一:

准备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文件中

        

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值