淘先锋技术网

首页 1 2 3 4 5 6 7

从0开始搭建自动化测试框架

断言

作用: 通过集中封装形式将unittest断言方式进行封装,并将需要断言的结果以形式方式传入

import unittest

class TestAssert(unittest.TestCase):
    """
        封装常用断言方法,在编写测试用例时,可以通过传参调用各种断言方法
    """

    def is_xd(self, text, mb_text, content=""):
        """
        判断是否相等
        :param text: 实际文本
        :param mb_text: 目标文本
        :param content: 断言结果
        :return:
        """
        self.assertEqual(text, mb_text, msg=content)

数据生成器

  1. 数据生成器的作用是为测试提供必要测试数据(可变数据,非业务类型数据)
  2. 数据生成器采用的是Faker
什么是faker
	    Faker是一个Python包,开源的GITHUB项目,主要用来创建伪数据,使用Faker包,
	    无需再手动生成或者手写随机数来生成数据,只需要调用Faker提供的方法,即可完成数据的生成。
Faker安装
	pip install Faker
Faker使用(可以将faker封装成数据生成方法并将数据返回出来既可)
	import faker
	
	# 初始化参数locale:为生成数据的文化选项,默认为en_US,zh_CN汉化后,才能生成相对应的随机信息(比如:名字,地址,邮编,城市,省份等)
	f = faker.Faker(locale='zh_CN)
	
	print(f.name())#姓名
	print(f.credit_card_number())#随机信用卡号
	print(f.email())#随机信用卡号
	print(f.ipv4())#随机ip地址
	print(f.user_name())#随机用户名
	print(f.phone_number())#随机电话
	print(f.ssn())#随机身份证号	    

faker常用函数:参考

	country():国家
	province():省份
	city_suffix():市,县
	district():区
	street_address():街道地址
	street_name():街道名
	street_suffix():街、路
	country_code():国家编码
	postcode():邮编
	geo_coordinate():地理坐标
	longitude():经度
	latitude():纬度
	lexify():替换所有问号?带有随机事件
	numerify():生成三位随机数
	random_digit():生成0~9随机数
	random_digit_not_null():生成1~9的随机数
	random_element():生成随机字母
	random_int():随机数字,默认0~9999,可通过min,max参数修改
	random_letter():随机字母
	random_number():随机数字,参数digits设置生成的数字位数
	color_name():随机颜色名
	hex_color():随机HEX颜色
	rgb_color():随机RGB颜色
	safe_color_name():随机安全色名
	safe_hex_color():随机安全HEX颜色
	bs():随机公司服务名
	company():随机公司名(长)
	company_prefix():随机公司名(短)
	company_suffix():公司性质
	credit_card_expire():随机信用卡到期日
	credit_card_full():生成完整信用卡信息
	credit_card_number():信用卡号
	credit_card_provider():信用卡类型
	credit_card_security_code():信用卡安全码
	currency_code():货币编码
	am_pm():AM/PM
	century():随机世纪
	date():随机日期
	date_between():随机生成指定范围内日期,参数:start_date,end_date
	date_between_dates():随机生成指定范围内日期,用法同上
	date_object():随机生产从1970-1-1到指定日期的随机日期。
	date_this_month():
	date_this_year():
	date_time():随机生成指定时间(1970年1月1日至今)
	date_time_ad():生成公元1年到现在的随机时间
	date_time_between():用法同dates
	future_date():未来日期
	future_datetime():未来时间
	month():随机月份
	month_name():随机月份(英文)
	past_date():随机生成已经过去的日期
	past_datetime():随机生成已经过去的时间
	time():随机24小时时间
	timedelta():随机获取时间差
	time_object():随机24小时时间,time对象
	time_series():随机TimeSeries对象
	timezone():随机时区
	unix_time():随机Unix时间
	year():随机年份
	file_extension():随机文件扩展名
	file_name():随机文件名(包含扩展名,不包含路径)
	file_path():随机文件路径(包含文件名,扩展名)
	mime_type():随机mime Type
	ascii_company_email():随机ASCII公司邮箱名
	ascii_email():随机ASCII邮箱
	ascii_free_email():
	ascii_safe_email():
	company_email():
	domain_name():生成域名
	domain_word():域词(即,不包含后缀)
	email():
	free_email():
	free_email_domain():
	f.safe_email():安全邮箱
	f.image_url():随机URL地址
	ipv4():随机IP4地址
	ipv6():随机IP6地址
	mac_address():随机MAC地址
	tld():网址域名后缀
	uri():随机URI地址
	uri_extension():网址文件后缀
	uri_page():网址文件(不包含后缀)
	uri_path():网址文件路径(不包含文件名)
	url():随机URL地址
	user_name():随机用户名
	isbn10():随机ISBN(10位)
	isbn13():随机ISBN(13位)
	job():随机职位
	paragraph():随机生成一个段落
	paragraphs():随机生成多个段落,通过参数nb来控制段落数,返回数组
	sentence():随机生成一句话
	sentences():随机生成多句话,与段落类似
	text():随机生成一篇文章
	word():随机生成词语
	words():随机生成多个词语,用法与段落,句子,类似
	binary():随机生成二进制编码
	boolean():True/False
	language_code():随机生成两位语言编码
	locale():随机生成语言/国际 信息
	md5():随机生成MD5
	null_boolean():NULL/True/False
	password():随机生成密码,
		可选参数:
			length:密码长度;
			special_chars:是否能使用特殊字符;
			digits:是否包含数字;
			upper_case:是否包含大写字母;
			lower_case:是否包含小写字母
	sha1():随机SHA1
	sha256():随机SHA256
	uuid4():随机UUID
	first_name():
	first_name_female():女性名
	first_name_male():男性名
	first_romanized_name():罗马名
	last_name():
	last_name_female():女
	last_name_male():男
	last_romanized_name():
	name():随机生成姓名
	name_female():男性姓名
	name_male():女性姓名
	romanized_name():罗马名
	msisdn():移动台国际用户识别码,即移动用户的ISDN号码
	phone_number():随机生成手机号
	phonenumber_prefix():随机生成手机号段
	profile():随机生成档案信息
	simple_profile():随机生成简单档案信息
	ssn():生成身份证号
	chrome():随机生成Chrome的浏览器user_agent信息
	firefox():随机生成FireFox的浏览器user_agent信息
	internet_explorer():随机生成IE的浏览器user_agent信息
	opera():随机生成Opera的浏览器user_agent信息
	safari():随机生成Safari的浏览器user_agent信息
	linux_platform_token():随机Linux信息
	user_agent():随机user_agent信息

测试套件(unittest)

import time
import unittest
from comm import settings
from comm.log import Logger
from comm.emailA import send_mail
from comm.HTMLTestRunner2 import HTMLTestRunner
from comm.settings import CASE_DIR_KP, CASE_DIR_YX, CASE_DIR_YY, CASE_DIR_XS, CASE_DIR_Y


def run():
    """
    报告目录执行TestSuite
    """
    # 加载测试套件
    suite = unittest.TestSuite()
    # 加载测试用例
    loader = unittest.TestLoader()

    # 运营平台
    Logger.info("正在进行测试")
    report = settings.REPORT_PATH + "/test/report.html"
    with open(report, 'wb') as f:
        # 设置报告title 和 描述
        runner = HTMLTestRunner(f, verbosity=2, title="自动化测试框架", description="自动化测试框架")
        suite.addTest(loader.discover(CASE_DIR_YY, pattern='test*.py', top_level_dir=None))
        runner.run(suite)
    # 获取当日时间并进行格式化
    day_time = time.strftime('%Y-%m-%d', time.localtime(time.time()))
    # 发送邮件
    send_mail(day_time, report)

数据校验

作用: 通过递归形式进行数据循环判断(需要做好处理放置数据溢出)

def get_target_value(key, dic, tmp_list):
    """
    :param key: 目标key值
    :param dic: JSON数据
    :param tmp_list: 用于存储获取的数据
    :return: list
    """
    if not isinstance(dic, dict) or not isinstance(tmp_list, list):  # 对传入数据进行格式校验
        return 'argv[1] not an dict or argv[-1] not an list '

    if key in dic.keys():
        tmp_list.append(dic[key])  # 传入数据存在则存入tmp_list

    for value in dic.values():  # 传入数据不符合则对其value值进行遍历
        if isinstance(value, dict):
            get_target_value(key, value, tmp_list)  # 传入数据的value值是字典,则直接调用自身
        elif isinstance(value, (list, tuple)):
            _get_value(key, value, tmp_list)  # 传入数据的value值是列表或者元组,则调用_get_value

    return tmp_list


def _get_value(key, val, tmp_list):
    for val_ in val:
        if isinstance(val_, dict):
            get_target_value(key, val_, tmp_list)  # 传入数据的value值是字典,则调用get_target_value
        elif isinstance(val_, (list, tuple)):
            _get_value(key, val_, tmp_list)  # 传入数据的value值是列表或者元组,则调用自身

d = {
    "base_config": {
        "enforce": {
            "value": "0",
            "inherit": "0",
            "global": "0"
        },
        "modify": {
            "value": "0",
            "inherit": "0",
            "global": "0"
        }
    },
    "safe_control_list": {
        "list": [
            {
                "gid": "0",
                "gname": "全网计算机",
                "isactive": "1",
                "rule_id": "0",
                "rule_name": "请选择规则",
                "time_range": "所有时间",
                "time_range_id": "1",
                "policy_tpl": "33",
                "policy_tpl_id": "17",
                "isonline": "3",
                "priority": "1"
            }
        ]
    }
}

    def get_params(self, params, content):
        """
        在内容中获取某一值
        :return:
        """
        paramsVal = None
        if isinstance(content, str):
            try:
                content = json.loads(content)
            except:
                content = ""
        if isinstance(content, dict):
            paramsVal = self.getParamsRes(params, content)
        if isinstance(content, list):
            dict_data = {}
            for i in range(len(content)):
                try:
                    dict_data[str(i)] = eval(content[i])
                except:
                    dict_data[str(i)] = content[i]
            paramsVal = self.getParamsRes(params, dict_data)
        if paramsVal is None:
            return paramsVal
        else:
            if "$" + params == paramsVal:
                paramsVal = None
            return paramsVal

    def getParamsRes(self, params_name, dict_data, default=None):
        """
        获取返回参数
        :param params_name:
        :param dict_data:
        :param default:
        :return:
        """
        for k, v in dict_data.items():
            if k == params_name:
                return v
            else:
                if isinstance(v, dict):
                    ret = self.getParamsRes(params_name, v)
                    if ret is not default:
                        return ret
                if isinstance(v, list):
                    for i in v:
                        if isinstance(i, dict):
                            ret = self.getParamsRes(params_name, i)
                            if ret is not default:
                                return ret
                            else:
                                pass
        return default