从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)
数据生成器
- 数据生成器的作用是为测试提供必要测试数据(可变数据,非业务类型数据)
- 数据生成器采用的是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())#随机身份证号
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