mirror of
https://github.com/httprunner/httprunner.git
synced 2026-05-11 18:11:21 +08:00
HttpRunner 2.0 is comming!
1, new design for testcase format; 2, refactor testcase layer mechanism.
This commit is contained in:
@@ -1,7 +1,7 @@
|
||||
__title__ = 'HttpRunner'
|
||||
__description__ = 'One-stop solution for HTTP(S) testing.'
|
||||
__url__ = 'https://github.com/HttpRunner/HttpRunner'
|
||||
__version__ = '1.6.0.alpha'
|
||||
__version__ = '2.0.0.beta'
|
||||
__author__ = 'debugtalk'
|
||||
__author_email__ = 'mail@debugtalk.com'
|
||||
__license__ = 'MIT'
|
||||
|
||||
@@ -28,17 +28,17 @@ class HttpRunner(object):
|
||||
self.test_loader = unittest.TestLoader()
|
||||
self.summary = None
|
||||
|
||||
def _add_tests(self, testcases):
|
||||
def _add_tests(self, tests_mapping):
|
||||
""" initialize testcase with Runner() and add to test suite.
|
||||
|
||||
Args:
|
||||
testcases (list): parsed testcases list
|
||||
tests_mapping (dict): project info and testcases list.
|
||||
|
||||
Returns:
|
||||
unittest.TestSuite()
|
||||
|
||||
"""
|
||||
def _add_teststep(test_runner, config, teststep_dict):
|
||||
def _add_teststep(test_runner, teststep_dict):
|
||||
""" add teststep to testcase.
|
||||
"""
|
||||
def test(self):
|
||||
@@ -52,22 +52,16 @@ class HttpRunner(object):
|
||||
self.meta_data["validators"] = test_runner.evaluated_validators
|
||||
test_runner.http_client_session.init_meta_data()
|
||||
|
||||
try:
|
||||
teststep_dict["name"] = parser.parse_data(
|
||||
teststep_dict["name"],
|
||||
config.get("variables", {}),
|
||||
config.get("functions", {})
|
||||
)
|
||||
except exceptions.VariableNotFound:
|
||||
pass
|
||||
|
||||
test.__doc__ = teststep_dict["name"]
|
||||
# TODO: refactor
|
||||
test.__doc__ = teststep_dict.get("name") or teststep_dict.get("config", {}).get("name")
|
||||
return test
|
||||
|
||||
test_suite = unittest.TestSuite()
|
||||
for testcase in testcases:
|
||||
functions = tests_mapping.get("project_mapping", {}).get("functions", {})
|
||||
|
||||
for testcase in tests_mapping["testcases"]:
|
||||
config = testcase.get("config", {})
|
||||
test_runner = runner.Runner(config, self.http_client_session)
|
||||
test_runner = runner.Runner(config, functions, self.http_client_session)
|
||||
TestSequense = type('TestSequense', (unittest.TestCase,), {})
|
||||
|
||||
teststeps = testcase.get("teststeps", [])
|
||||
@@ -76,12 +70,12 @@ class HttpRunner(object):
|
||||
# suppose one testcase should not have more than 9999 steps,
|
||||
# and one step should not run more than 999 times.
|
||||
test_method_name = 'test_{:04}_{:03}'.format(index, times_index)
|
||||
test_method = _add_teststep(test_runner, config, teststep_dict)
|
||||
test_method = _add_teststep(test_runner, teststep_dict)
|
||||
setattr(TestSequense, test_method_name, test_method)
|
||||
|
||||
loaded_testcase = self.test_loader.loadTestsFromTestCase(TestSequense)
|
||||
setattr(loaded_testcase, "config", config)
|
||||
setattr(loaded_testcase, "teststeps", testcase.get("teststeps", []))
|
||||
setattr(loaded_testcase, "teststeps", teststeps)
|
||||
setattr(loaded_testcase, "runner", test_runner)
|
||||
test_suite.addTest(loaded_testcase)
|
||||
|
||||
@@ -140,49 +134,21 @@ class HttpRunner(object):
|
||||
|
||||
self.summary["details"].append(testcase_summary)
|
||||
|
||||
def _run_tests(self, testcases, mapping=None):
|
||||
def _run_tests(self, tests_mapping):
|
||||
""" start to run test with variables mapping.
|
||||
|
||||
Args:
|
||||
testcases (list): list of testcase_dict, each testcase is corresponding to a YAML/JSON file
|
||||
[
|
||||
{ # testcase data structure
|
||||
"config": {
|
||||
"name": "desc1",
|
||||
"path": "testcase1_path",
|
||||
"variables": [], # optional
|
||||
"request": {}, # optional
|
||||
"functions": {},
|
||||
"env": {},
|
||||
"def-api": {},
|
||||
"def-testcase": {}
|
||||
},
|
||||
"teststeps": [
|
||||
# teststep data structure
|
||||
{
|
||||
'name': 'test step desc2',
|
||||
'variables': [], # optional
|
||||
'extract': [], # optional
|
||||
'validate': [],
|
||||
'request': {},
|
||||
'function_meta': {}
|
||||
},
|
||||
teststep2 # another teststep dict
|
||||
]
|
||||
},
|
||||
testcase_dict_2 # another testcase dict
|
||||
]
|
||||
mapping (dict): if mapping is specified, it will override variables in config block.
|
||||
tests_mapping (dict): list of testcase_dict, each testcase is corresponding to a YAML/JSON file
|
||||
|
||||
Returns:
|
||||
instance: HttpRunner() instance
|
||||
|
||||
"""
|
||||
self.exception_stage = "parse tests"
|
||||
parsed_testcases_list = parser.parse_tests(testcases, mapping)
|
||||
parser.parse_tests(tests_mapping)
|
||||
|
||||
self.exception_stage = "add tests to test suite"
|
||||
test_suite = self._add_tests(parsed_testcases_list)
|
||||
test_suite = self._add_tests(tests_mapping)
|
||||
|
||||
self.exception_stage = "run test suite"
|
||||
results = self._run_suite(test_suite)
|
||||
@@ -207,16 +173,14 @@ class HttpRunner(object):
|
||||
self.exception_stage = "load tests"
|
||||
|
||||
if validator.is_testcases(path_or_testcases):
|
||||
if isinstance(path_or_testcases, dict):
|
||||
testcases = [path_or_testcases]
|
||||
else:
|
||||
testcases = path_or_testcases
|
||||
tests_mapping = path_or_testcases
|
||||
elif validator.is_testcase_path(path_or_testcases):
|
||||
testcases = loader.load_tests(path_or_testcases, dot_env_path)
|
||||
tests_mapping = loader.load_tests(path_or_testcases, dot_env_path)
|
||||
tests_mapping["project_mapping"]["variables"] = mapping or {}
|
||||
else:
|
||||
raise exceptions.ParamsError("invalid testcase path or testcases.")
|
||||
|
||||
return self._run_tests(testcases, mapping)
|
||||
return self._run_tests(tests_mapping)
|
||||
|
||||
def gen_html_report(self, html_report_name=None, html_report_template=None):
|
||||
""" generate html report and return report path.
|
||||
|
||||
@@ -77,17 +77,13 @@ def main_hrun():
|
||||
create_scaffold(project_name)
|
||||
exit(0)
|
||||
|
||||
try:
|
||||
runner = HttpRunner(
|
||||
failfast=args.failfast
|
||||
)
|
||||
runner.run(
|
||||
args.testcase_paths,
|
||||
dot_env_path=args.dot_env_path
|
||||
)
|
||||
except Exception:
|
||||
logger.log_error("!!!!!!!!!! exception stage: {} !!!!!!!!!!".format(runner.exception_stage))
|
||||
raise
|
||||
for path in args.testcase_paths:
|
||||
try:
|
||||
runner = HttpRunner(failfast=args.failfast)
|
||||
runner.run(path, dot_env_path=args.dot_env_path)
|
||||
except Exception:
|
||||
logger.log_error("!!!!!!!!!! exception stage: {} !!!!!!!!!!".format(runner.exception_stage))
|
||||
raise
|
||||
|
||||
if not args.no_html_report:
|
||||
runner.gen_html_report(
|
||||
|
||||
@@ -1,75 +1,59 @@
|
||||
# encoding: utf-8
|
||||
|
||||
import copy
|
||||
|
||||
from httprunner import exceptions, logger, parser, utils
|
||||
|
||||
|
||||
class Context(object):
|
||||
""" Manages context functions and variables.
|
||||
context has two levels, testcase and teststep.
|
||||
class SessionContext(object):
|
||||
""" HttpRunner session, store runtime variables.
|
||||
|
||||
Examples:
|
||||
>>> functions={...}
|
||||
>>> variables = {"SECRET_KEY": "DebugTalk"}
|
||||
>>> context = SessionContext(functions, variables)
|
||||
|
||||
Equivalent to:
|
||||
>>> context = SessionContext(functions)
|
||||
>>> context.update_seesion_variables(variables)
|
||||
|
||||
"""
|
||||
def __init__(self, variables=None, functions=None):
|
||||
""" init Context with testcase variables and functions.
|
||||
"""
|
||||
variables = variables or {}
|
||||
functions = functions or {}
|
||||
# testcase level context
|
||||
## TESTCASE_SHARED_VARIABLES_MAPPING and TESTCASE_SHARED_FUNCTIONS_MAPPING are unchangeable.
|
||||
self.TESTCASE_SHARED_VARIABLES_MAPPING = utils.ensure_mapping_format(variables)
|
||||
self.TESTCASE_SHARED_FUNCTIONS_MAPPING = functions
|
||||
def __init__(self, functions, variables=None):
|
||||
self.session_variables_mapping = utils.ensure_mapping_format(variables or {})
|
||||
self.FUNCTIONS_MAPPING = functions
|
||||
self.teststep_variables_mapping = {}
|
||||
self.init_teststep_variables()
|
||||
|
||||
# testcase level request, will not change
|
||||
self.TESTCASE_SHARED_REQUEST_MAPPING = {}
|
||||
|
||||
self.evaluated_validators = []
|
||||
self.init_context_variables(level="testcase")
|
||||
|
||||
def init_context_variables(self, level="testcase"):
|
||||
""" initialize testcase/teststep context
|
||||
def init_teststep_variables(self, variables_mapping=None):
|
||||
""" init teststep variables, called when each teststep(api) starts.
|
||||
variables_mapping will be evaluated first.
|
||||
|
||||
Args:
|
||||
level (enum): "testcase" or "teststep"
|
||||
|
||||
"""
|
||||
if level == "testcase":
|
||||
# testcase level runtime context, will be updated with extracted variables in each teststep.
|
||||
self.testcase_runtime_variables_mapping = copy.deepcopy(self.TESTCASE_SHARED_VARIABLES_MAPPING)
|
||||
|
||||
# teststep level context, will be altered in each teststep.
|
||||
# teststep config shall inherit from testcase configs,
|
||||
# but can not change testcase configs, that's why we use copy.deepcopy here.
|
||||
self.teststep_variables_mapping = copy.deepcopy(self.testcase_runtime_variables_mapping)
|
||||
|
||||
def update_context_variables(self, variables, level):
|
||||
""" update context variables, with level specified.
|
||||
|
||||
Args:
|
||||
variables (list/OrderedDict): testcase config block or teststep block
|
||||
variables_mapping (dict/list)
|
||||
[
|
||||
{"TOKEN": "debugtalk"},
|
||||
{"random": "${gen_random_string(5)}"},
|
||||
{"json": {'name': 'user', 'password': '123456'}},
|
||||
{"md5": "${gen_md5($TOKEN, $json, $random)}"}
|
||||
{"data": '{"name": "user", "password": "123456"}'},
|
||||
{"authorization": "${gen_md5($TOKEN, $data, $random)}"}
|
||||
]
|
||||
OrderDict({
|
||||
"TOKEN": "debugtalk",
|
||||
"random": "${gen_random_string(5)}",
|
||||
"json": {'name': 'user', 'password': '123456'},
|
||||
"md5": "${gen_md5($TOKEN, $json, $random)}"
|
||||
})
|
||||
level (enum): "testcase" or "teststep"
|
||||
|
||||
"""
|
||||
variables_mapping = utils.ensure_mapping_format(variables)
|
||||
|
||||
variables_mapping = variables_mapping or {}
|
||||
variables_mapping = utils.ensure_mapping_format(variables_mapping)
|
||||
for variable_name, variable_value in variables_mapping.items():
|
||||
variable_eval_value = self.eval_content(variable_value)
|
||||
variable_value = self.eval_content(variable_value)
|
||||
self.update_teststep_variables(variable_name, variable_value)
|
||||
|
||||
if level == "testcase":
|
||||
self.testcase_runtime_variables_mapping[variable_name] = variable_eval_value
|
||||
self.teststep_variables_mapping.update(self.session_variables_mapping)
|
||||
|
||||
self.update_teststep_variables_mapping(variable_name, variable_eval_value)
|
||||
def update_teststep_variables(self, variable_name, variable_value):
|
||||
""" update teststep variables, these variables are only valid in the current teststep.
|
||||
"""
|
||||
self.teststep_variables_mapping[variable_name] = variable_value
|
||||
|
||||
def update_seesion_variables(self, variables_mapping):
|
||||
""" update session with extracted variables mapping.
|
||||
these variables are valid in the whole running session.
|
||||
"""
|
||||
variables_mapping = utils.ensure_mapping_format(variables_mapping)
|
||||
self.session_variables_mapping.update(variables_mapping)
|
||||
self.teststep_variables_mapping.update(self.session_variables_mapping)
|
||||
|
||||
def eval_content(self, content):
|
||||
""" evaluate content recursively, take effect on each variable and function in content.
|
||||
@@ -78,50 +62,9 @@ class Context(object):
|
||||
return parser.parse_data(
|
||||
content,
|
||||
self.teststep_variables_mapping,
|
||||
self.TESTCASE_SHARED_FUNCTIONS_MAPPING
|
||||
self.FUNCTIONS_MAPPING
|
||||
)
|
||||
|
||||
def update_testcase_runtime_variables_mapping(self, variables):
|
||||
""" update testcase_runtime_variables_mapping with extracted vairables in teststep.
|
||||
|
||||
Args:
|
||||
variables (OrderDict): extracted variables in teststep
|
||||
|
||||
"""
|
||||
for variable_name, variable_value in variables.items():
|
||||
self.testcase_runtime_variables_mapping[variable_name] = variable_value
|
||||
self.update_teststep_variables_mapping(variable_name, variable_value)
|
||||
|
||||
def update_teststep_variables_mapping(self, variable_name, variable_value):
|
||||
""" bind and update testcase variables mapping
|
||||
"""
|
||||
self.teststep_variables_mapping[variable_name] = variable_value
|
||||
|
||||
def get_parsed_request(self, request_dict, level="teststep"):
|
||||
""" get parsed request with variables and functions.
|
||||
|
||||
Args:
|
||||
request_dict (dict): request config mapping
|
||||
level (enum): "testcase" or "teststep"
|
||||
|
||||
Returns:
|
||||
dict: parsed request dict
|
||||
|
||||
"""
|
||||
if level == "testcase":
|
||||
# testcase config request dict has been parsed in parse_tests
|
||||
self.TESTCASE_SHARED_REQUEST_MAPPING = copy.deepcopy(request_dict)
|
||||
return self.TESTCASE_SHARED_REQUEST_MAPPING
|
||||
|
||||
else:
|
||||
# teststep
|
||||
return self.eval_content(
|
||||
utils.deep_update_dict(
|
||||
copy.deepcopy(self.TESTCASE_SHARED_REQUEST_MAPPING),
|
||||
request_dict
|
||||
)
|
||||
)
|
||||
|
||||
def __eval_check_item(self, validator, resp_obj):
|
||||
""" evaluate check item in validator.
|
||||
|
||||
@@ -183,7 +126,7 @@ class Context(object):
|
||||
"""
|
||||
# TODO: move comparator uniform to init_test_suites
|
||||
comparator = utils.get_uniform_comparator(validator_dict["comparator"])
|
||||
validate_func = parser.get_mapping_function(comparator, self.TESTCASE_SHARED_FUNCTIONS_MAPPING)
|
||||
validate_func = parser.get_mapping_function(comparator, self.FUNCTIONS_MAPPING)
|
||||
|
||||
check_item = validator_dict["check"]
|
||||
check_value = validator_dict["check_value"]
|
||||
|
||||
@@ -273,15 +273,21 @@ def load_debugtalk_functions():
|
||||
## testcase loader
|
||||
###############################################################################
|
||||
|
||||
def _load_teststeps(test_block, project_mapping):
|
||||
""" load teststeps with api/testcase references
|
||||
project_mapping = {}
|
||||
tests_def_mapping = {
|
||||
"api": {},
|
||||
"testcases": {}
|
||||
}
|
||||
|
||||
def load_teststep(raw_stepinfo):
|
||||
""" load teststep with api/testcase/proc references
|
||||
|
||||
Args:
|
||||
test_block (dict): test block content, maybe in 3 formats.
|
||||
raw_stepinfo (dict): teststep data, maybe in 3 formats.
|
||||
# api reference
|
||||
{
|
||||
"name": "add product to cart",
|
||||
"api": "api_add_cart()",
|
||||
"api": "api_add_cart",
|
||||
"variables": [],
|
||||
"validate": [],
|
||||
"extract": []
|
||||
@@ -289,7 +295,7 @@ def _load_teststeps(test_block, project_mapping):
|
||||
# testcase reference
|
||||
{
|
||||
"name": "add product to cart",
|
||||
"suite": "create_and_check()",
|
||||
"testcase": "create_and_check",
|
||||
"variables": []
|
||||
}
|
||||
# define directly
|
||||
@@ -304,33 +310,43 @@ def _load_teststeps(test_block, project_mapping):
|
||||
Returns:
|
||||
list: loaded teststeps list
|
||||
|
||||
"""
|
||||
teststeps = []
|
||||
Args:
|
||||
raw_stepinfo (dict): teststep info
|
||||
|
||||
"""
|
||||
# reference api
|
||||
if "api" in test_block:
|
||||
ref_call = test_block.pop("api")
|
||||
def_block = _get_block_by_name(ref_call, "def-api", project_mapping)
|
||||
extended_block = _extend_block(test_block, def_block)
|
||||
teststeps.append(extended_block)
|
||||
if "api" in raw_stepinfo:
|
||||
api_name = raw_stepinfo["api"]
|
||||
raw_stepinfo["api_def"] = _get_api_definition(api_name)
|
||||
|
||||
# TODO: reference proc functions
|
||||
elif "func" in raw_stepinfo:
|
||||
pass
|
||||
|
||||
# reference testcase
|
||||
elif "suite" in test_block: # TODO: replace suite with testcase
|
||||
ref_call = test_block.pop("suite")
|
||||
def_block = _get_block_by_name(ref_call, "def-testcase", project_mapping)
|
||||
# TODO: bugfix lost block config variables
|
||||
for teststep in def_block["teststeps"]:
|
||||
_teststeps = _load_teststeps(teststep, project_mapping)
|
||||
teststeps.extend(_teststeps)
|
||||
elif "testcase" in raw_stepinfo:
|
||||
testcase_path = raw_stepinfo["testcase"]
|
||||
|
||||
if testcase_path not in tests_def_mapping["testcases"]:
|
||||
testcase_path = os.path.join(
|
||||
project_mapping["PWD"],
|
||||
testcase_path
|
||||
)
|
||||
testcase_dict = load_testcase(load_file(testcase_path))
|
||||
tests_def_mapping[testcase_path] = testcase_dict
|
||||
else:
|
||||
testcase_dict = tests_def_mapping[testcase_path]
|
||||
|
||||
raw_stepinfo["testcase_def"] = testcase_dict
|
||||
|
||||
# define directly
|
||||
else:
|
||||
teststeps.append(test_block)
|
||||
pass
|
||||
|
||||
return teststeps
|
||||
return raw_stepinfo
|
||||
|
||||
|
||||
def _load_testcase(raw_testcase, project_mapping):
|
||||
def load_testcase(raw_testcase):
|
||||
""" load testcase/testsuite with api/testcase references
|
||||
|
||||
Args:
|
||||
@@ -352,20 +368,18 @@ def _load_testcase(raw_testcase, project_mapping):
|
||||
"test": {...}
|
||||
}
|
||||
]
|
||||
project_mapping (dict): project_mapping
|
||||
|
||||
Returns:
|
||||
dict: loaded testcase content
|
||||
{
|
||||
"name": "XYZ",
|
||||
"config": {},
|
||||
"teststeps": [teststep11, teststep12]
|
||||
}
|
||||
|
||||
"""
|
||||
loaded_testcase = {
|
||||
"config": {},
|
||||
"teststeps": []
|
||||
}
|
||||
config = {}
|
||||
teststeps = []
|
||||
|
||||
for item in raw_testcase:
|
||||
# TODO: add json schema validation
|
||||
@@ -377,290 +391,38 @@ def _load_testcase(raw_testcase, project_mapping):
|
||||
raise exceptions.FileFormatError("Testcase format error: {}".format(item))
|
||||
|
||||
if key == "config":
|
||||
loaded_testcase["config"].update(test_block)
|
||||
config.update(test_block)
|
||||
|
||||
elif key == "test":
|
||||
loaded_testcase["teststeps"].extend(_load_teststeps(test_block, project_mapping))
|
||||
teststeps.append(load_teststep(test_block))
|
||||
|
||||
else:
|
||||
logger.log_warning(
|
||||
"unexpected block key: {}. block key should only be 'config' or 'test'.".format(key)
|
||||
)
|
||||
|
||||
return loaded_testcase
|
||||
return {
|
||||
"config": config,
|
||||
"teststeps": teststeps
|
||||
}
|
||||
|
||||
|
||||
def _get_block_by_name(ref_call, ref_type, project_mapping):
|
||||
""" get test content by reference name.
|
||||
|
||||
Args:
|
||||
ref_call (str): call function.
|
||||
e.g. api_v1_Account_Login_POST($UserName, $Password)
|
||||
ref_type (enum): "def-api" or "def-testcase"
|
||||
project_mapping (dict): project_mapping
|
||||
def _get_api_definition(name):
|
||||
""" get api definition by name.
|
||||
|
||||
Returns:
|
||||
dict: api/testcase definition.
|
||||
|
||||
Raises:
|
||||
exceptions.ParamsError: call args number is not equal to defined args number.
|
||||
|
||||
"""
|
||||
function_meta = parser.parse_function(ref_call)
|
||||
func_name = function_meta["func_name"]
|
||||
call_args = function_meta["args"]
|
||||
block = _get_test_definition(func_name, ref_type, project_mapping)
|
||||
def_args = block.get("function_meta", {}).get("args", [])
|
||||
|
||||
if len(call_args) != len(def_args):
|
||||
err_msg = "{}: call args number is not equal to defined args number!\n".format(func_name)
|
||||
err_msg += "defined args: {}\n".format(def_args)
|
||||
err_msg += "reference args: {}".format(call_args)
|
||||
logger.log_error(err_msg)
|
||||
raise exceptions.ParamsError(err_msg)
|
||||
|
||||
args_mapping = {}
|
||||
for index, item in enumerate(def_args):
|
||||
if call_args[index] == item:
|
||||
continue
|
||||
|
||||
args_mapping[item] = call_args[index]
|
||||
|
||||
if args_mapping:
|
||||
block = parser.substitute_variables(block, args_mapping)
|
||||
|
||||
return block
|
||||
|
||||
|
||||
def _get_test_definition(name, ref_type, project_mapping):
|
||||
""" get expected api or testcase.
|
||||
|
||||
Args:
|
||||
name (str): api or testcase name
|
||||
ref_type (enum): "def-api" or "def-testcase"
|
||||
project_mapping (dict): project_mapping
|
||||
|
||||
Returns:
|
||||
dict: expected api/testcase info if found.
|
||||
dict: expected api definition if found.
|
||||
|
||||
Raises:
|
||||
exceptions.ApiNotFound: api not found
|
||||
exceptions.TestcaseNotFound: testcase not found
|
||||
|
||||
"""
|
||||
block = project_mapping.get(ref_type, {}).get(name)
|
||||
# NOTICE: avoid project_mapping been changed during iteration.
|
||||
block = copy.deepcopy(block)
|
||||
|
||||
if not block:
|
||||
err_msg = "{} not found!".format(name)
|
||||
if ref_type == "def-api":
|
||||
raise exceptions.ApiNotFound(err_msg)
|
||||
else:
|
||||
# ref_type == "def-testcase":
|
||||
raise exceptions.TestcaseNotFound(err_msg)
|
||||
|
||||
return block
|
||||
|
||||
|
||||
def _extend_block(ref_block, def_block):
|
||||
""" extend ref_block with def_block, ref_block will merge and override def_block.
|
||||
|
||||
Args:
|
||||
def_block (dict): api definition dict.
|
||||
ref_block (dict): reference block
|
||||
|
||||
Returns:
|
||||
dict: extended reference block.
|
||||
|
||||
Examples:
|
||||
>>> def_block = {
|
||||
"name": "get token 1",
|
||||
"request": {...},
|
||||
"validate": [{'eq': ['status_code', 200]}]
|
||||
}
|
||||
>>> ref_block = {
|
||||
"name": "get token 2",
|
||||
"extract": [{"token": "content.token"}],
|
||||
"validate": [{'eq': ['status_code', 201]}, {'len_eq': ['content.token', 16]}]
|
||||
}
|
||||
>>> _extend_block(def_block, ref_block)
|
||||
{
|
||||
"name": "get token 2",
|
||||
"request": {...},
|
||||
"extract": [{"token": "content.token"}],
|
||||
"validate": [{'eq': ['status_code', 201]}, {'len_eq': ['content.token', 16]}]
|
||||
}
|
||||
|
||||
"""
|
||||
extended_block = {}
|
||||
|
||||
# override name
|
||||
extended_block["name"] = ref_block.pop("name", None) or def_block.pop("name", "")
|
||||
|
||||
# override variables
|
||||
def_variables = def_block.pop("variables", [])
|
||||
ref_variables = ref_block.pop("variables", [])
|
||||
extended_block["variables"] = _extend_variables(
|
||||
def_variables,
|
||||
ref_variables
|
||||
)
|
||||
|
||||
# merge & override validators
|
||||
def_validators = def_block.pop("validate", [])
|
||||
ref_validators = ref_block.pop("validate", [])
|
||||
extended_block["validate"] = _extend_validators(
|
||||
def_validators,
|
||||
ref_validators
|
||||
)
|
||||
|
||||
# merge & override extractors
|
||||
def_extrators = def_block.pop("extract", [])
|
||||
ref_extractors = ref_block.pop("extract", [])
|
||||
extended_block["extract"] = _extend_variables(
|
||||
def_extrators,
|
||||
ref_extractors
|
||||
)
|
||||
|
||||
# TODO: merge & override request
|
||||
def_request = def_block.pop("request", {})
|
||||
ref_request = ref_block.pop("request", {})
|
||||
extended_block["request"] = def_request
|
||||
|
||||
# merge & override setup_hooks
|
||||
def_setup_hooks = def_block.pop("setup_hooks", [])
|
||||
ref_setup_hooks = ref_block.pop("setup_hooks", [])
|
||||
extended_block["setup_hooks"] = list(set(def_setup_hooks + ref_setup_hooks))
|
||||
# merge & override teardown_hooks
|
||||
def_teardown_hooks = def_block.pop("teardown_hooks", [])
|
||||
ref_teardown_hooks = ref_block.pop("teardown_hooks", [])
|
||||
extended_block["teardown_hooks"] = list(set(def_teardown_hooks + ref_teardown_hooks))
|
||||
|
||||
# TODO: extend with other ref block items, e.g. times
|
||||
# extended_block.update(def_block)
|
||||
extended_block.update(ref_block)
|
||||
|
||||
return extended_block
|
||||
|
||||
|
||||
def _convert_validators_to_mapping(validators):
|
||||
""" convert validators list to mapping.
|
||||
|
||||
Args:
|
||||
validators (list): validators in list
|
||||
|
||||
Returns:
|
||||
dict: validators mapping, use (check, comparator) as key.
|
||||
|
||||
Examples:
|
||||
>>> validators = [
|
||||
{"check": "v1", "expect": 201, "comparator": "eq"},
|
||||
{"check": {"b": 1}, "expect": 200, "comparator": "eq"}
|
||||
]
|
||||
>>> _convert_validators_to_mapping(validators)
|
||||
{
|
||||
("v1", "eq"): {"check": "v1", "expect": 201, "comparator": "eq"},
|
||||
('{"b": 1}', "eq"): {"check": {"b": 1}, "expect": 200, "comparator": "eq"}
|
||||
}
|
||||
|
||||
"""
|
||||
validators_mapping = {}
|
||||
|
||||
for validator in validators:
|
||||
validator = parser.parse_validator(validator)
|
||||
|
||||
if not isinstance(validator["check"], collections.Hashable):
|
||||
check = json.dumps(validator["check"])
|
||||
else:
|
||||
check = validator["check"]
|
||||
|
||||
key = (check, validator["comparator"])
|
||||
validators_mapping[key] = validator
|
||||
|
||||
return validators_mapping
|
||||
|
||||
|
||||
def _extend_validators(def_validators, ref_validators):
|
||||
""" extend ref_validators with def_validators.
|
||||
ref_validators will merge and override def_validators.
|
||||
|
||||
Args:
|
||||
def_validators (list):
|
||||
ref_validators (list):
|
||||
|
||||
Returns:
|
||||
list: extended validators
|
||||
|
||||
Examples:
|
||||
>>> def_validators = [{'eq': ['v1', 200]}, {"check": "s2", "expect": 16, "comparator": "len_eq"}]
|
||||
>>> ref_validators = [{"check": "v1", "expect": 201}, {'len_eq': ['s3', 12]}]
|
||||
>>> _extend_validators(def_validators, ref_validators)
|
||||
[
|
||||
{"check": "v1", "expect": 201, "comparator": "eq"},
|
||||
{"check": "s2", "expect": 16, "comparator": "len_eq"},
|
||||
{"check": "s3", "expect": 12, "comparator": "len_eq"}
|
||||
]
|
||||
|
||||
"""
|
||||
if not def_validators:
|
||||
return ref_validators
|
||||
|
||||
elif not ref_validators:
|
||||
return def_validators
|
||||
|
||||
else:
|
||||
def_validators_mapping = _convert_validators_to_mapping(def_validators)
|
||||
ref_validators_mapping = _convert_validators_to_mapping(ref_validators)
|
||||
|
||||
def_validators_mapping.update(ref_validators_mapping)
|
||||
return list(def_validators_mapping.values())
|
||||
|
||||
|
||||
def _extend_variables(def_variables, ref_variables):
|
||||
""" extend ref_variables with def_variables.
|
||||
ref_variables will merge and override def_variables.
|
||||
|
||||
Args:
|
||||
def_variables (list):
|
||||
ref_variables (list):
|
||||
|
||||
Returns:
|
||||
list: extended variables
|
||||
|
||||
Examples:
|
||||
>>> def_variables = [{"var1": "val1"}, {"var2": "val2"}]
|
||||
>>> ref_variables = [{"var1": "val111"}, {"var3": "val3"}]
|
||||
>>> _extend_variables(def_variables, ref_variables)
|
||||
[
|
||||
{"var1": "val111"},
|
||||
{"var2": "val2"},
|
||||
{"var3": "val3"}
|
||||
]
|
||||
|
||||
"""
|
||||
if not def_variables:
|
||||
return ref_variables
|
||||
|
||||
elif not ref_variables:
|
||||
return def_variables
|
||||
|
||||
else:
|
||||
extended_variables_dict = OrderedDict()
|
||||
for def_variable in def_variables:
|
||||
var_name = list(def_variable.keys())[0]
|
||||
extended_variables_dict[var_name] = def_variable[var_name]
|
||||
|
||||
for ref_variable in ref_variables:
|
||||
if not ref_variable:
|
||||
continue
|
||||
var_name = list(ref_variable.keys())[0]
|
||||
extended_variables_dict[var_name] = ref_variable[var_name]
|
||||
|
||||
extended_variables = []
|
||||
for key, value in extended_variables_dict.items():
|
||||
extended_variables.append({key: value})
|
||||
|
||||
return extended_variables
|
||||
try:
|
||||
block = tests_def_mapping["api"][name]
|
||||
# NOTICE: avoid project_mapping been changed during iteration.
|
||||
return utils.deepcopy_dict(block)
|
||||
except KeyError:
|
||||
raise exceptions.ApiNotFound("{} not found!".format(name))
|
||||
|
||||
|
||||
def load_folder_content(folder_path):
|
||||
@@ -737,99 +499,16 @@ def load_api_folder(api_folder_path):
|
||||
for api_item in api_items:
|
||||
key, api_dict = api_item.popitem()
|
||||
|
||||
# TODO: replace def with api file path
|
||||
api_def = api_dict.pop("def")
|
||||
function_meta = parser.parse_function(api_def)
|
||||
func_name = function_meta["func_name"]
|
||||
# TODO: replace id with api file path
|
||||
api_id = api_dict.get("id")
|
||||
if api_id in api_definition_mapping:
|
||||
logger.log_warning("API definition duplicated: {}".format(api_id))
|
||||
|
||||
if func_name in api_definition_mapping:
|
||||
logger.log_warning("API definition duplicated: {}".format(func_name))
|
||||
|
||||
api_dict["function_meta"] = function_meta
|
||||
api_definition_mapping[func_name] = api_dict
|
||||
api_definition_mapping[api_id] = api_dict
|
||||
|
||||
return api_definition_mapping
|
||||
|
||||
|
||||
def load_test_folder(test_folder_path):
|
||||
""" load testcases definitions from folder.
|
||||
|
||||
Args:
|
||||
test_folder_path (str): testcases files folder.
|
||||
|
||||
testcase file should be in the following format:
|
||||
[
|
||||
{
|
||||
"config": {
|
||||
"def": "create_and_check",
|
||||
"request": {},
|
||||
"validate": []
|
||||
}
|
||||
},
|
||||
{
|
||||
"test": {
|
||||
"api": "get_user",
|
||||
"validate": []
|
||||
}
|
||||
}
|
||||
]
|
||||
|
||||
Returns:
|
||||
dict: testcases definition mapping.
|
||||
|
||||
{
|
||||
"create_and_check": [
|
||||
{"config": {}},
|
||||
{"test": {}},
|
||||
{"test": {}}
|
||||
],
|
||||
"tests/testcases/create_and_get.yml": [
|
||||
{"config": {}},
|
||||
{"test": {}},
|
||||
{"test": {}}
|
||||
]
|
||||
}
|
||||
|
||||
"""
|
||||
test_definition_mapping = {}
|
||||
|
||||
test_items_mapping = load_folder_content(test_folder_path)
|
||||
|
||||
for test_file_path, items in test_items_mapping.items():
|
||||
# TODO: add JSON schema validation
|
||||
|
||||
testcase = {
|
||||
"config": {},
|
||||
"teststeps": []
|
||||
}
|
||||
for item in items:
|
||||
key, block = item.popitem()
|
||||
|
||||
if key == "config":
|
||||
testcase["config"].update(block)
|
||||
|
||||
if "def" not in block:
|
||||
test_definition_mapping[test_file_path] = testcase
|
||||
continue
|
||||
|
||||
# TODO: replace def with testcase file path
|
||||
testcase_def = block.pop("def")
|
||||
function_meta = parser.parse_function(testcase_def)
|
||||
func_name = function_meta["func_name"]
|
||||
|
||||
if func_name in test_definition_mapping:
|
||||
logger.log_warning("API definition duplicated: {}".format(func_name))
|
||||
|
||||
testcase["function_meta"] = function_meta
|
||||
test_definition_mapping[func_name] = testcase
|
||||
else:
|
||||
# key == "test":
|
||||
### TODO: extend suite with api
|
||||
testcase["teststeps"].append(block)
|
||||
|
||||
return test_definition_mapping
|
||||
|
||||
|
||||
def load_debugtalk_py(start_path):
|
||||
""" locate debugtalk.py file and returns PWD and debugtalk.py functions.
|
||||
|
||||
@@ -878,22 +557,17 @@ def load_project_tests(test_path, dot_env_path=None):
|
||||
dict: project loaded api/testcases definitions, environments and debugtalk.py functions.
|
||||
|
||||
"""
|
||||
project_mapping = {}
|
||||
|
||||
# locate PWD and load debugtalk.py functions
|
||||
project_working_directory, debugtalk_functions = load_debugtalk_py(test_path)
|
||||
project_mapping["PWD"] = project_working_directory
|
||||
project_mapping["functions"] = debugtalk_functions
|
||||
|
||||
# load .env
|
||||
dot_env_path = dot_env_path or os.path.join(project_working_directory, ".env")
|
||||
project_mapping["env"] = load_dot_env_file(dot_env_path)
|
||||
|
||||
# load api/testcases
|
||||
project_mapping["def-api"] = load_api_folder(os.path.join(project_working_directory, "api"))
|
||||
# TODO: replace suite with testcases
|
||||
project_mapping["def-testcase"] = load_test_folder(os.path.join(project_working_directory, "suite"))
|
||||
|
||||
return project_mapping
|
||||
# load api
|
||||
tests_def_mapping["api"] = load_api_folder(os.path.join(project_working_directory, "api"))
|
||||
|
||||
|
||||
def load_tests(path, dot_env_path=None):
|
||||
@@ -901,54 +575,44 @@ def load_tests(path, dot_env_path=None):
|
||||
|
||||
Args:
|
||||
path (str/list): testcase file/foler path.
|
||||
path could be in several types:
|
||||
path could be in 2 types:
|
||||
- absolute/relative file path
|
||||
- absolute/relative folder path
|
||||
- list/set container with file(s) and/or folder(s)
|
||||
dot_env_path (str): specified .env file path
|
||||
|
||||
Returns:
|
||||
list: testcases list, each testcase is corresponding to a file
|
||||
[
|
||||
{ # testcase data structure
|
||||
"config": {
|
||||
"name": "desc1",
|
||||
"path": "testcase1_path",
|
||||
"variables": [], # optional
|
||||
"request": {}, # optional
|
||||
dict: tests mapping, include project_mapping and testcases.
|
||||
each testcase is corresponding to a file.
|
||||
{
|
||||
"project_mapping": {
|
||||
"PWD": "XXXXX",
|
||||
"functions": {},
|
||||
"env": {},
|
||||
"def-api": {},
|
||||
"def-testcase": {}
|
||||
"env": {}
|
||||
},
|
||||
"teststeps": [
|
||||
# teststep data structure
|
||||
{
|
||||
'name': 'test step desc1',
|
||||
'variables': [], # optional
|
||||
'extract': [], # optional
|
||||
'validate': [],
|
||||
'request': {},
|
||||
'function_meta': {}
|
||||
"testcases": [
|
||||
{ # testcase data structure
|
||||
"config": {
|
||||
"name": "desc1",
|
||||
"path": "testcase1_path",
|
||||
"variables": [], # optional
|
||||
},
|
||||
"teststeps": [
|
||||
# teststep data structure
|
||||
{
|
||||
'name': 'test step desc1',
|
||||
'variables': [], # optional
|
||||
'extract': [], # optional
|
||||
'validate': [],
|
||||
'request': {}
|
||||
},
|
||||
teststep2 # another teststep dict
|
||||
]
|
||||
},
|
||||
teststep2 # another teststep dict
|
||||
testcase_dict_2 # another testcase dict
|
||||
]
|
||||
},
|
||||
testcase_dict_2 # another testcase dict
|
||||
]
|
||||
}
|
||||
|
||||
"""
|
||||
if isinstance(path, (list, set)):
|
||||
testcases_list = []
|
||||
|
||||
for file_path in set(path):
|
||||
testcases = load_tests(file_path, dot_env_path)
|
||||
if not testcases:
|
||||
continue
|
||||
testcases_list.extend(testcases)
|
||||
|
||||
return testcases_list
|
||||
|
||||
if not os.path.exists(path):
|
||||
err_msg = "path not exist: {}".format(path)
|
||||
logger.log_error(err_msg)
|
||||
@@ -957,22 +621,41 @@ def load_tests(path, dot_env_path=None):
|
||||
if not os.path.isabs(path):
|
||||
path = os.path.join(os.getcwd(), path)
|
||||
|
||||
load_project_tests(path, dot_env_path)
|
||||
tests_mapping = {
|
||||
"project_mapping": project_mapping
|
||||
}
|
||||
|
||||
def load_test_file(path):
|
||||
raw_testcase = load_file(path)
|
||||
|
||||
try:
|
||||
testcase = load_testcase(raw_testcase)
|
||||
testcase["config"]["path"] = path
|
||||
except exceptions.FileFormatError:
|
||||
testcase = {}
|
||||
|
||||
return testcase
|
||||
|
||||
testcases_list = []
|
||||
|
||||
if os.path.isdir(path):
|
||||
files_list = load_folder_files(path)
|
||||
testcases_list = load_tests(files_list, dot_env_path)
|
||||
for path in files_list:
|
||||
testcase = load_test_file(path)
|
||||
if not testcase:
|
||||
continue
|
||||
testcases_list.append(testcase)
|
||||
|
||||
elif os.path.isfile(path):
|
||||
try:
|
||||
raw_testcase = load_file(path)
|
||||
project_mapping = load_project_tests(path, dot_env_path)
|
||||
testcase = _load_testcase(raw_testcase, project_mapping)
|
||||
testcase["config"]["path"] = path
|
||||
testcase["config"].update(project_mapping)
|
||||
testcases_list = [testcase]
|
||||
except exceptions.FileFormatError:
|
||||
testcases_list = []
|
||||
|
||||
return testcases_list
|
||||
testcase = load_test_file(path)
|
||||
if testcase:
|
||||
testcases_list.append(testcase)
|
||||
|
||||
tests_mapping["testcases"] = testcases_list
|
||||
|
||||
return tests_mapping
|
||||
|
||||
|
||||
def load_locust_tests(path, dot_env_path=None):
|
||||
@@ -999,7 +682,7 @@ def load_locust_tests(path, dot_env_path=None):
|
||||
|
||||
"""
|
||||
raw_testcase = load_file(path)
|
||||
project_mapping = load_project_tests(path, dot_env_path)
|
||||
load_project_tests(path, dot_env_path)
|
||||
|
||||
config = {}
|
||||
tests = []
|
||||
@@ -1009,10 +692,10 @@ def load_locust_tests(path, dot_env_path=None):
|
||||
if key == "config":
|
||||
config.update(test_block)
|
||||
elif key == "test":
|
||||
teststeps = _load_teststeps(test_block, project_mapping)
|
||||
weight = test_block.pop("weight", 1)
|
||||
teststep = load_teststep(test_block)
|
||||
weight = test_block.get("weight", 1)
|
||||
for _ in range(weight):
|
||||
tests.append(teststeps)
|
||||
tests.append(teststep)
|
||||
|
||||
# parse config variables
|
||||
raw_config_variables = config.get("variables", [])
|
||||
|
||||
@@ -3,6 +3,7 @@
|
||||
import ast
|
||||
import os
|
||||
import re
|
||||
from collections import OrderedDict
|
||||
|
||||
from httprunner import exceptions, utils
|
||||
from httprunner.compat import basestring, builtin_str, numeric_types, str
|
||||
@@ -516,7 +517,7 @@ def parse_data(content, variables_mapping=None, functions_mapping=None):
|
||||
for item in content
|
||||
]
|
||||
|
||||
if isinstance(content, dict):
|
||||
if isinstance(content, (dict, OrderedDict)):
|
||||
parsed_content = {}
|
||||
for key, value in content.items():
|
||||
parsed_key = parse_data(key, variables_mapping, functions_mapping)
|
||||
@@ -527,7 +528,7 @@ def parse_data(content, variables_mapping=None, functions_mapping=None):
|
||||
|
||||
if isinstance(content, basestring):
|
||||
# content is in string format here
|
||||
variables_mapping = variables_mapping or {}
|
||||
variables_mapping = utils.ensure_mapping_format(variables_mapping or {})
|
||||
functions_mapping = functions_mapping or {}
|
||||
content = content.strip()
|
||||
|
||||
@@ -541,110 +542,327 @@ def parse_data(content, variables_mapping=None, functions_mapping=None):
|
||||
return content
|
||||
|
||||
|
||||
def parse_tests(testcases, variables_mapping=None):
|
||||
""" parse testcases configs, including variables/parameters/name/request.
|
||||
def _extend_with_api(teststep_dict, api_def_dict):
|
||||
""" extend teststep with api definition, teststep will merge and override api definition.
|
||||
|
||||
Args:
|
||||
testcases (list): testcase list, with config unparsed.
|
||||
[
|
||||
{ # testcase data structure
|
||||
"config": {
|
||||
"name": "desc1",
|
||||
"path": "testcase1_path",
|
||||
"variables": {}, # optional
|
||||
"request": {} # optional
|
||||
"functions": {},
|
||||
"env": {},
|
||||
"def-api": {},
|
||||
"def-testcase": {}
|
||||
},
|
||||
"teststeps": [
|
||||
# teststep data structure
|
||||
{
|
||||
'name': 'test step desc2',
|
||||
'variables': [], # optional
|
||||
'extract': [], # optional
|
||||
'validate': [],
|
||||
'request': {},
|
||||
'function_meta': {}
|
||||
},
|
||||
teststep2 # another teststep dict
|
||||
]
|
||||
teststep_dict (dict): teststep block
|
||||
api_def_dict (dict): api definition
|
||||
|
||||
Returns:
|
||||
dict: extended teststep dict.
|
||||
|
||||
Examples:
|
||||
>>> api_def_dict = {
|
||||
"name": "get token 1",
|
||||
"request": {...},
|
||||
"validate": [{'eq': ['status_code', 200]}]
|
||||
}
|
||||
>>> teststep_dict = {
|
||||
"name": "get token 2",
|
||||
"extract": [{"token": "content.token"}],
|
||||
"validate": [{'eq': ['status_code', 201]}, {'len_eq': ['content.token', 16]}]
|
||||
}
|
||||
>>> _extend_with_api(teststep_dict, api_def_dict)
|
||||
{
|
||||
"name": "get token 2",
|
||||
"request": {...},
|
||||
"extract": [{"token": "content.token"}],
|
||||
"validate": [{'eq': ['status_code', 201]}, {'len_eq': ['content.token', 16]}]
|
||||
}
|
||||
|
||||
"""
|
||||
# override name
|
||||
api_def_name = api_def_dict.pop("name", "")
|
||||
teststep_dict["name"] = teststep_dict.get("name") or api_def_name
|
||||
|
||||
# override variables
|
||||
def_variables = api_def_dict.pop("variables", [])
|
||||
teststep_dict["variables"] = utils.extend_variables(
|
||||
def_variables,
|
||||
teststep_dict.get("variables", {})
|
||||
)
|
||||
|
||||
# merge & override validators TODO: relocate
|
||||
def_raw_validators = api_def_dict.pop("validate", [])
|
||||
ref_raw_validators = teststep_dict.get("validate", [])
|
||||
def_validators = [
|
||||
parse_validator(validator)
|
||||
for validator in def_raw_validators
|
||||
]
|
||||
ref_validators = [
|
||||
parse_validator(validator)
|
||||
for validator in ref_raw_validators
|
||||
]
|
||||
teststep_dict["validate"] = utils.extend_validators(
|
||||
def_validators,
|
||||
ref_validators
|
||||
)
|
||||
|
||||
# merge & override extractors
|
||||
def_extrators = api_def_dict.pop("extract", [])
|
||||
teststep_dict["extract"] = utils.extend_variables(
|
||||
def_extrators,
|
||||
teststep_dict.get("extract", [])
|
||||
)
|
||||
|
||||
# TODO: merge & override request
|
||||
teststep_dict["request"] = api_def_dict.pop("request", {})
|
||||
# base_url
|
||||
base_url = teststep_dict.pop("base_url", None)
|
||||
if base_url:
|
||||
teststep_dict["request"]["url"] = "{}/{}".format(base_url.rstrip("/"), teststep_dict["request"]["url"].lstrip("/"))
|
||||
|
||||
# verify
|
||||
if "verify" in teststep_dict:
|
||||
verify = teststep_dict.pop("verify")
|
||||
elif "verify" in api_def_dict:
|
||||
verify = api_def_dict.pop("verify")
|
||||
else:
|
||||
verify = True
|
||||
teststep_dict["request"]["verify"] = verify
|
||||
|
||||
# merge & override setup_hooks
|
||||
def_setup_hooks = api_def_dict.pop("setup_hooks", [])
|
||||
ref_setup_hooks = teststep_dict.get("setup_hooks", [])
|
||||
extended_setup_hooks = list(set(def_setup_hooks + ref_setup_hooks))
|
||||
if extended_setup_hooks:
|
||||
teststep_dict["setup_hooks"] = extended_setup_hooks
|
||||
# merge & override teardown_hooks
|
||||
def_teardown_hooks = api_def_dict.pop("teardown_hooks", [])
|
||||
ref_teardown_hooks = teststep_dict.get("teardown_hooks", [])
|
||||
extended_teardown_hooks = list(set(def_teardown_hooks + ref_teardown_hooks))
|
||||
if extended_teardown_hooks:
|
||||
teststep_dict["teardown_hooks"] = extended_teardown_hooks
|
||||
|
||||
# TODO: extend with other api definition items, e.g. times
|
||||
teststep_dict.update(api_def_dict)
|
||||
|
||||
return teststep_dict
|
||||
|
||||
|
||||
def _extend_with_testcase(teststep_dict, testcase_def_dict):
|
||||
""" extend teststep with testcase definition
|
||||
teststep will merge and override testcase config definition.
|
||||
|
||||
Args:
|
||||
teststep_dict (dict): teststep block
|
||||
testcase_def_dict (dict): testcase definition
|
||||
|
||||
Returns:
|
||||
dict: extended teststep dict.
|
||||
|
||||
"""
|
||||
# override testcase config variables
|
||||
testcase_def_dict["config"].setdefault("variables", {})
|
||||
testcase_def_variables = utils.ensure_mapping_format(testcase_def_dict["config"].get("variables", {}))
|
||||
testcase_def_variables.update(teststep_dict.pop("variables", {}))
|
||||
testcase_def_dict["config"]["variables"] = testcase_def_variables
|
||||
|
||||
# override base_url, verify
|
||||
# priority: testcase config > testsuite teststep
|
||||
teststep_base_url = teststep_dict.pop("base_url", None)
|
||||
teststep_verify = teststep_dict.pop("verify", True)
|
||||
testcase_def_dict["config"].setdefault("base_url", teststep_base_url)
|
||||
testcase_def_dict["config"].setdefault("verify", teststep_verify)
|
||||
|
||||
# override testcase config name, output, etc.
|
||||
testcase_def_dict["config"].update(teststep_dict)
|
||||
|
||||
teststep_dict.clear()
|
||||
teststep_dict.update(testcase_def_dict)
|
||||
|
||||
|
||||
def __parse_config(config, project_mapping):
|
||||
""" parse testcase config, include variables and name.
|
||||
"""
|
||||
# get config variables
|
||||
raw_config_variables = config.pop("variables", {})
|
||||
raw_config_variables_mapping = utils.ensure_mapping_format(raw_config_variables)
|
||||
override_variables = utils.deepcopy_dict(project_mapping.get("variables", {}))
|
||||
functions = project_mapping.get("functions", {})
|
||||
|
||||
# override testcase config variables with passed in variables
|
||||
for key, value in raw_config_variables_mapping.items():
|
||||
|
||||
if key in override_variables:
|
||||
# passed in
|
||||
continue
|
||||
else:
|
||||
# config variables
|
||||
try:
|
||||
parsed_value = parse_data(
|
||||
value,
|
||||
override_variables,
|
||||
functions
|
||||
)
|
||||
except exceptions.VariableNotFound:
|
||||
pass
|
||||
override_variables[key] = parsed_value
|
||||
|
||||
if override_variables:
|
||||
config["variables"] = override_variables
|
||||
|
||||
# parse config name
|
||||
config["name"] = parse_data(
|
||||
config.get("name", ""),
|
||||
override_variables,
|
||||
functions
|
||||
)
|
||||
|
||||
# parse config base_url
|
||||
if "base_url" in config:
|
||||
config["base_url"] = parse_data(
|
||||
config["base_url"],
|
||||
override_variables,
|
||||
functions
|
||||
)
|
||||
|
||||
|
||||
def __parse_teststeps(teststeps, config, project_mapping):
|
||||
""" override teststeps with testcase config variables, base_url and verify.
|
||||
teststep maybe nested testcase.
|
||||
|
||||
variables priority:
|
||||
testsuite config > testsuite teststep > testcase config > testcase teststep > api
|
||||
|
||||
base_url/verify priority:
|
||||
testcase teststep > testcase config > testsuite teststep > testsuite config > api
|
||||
|
||||
Args:
|
||||
teststeps (list):
|
||||
config (dict):
|
||||
|
||||
Returns:
|
||||
list: overrided teststeps
|
||||
|
||||
"""
|
||||
config_variables = config.pop("variables", {})
|
||||
config_base_url = config.pop("base_url", None)
|
||||
config_verify = config.pop("verify", True)
|
||||
functions = project_mapping.get("functions", {})
|
||||
|
||||
for teststep in teststeps:
|
||||
|
||||
# priority teststep > config
|
||||
teststep.setdefault("base_url", config_base_url)
|
||||
teststep.setdefault("verify", config_verify)
|
||||
|
||||
if "testcase_def" in teststep:
|
||||
# teststep is nested testcase
|
||||
|
||||
# 1, testsuite config => testsuite teststeps
|
||||
# override teststep variables
|
||||
teststep["variables"] = utils.extend_variables(
|
||||
teststep.pop("variables", {}),
|
||||
config_variables
|
||||
)
|
||||
|
||||
# parse teststep name
|
||||
try:
|
||||
teststep["name"] = parse_data(
|
||||
teststep.pop("name", ""),
|
||||
teststep["variables"],
|
||||
functions
|
||||
)
|
||||
except exceptions.VariableNotFound:
|
||||
pass
|
||||
|
||||
# 2, testsuite teststep => testcase config
|
||||
testcase_def = teststep.pop("testcase_def")
|
||||
_extend_with_testcase(teststep, testcase_def)
|
||||
|
||||
# 3, testcase config => testcase teststep
|
||||
_parse_testcase(teststep, project_mapping)
|
||||
|
||||
else:
|
||||
# 1, config => teststeps
|
||||
# override teststep variables
|
||||
teststep["variables"] = utils.extend_variables(
|
||||
teststep.pop("variables", {}),
|
||||
config_variables
|
||||
)
|
||||
|
||||
# parse teststep name
|
||||
try:
|
||||
teststep["name"] = parse_data(
|
||||
teststep.pop("name", ""),
|
||||
teststep["variables"],
|
||||
functions
|
||||
)
|
||||
except exceptions.VariableNotFound:
|
||||
pass
|
||||
|
||||
if "api_def" in teststep:
|
||||
# 2, teststep => api
|
||||
api_def_dict = teststep.pop("api_def")
|
||||
_extend_with_api(teststep, api_def_dict)
|
||||
else:
|
||||
# base_url
|
||||
base_url = teststep.pop("base_url", None)
|
||||
if base_url:
|
||||
teststep["request"]["url"] = "{}/{}".format(base_url.rstrip("/"), teststep["request"]["url"].lstrip("/"))
|
||||
|
||||
|
||||
def _parse_testcase(testcase, project_mapping):
|
||||
__parse_config(testcase["config"], project_mapping)
|
||||
__parse_teststeps(testcase["teststeps"], testcase["config"], project_mapping)
|
||||
|
||||
|
||||
def parse_tests(tests_mapping):
|
||||
""" parse testcases configs, including variables/name/request.
|
||||
|
||||
Args:
|
||||
tests_mapping (dict): project info and testcases list.
|
||||
|
||||
{
|
||||
"project_mapping": {
|
||||
"PWD": "XXXXX",
|
||||
"functions": {},
|
||||
"variables": {}, # optional, priority 1
|
||||
"env": {}
|
||||
},
|
||||
testcase_dict_2 # another testcase dict
|
||||
]
|
||||
"testcases": [
|
||||
{ # testcase data structure
|
||||
"config": {
|
||||
"name": "desc1",
|
||||
"path": "testcase1_path",
|
||||
"variables": [], # optional, priority 2
|
||||
},
|
||||
"teststeps": [
|
||||
# teststep data structure
|
||||
{
|
||||
'name': 'test step desc1',
|
||||
'variables': [], # optional, priority 3
|
||||
'extract': [],
|
||||
'validate': [],
|
||||
'api_def': {
|
||||
"variables": {} # optional, priority 4
|
||||
'request': {},
|
||||
}
|
||||
},
|
||||
teststep2 # another teststep dict
|
||||
]
|
||||
},
|
||||
testcase_dict_2 # another testcase dict
|
||||
]
|
||||
}
|
||||
|
||||
variables_mapping (dict): if variables_mapping is specified, it will override variables in config block.
|
||||
|
||||
Returns:
|
||||
list: parsed testcases list, with config variables/parameters/name/request parsed.
|
||||
|
||||
"""
|
||||
variables_mapping = variables_mapping or {}
|
||||
parsed_testcases_list = []
|
||||
project_mapping = tests_mapping.get("project_mapping", {})
|
||||
|
||||
for testcase in testcases:
|
||||
testcase_config = testcase.setdefault("config", {})
|
||||
functions = testcase_config.get("functions", {})
|
||||
env_mapping = project_mapping.get("env", {})
|
||||
# set OS environment variables
|
||||
utils.set_os_environ(env_mapping)
|
||||
|
||||
env_mapping = testcase_config.get("env", {})
|
||||
# set OS environment variables
|
||||
utils.set_os_environ(env_mapping)
|
||||
for testcase in tests_mapping["testcases"]:
|
||||
testcase.setdefault("config", {})
|
||||
_parse_testcase(testcase, project_mapping)
|
||||
|
||||
# parse config parameters
|
||||
config_parameters = testcase_config.pop("parameters", [])
|
||||
cartesian_product_parameters_list = parse_parameters(
|
||||
config_parameters,
|
||||
testcase_config.get("variables", {}),
|
||||
functions
|
||||
) or [{}]
|
||||
|
||||
for parameter_mapping in cartesian_product_parameters_list:
|
||||
testcase_dict = utils.deepcopy_dict(testcase)
|
||||
config = testcase_dict.get("config")
|
||||
# get config variables
|
||||
raw_config_variables = config.get("variables", [])
|
||||
raw_config_variables_mapping = utils.ensure_mapping_format(raw_config_variables)
|
||||
|
||||
# priority: passed in > parameters > variables
|
||||
|
||||
config_variables = utils.deepcopy_dict(parameter_mapping)
|
||||
config_variables.update(variables_mapping)
|
||||
|
||||
for key, value in raw_config_variables_mapping.items():
|
||||
|
||||
if key in config_variables:
|
||||
# passed in & .env & parameters
|
||||
continue
|
||||
else:
|
||||
# config variables
|
||||
parsed_value = parse_data(
|
||||
value,
|
||||
config_variables,
|
||||
functions
|
||||
)
|
||||
config_variables[key] = parsed_value
|
||||
|
||||
testcase_dict["config"]["variables"] = config_variables
|
||||
|
||||
# parse config name
|
||||
testcase_dict["config"]["name"] = parse_data(
|
||||
testcase_dict["config"].get("name", ""),
|
||||
config_variables,
|
||||
functions
|
||||
)
|
||||
|
||||
# parse config request
|
||||
testcase_dict["config"]["request"] = parse_data(
|
||||
testcase_dict["config"].get("request", {}),
|
||||
config_variables,
|
||||
functions
|
||||
)
|
||||
|
||||
# put loaded project functions to config
|
||||
testcase_dict["config"]["functions"] = functions
|
||||
parsed_testcases_list.append(testcase_dict)
|
||||
|
||||
# unset OS environment variables
|
||||
utils.unset_os_environ(env_mapping)
|
||||
|
||||
return parsed_testcases_list
|
||||
# unset OS environment variables
|
||||
utils.unset_os_environ(env_mapping)
|
||||
|
||||
@@ -4,31 +4,62 @@ from unittest.case import SkipTest
|
||||
|
||||
from httprunner import exceptions, logger, response, utils
|
||||
from httprunner.client import HttpSession
|
||||
from httprunner.compat import OrderedDict
|
||||
from httprunner.context import Context
|
||||
from httprunner.context import SessionContext
|
||||
|
||||
|
||||
class Runner(object):
|
||||
""" Running testcases.
|
||||
|
||||
Examples:
|
||||
>>> functions={...}
|
||||
>>> config = {
|
||||
"name": "XXXX",
|
||||
"base_url": "http://127.0.0.1",
|
||||
"verify": False
|
||||
}
|
||||
>>> runner = Runner(config, functions)
|
||||
|
||||
>>> teststep = {
|
||||
"name": "teststep description",
|
||||
"variables": [], # optional
|
||||
"request": {
|
||||
"url": "http://127.0.0.1:5000/api/users/1000",
|
||||
"method": "GET"
|
||||
}
|
||||
}
|
||||
>>> runner.run_test(teststep)
|
||||
|
||||
"""
|
||||
|
||||
def __init__(self, config, functions, http_client_session=None):
|
||||
""" run testcase or testsuite.
|
||||
|
||||
Args:
|
||||
config (dict): testcase/testsuite config dict
|
||||
|
||||
{
|
||||
"name": "ABC",
|
||||
"variables": {},
|
||||
"setup_hooks", [],
|
||||
"teardown_hooks", []
|
||||
}
|
||||
|
||||
http_client_session (instance): requests.Session(), or locust.client.Session() instance.
|
||||
|
||||
def __init__(self, config_dict=None, http_client_session=None):
|
||||
""" Each testcase is corresponding to one Runner() instance
|
||||
and one Context() instance.
|
||||
"""
|
||||
self.http_client_session = http_client_session
|
||||
config_dict = config_dict or {}
|
||||
base_url = config.get("base_url")
|
||||
self.verify = config.get("verify", True)
|
||||
self.output = config.get("output", [])
|
||||
self.functions = functions
|
||||
self.evaluated_validators = []
|
||||
|
||||
# testcase variables
|
||||
config_variables = config_dict.get("variables", {})
|
||||
# testcase functions
|
||||
config_functions = config_dict.get("functions", {})
|
||||
# testcase setup hooks
|
||||
testcase_setup_hooks = config_dict.pop("setup_hooks", [])
|
||||
testcase_setup_hooks = config.get("setup_hooks", [])
|
||||
# testcase teardown hooks
|
||||
self.testcase_teardown_hooks = config_dict.pop("teardown_hooks", [])
|
||||
self.testcase_teardown_hooks = config.get("teardown_hooks", [])
|
||||
|
||||
self.context = Context(config_variables, config_functions)
|
||||
self.init_test(config_dict, "testcase")
|
||||
self.http_client_session = http_client_session or HttpSession(base_url)
|
||||
self.session_context = SessionContext(self.functions)
|
||||
|
||||
if testcase_setup_hooks:
|
||||
self.do_hook_actions(testcase_setup_hooks)
|
||||
@@ -37,57 +68,6 @@ class Runner(object):
|
||||
if self.testcase_teardown_hooks:
|
||||
self.do_hook_actions(self.testcase_teardown_hooks)
|
||||
|
||||
def init_test(self, test_dict, level):
|
||||
""" create/update context variables binds
|
||||
|
||||
Args:
|
||||
test_dict (dict):
|
||||
level (enum): "testcase" or "teststep"
|
||||
testcase:
|
||||
{
|
||||
"name": "testcase description",
|
||||
"variables": [], # optional
|
||||
"request": {
|
||||
"base_url": "http://127.0.0.1:5000",
|
||||
"headers": {
|
||||
"User-Agent": "iOS/2.8.3"
|
||||
}
|
||||
}
|
||||
}
|
||||
teststep:
|
||||
{
|
||||
"name": "teststep description",
|
||||
"variables": [], # optional
|
||||
"request": {
|
||||
"url": "/api/get-token",
|
||||
"method": "POST",
|
||||
"headers": {
|
||||
"Content-Type": "application/json"
|
||||
}
|
||||
},
|
||||
"json": {
|
||||
"sign": "f1219719911caae89ccc301679857ebfda115ca2"
|
||||
}
|
||||
}
|
||||
|
||||
Returns:
|
||||
dict: parsed request dict
|
||||
|
||||
"""
|
||||
test_dict = utils.lower_test_dict_keys(test_dict)
|
||||
|
||||
self.context.init_context_variables(level)
|
||||
variables = test_dict.get('variables', OrderedDict())
|
||||
self.context.update_context_variables(variables, level)
|
||||
|
||||
request_config = test_dict.get('request', {})
|
||||
parsed_request = self.context.get_parsed_request(request_config, level)
|
||||
|
||||
base_url = parsed_request.pop("base_url", None)
|
||||
self.http_client_session = self.http_client_session or HttpSession(base_url)
|
||||
|
||||
return parsed_request
|
||||
|
||||
def _handle_skip_feature(self, teststep_dict):
|
||||
""" handle skip feature for teststep
|
||||
- skip: skip current test unconditionally
|
||||
@@ -109,12 +89,12 @@ class Runner(object):
|
||||
|
||||
elif "skipIf" in teststep_dict:
|
||||
skip_if_condition = teststep_dict["skipIf"]
|
||||
if self.context.eval_content(skip_if_condition):
|
||||
if self.session_context.eval_content(skip_if_condition):
|
||||
skip_reason = "{} evaluate to True".format(skip_if_condition)
|
||||
|
||||
elif "skipUnless" in teststep_dict:
|
||||
skip_unless_condition = teststep_dict["skipUnless"]
|
||||
if not self.context.eval_content(skip_unless_condition):
|
||||
if not self.session_context.eval_content(skip_unless_condition):
|
||||
skip_reason = "{} evaluate to False".format(skip_unless_condition)
|
||||
|
||||
if skip_reason:
|
||||
@@ -124,9 +104,9 @@ class Runner(object):
|
||||
for action in actions:
|
||||
logger.log_debug("call hook: {}".format(action))
|
||||
# TODO: check hook function if valid
|
||||
self.context.eval_content(action)
|
||||
self.session_context.eval_content(action)
|
||||
|
||||
def run_test(self, teststep_dict):
|
||||
def _run_teststep(self, teststep_dict):
|
||||
""" run single teststep.
|
||||
|
||||
Args:
|
||||
@@ -135,7 +115,7 @@ class Runner(object):
|
||||
"name": "teststep description",
|
||||
"skip": "skip this test unconditionally",
|
||||
"times": 3,
|
||||
"variables": [], # optional, override
|
||||
"variables": [], # optional, override
|
||||
"request": {
|
||||
"url": "http://127.0.0.1:5000/api/users/1000",
|
||||
"method": "POST",
|
||||
@@ -162,10 +142,14 @@ class Runner(object):
|
||||
self._handle_skip_feature(teststep_dict)
|
||||
|
||||
# prepare
|
||||
extractors = teststep_dict.get("extract", [])
|
||||
validators = teststep_dict.get("validate", [])
|
||||
parsed_request = self.init_test(teststep_dict, level="teststep")
|
||||
self.context.update_teststep_variables_mapping("request", parsed_request)
|
||||
teststep_dict = utils.lower_test_dict_keys(teststep_dict)
|
||||
teststep_variables = teststep_dict.get("variables", {})
|
||||
self.session_context.init_teststep_variables(teststep_variables)
|
||||
|
||||
# parse teststep request
|
||||
raw_request = teststep_dict.get('request', {})
|
||||
parsed_teststep_request = self.session_context.eval_content(raw_request)
|
||||
self.session_context.update_teststep_variables("request", parsed_teststep_request)
|
||||
|
||||
# setup hooks
|
||||
setup_hooks = teststep_dict.get("setup_hooks", [])
|
||||
@@ -173,9 +157,10 @@ class Runner(object):
|
||||
self.do_hook_actions(setup_hooks)
|
||||
|
||||
try:
|
||||
url = parsed_request.pop('url')
|
||||
method = parsed_request.pop('method')
|
||||
group_name = parsed_request.pop("group", None)
|
||||
url = parsed_teststep_request.pop('url')
|
||||
method = parsed_teststep_request.pop('method')
|
||||
parsed_teststep_request.setdefault("verify", self.verify)
|
||||
group_name = parsed_teststep_request.pop("group", None)
|
||||
except KeyError:
|
||||
raise exceptions.ParamsError("URL or METHOD missed!")
|
||||
|
||||
@@ -188,14 +173,14 @@ class Runner(object):
|
||||
raise exceptions.ParamsError(err_msg)
|
||||
|
||||
logger.log_info("{method} {url}".format(method=method, url=url))
|
||||
logger.log_debug("request kwargs(raw): {kwargs}".format(kwargs=parsed_request))
|
||||
logger.log_debug("request kwargs(raw): {kwargs}".format(kwargs=parsed_teststep_request))
|
||||
|
||||
# request
|
||||
resp = self.http_client_session.request(
|
||||
method,
|
||||
url,
|
||||
name=group_name,
|
||||
**parsed_request
|
||||
**parsed_teststep_request
|
||||
)
|
||||
resp_obj = response.ResponseObject(resp)
|
||||
|
||||
@@ -203,21 +188,23 @@ class Runner(object):
|
||||
teardown_hooks = teststep_dict.get("teardown_hooks", [])
|
||||
if teardown_hooks:
|
||||
logger.log_info("start to run teardown hooks")
|
||||
self.context.update_teststep_variables_mapping("response", resp_obj)
|
||||
self.session_context.update_teststep_variables("response", resp_obj)
|
||||
self.do_hook_actions(teardown_hooks)
|
||||
|
||||
# extract
|
||||
extractors = teststep_dict.get("extract", [])
|
||||
extracted_variables_mapping = resp_obj.extract_response(extractors)
|
||||
self.context.update_testcase_runtime_variables_mapping(extracted_variables_mapping)
|
||||
self.session_context.update_seesion_variables(extracted_variables_mapping)
|
||||
|
||||
# validate
|
||||
validators = teststep_dict.get("validate", [])
|
||||
try:
|
||||
self.evaluated_validators = self.context.validate(validators, resp_obj)
|
||||
self.evaluated_validators = self.session_context.validate(validators, resp_obj)
|
||||
except (exceptions.ParamsError, exceptions.ValidationFailure, exceptions.ExtractFailure):
|
||||
# log request
|
||||
err_req_msg = "request: \n"
|
||||
err_req_msg += "headers: {}\n".format(parsed_request.pop("headers", {}))
|
||||
for k, v in parsed_request.items():
|
||||
err_req_msg += "headers: {}\n".format(parsed_teststep_request.pop("headers", {}))
|
||||
for k, v in parsed_teststep_request.items():
|
||||
err_req_msg += "{}: {}\n".format(k, repr(v))
|
||||
logger.log_error(err_req_msg)
|
||||
|
||||
@@ -230,10 +217,66 @@ class Runner(object):
|
||||
|
||||
raise
|
||||
|
||||
def _run_testcase(self, testcase_dict):
|
||||
""" run single testcase.
|
||||
"""
|
||||
config = testcase_dict.get("config", {})
|
||||
test_runner = Runner(config, self.functions, self.http_client_session)
|
||||
|
||||
teststeps = testcase_dict.get("teststeps", [])
|
||||
for index, teststep_dict in enumerate(teststeps):
|
||||
test_runner.run_test(teststep_dict)
|
||||
|
||||
self.session_context.update_seesion_variables(test_runner.extract_sessions())
|
||||
|
||||
def run_test(self, teststep_dict):
|
||||
""" run single teststep of testcase.
|
||||
teststep_dict may be in 3 types.
|
||||
|
||||
Args:
|
||||
teststep_dict (dict):
|
||||
|
||||
# teststep
|
||||
{
|
||||
"name": "teststep description",
|
||||
"variables": [], # optional
|
||||
"request": {
|
||||
"url": "http://127.0.0.1:5000/api/users/1000",
|
||||
"method": "GET"
|
||||
}
|
||||
}
|
||||
|
||||
# embeded testcase
|
||||
{
|
||||
"config": {...},
|
||||
"teststeps": [
|
||||
{...},
|
||||
{...}
|
||||
]
|
||||
}
|
||||
|
||||
# TODO: function
|
||||
{
|
||||
"name": "exec function",
|
||||
"function": "${func()}"
|
||||
}
|
||||
|
||||
"""
|
||||
if "config" in teststep_dict:
|
||||
self._run_testcase(teststep_dict)
|
||||
else:
|
||||
# api
|
||||
self._run_teststep(teststep_dict)
|
||||
|
||||
def extract_sessions(self):
|
||||
"""
|
||||
"""
|
||||
return self.extract_output(self.output)
|
||||
|
||||
def extract_output(self, output_variables_list):
|
||||
""" extract output variables
|
||||
"""
|
||||
variables_mapping = self.context.teststep_variables_mapping
|
||||
variables_mapping = self.session_context.session_variables_mapping
|
||||
|
||||
output = {}
|
||||
for variable in output_variables_list:
|
||||
|
||||
@@ -268,6 +268,113 @@ def ensure_mapping_format(variables):
|
||||
return variables_ordered_dict
|
||||
|
||||
|
||||
def _convert_validators_to_mapping(validators):
|
||||
""" convert validators list to mapping.
|
||||
|
||||
Args:
|
||||
validators (list): validators in list
|
||||
|
||||
Returns:
|
||||
dict: validators mapping, use (check, comparator) as key.
|
||||
|
||||
Examples:
|
||||
>>> validators = [
|
||||
{"check": "v1", "expect": 201, "comparator": "eq"},
|
||||
{"check": {"b": 1}, "expect": 200, "comparator": "eq"}
|
||||
]
|
||||
>>> _convert_validators_to_mapping(validators)
|
||||
{
|
||||
("v1", "eq"): {"check": "v1", "expect": 201, "comparator": "eq"},
|
||||
('{"b": 1}', "eq"): {"check": {"b": 1}, "expect": 200, "comparator": "eq"}
|
||||
}
|
||||
|
||||
"""
|
||||
validators_mapping = {}
|
||||
|
||||
for validator in validators:
|
||||
if not isinstance(validator["check"], collections.Hashable):
|
||||
check = json.dumps(validator["check"])
|
||||
else:
|
||||
check = validator["check"]
|
||||
|
||||
key = (check, validator["comparator"])
|
||||
validators_mapping[key] = validator
|
||||
|
||||
return validators_mapping
|
||||
|
||||
|
||||
def extend_validators(raw_validators, override_validators):
|
||||
""" extend raw_validators with override_validators.
|
||||
override_validators will merge and override raw_validators.
|
||||
|
||||
Args:
|
||||
raw_validators (dict):
|
||||
override_validators (dict):
|
||||
|
||||
Returns:
|
||||
list: extended validators
|
||||
|
||||
Examples:
|
||||
>>> raw_validators = [{'eq': ['v1', 200]}, {"check": "s2", "expect": 16, "comparator": "len_eq"}]
|
||||
>>> override_validators = [{"check": "v1", "expect": 201}, {'len_eq': ['s3', 12]}]
|
||||
>>> extend_validators(raw_validators, override_validators)
|
||||
[
|
||||
{"check": "v1", "expect": 201, "comparator": "eq"},
|
||||
{"check": "s2", "expect": 16, "comparator": "len_eq"},
|
||||
{"check": "s3", "expect": 12, "comparator": "len_eq"}
|
||||
]
|
||||
|
||||
"""
|
||||
|
||||
if not raw_validators:
|
||||
return override_validators
|
||||
|
||||
elif not override_validators:
|
||||
return raw_validators
|
||||
|
||||
else:
|
||||
def_validators_mapping = _convert_validators_to_mapping(raw_validators)
|
||||
ref_validators_mapping = _convert_validators_to_mapping(override_validators)
|
||||
|
||||
def_validators_mapping.update(ref_validators_mapping)
|
||||
return list(def_validators_mapping.values())
|
||||
|
||||
|
||||
def extend_variables(raw_variables, override_variables):
|
||||
""" extend raw_variables with override_variables.
|
||||
override_variables will merge and override raw_variables.
|
||||
|
||||
Args:
|
||||
raw_variables (list):
|
||||
override_variables (list):
|
||||
|
||||
Returns:
|
||||
OrderedDict: extended variables mapping
|
||||
|
||||
Examples:
|
||||
>>> raw_variables = [{"var1": "val1"}, {"var2": "val2"}]
|
||||
>>> override_variables = [{"var1": "val111"}, {"var3": "val3"}]
|
||||
>>> extend_variables(raw_variables, override_variables)
|
||||
OrderedDict([
|
||||
('var1', 'val111'),
|
||||
('var2', 'val2'),
|
||||
('var3', 'val3')
|
||||
])
|
||||
|
||||
"""
|
||||
if not raw_variables:
|
||||
return override_variables
|
||||
|
||||
elif not override_variables:
|
||||
return raw_variables
|
||||
|
||||
else:
|
||||
raw_variables_mapping = ensure_mapping_format(raw_variables)
|
||||
override_variables_mapping = ensure_mapping_format(override_variables)
|
||||
raw_variables_mapping.update(override_variables_mapping)
|
||||
return raw_variables_mapping
|
||||
|
||||
|
||||
def get_testcase_io(testcase):
|
||||
""" get testcase input(variables) and output.
|
||||
|
||||
|
||||
@@ -54,21 +54,50 @@ def is_testcases(data_structure):
|
||||
|
||||
Args:
|
||||
data_structure (dict): testcase(s) should always be in the following data structure:
|
||||
{
|
||||
"project_mapping": {
|
||||
"PWD": "XXXXX",
|
||||
"functions": {},
|
||||
"env": {}
|
||||
},
|
||||
"testcases": [
|
||||
{ # testcase data structure
|
||||
"config": {
|
||||
"name": "desc1",
|
||||
"path": "testcase1_path",
|
||||
"variables": [], # optional
|
||||
},
|
||||
"teststeps": [
|
||||
# teststep data structure
|
||||
{
|
||||
'name': 'test step desc1',
|
||||
'variables': [], # optional
|
||||
'extract': [], # optional
|
||||
'validate': [],
|
||||
'request': {}
|
||||
},
|
||||
teststep2 # another teststep dict
|
||||
]
|
||||
},
|
||||
testcase_dict_2 # another testcase dict
|
||||
]
|
||||
}
|
||||
|
||||
testcase_dict
|
||||
or
|
||||
[
|
||||
testcase_dict_1,
|
||||
testcase_dict_2
|
||||
]
|
||||
Returns:
|
||||
bool: True if data_structure is valid testcase(s), otherwise False.
|
||||
|
||||
"""
|
||||
if not isinstance(data_structure, list):
|
||||
return is_testcase(data_structure)
|
||||
if not isinstance(data_structure, dict):
|
||||
return False
|
||||
|
||||
for item in data_structure:
|
||||
if "testcases" not in data_structure:
|
||||
return False
|
||||
|
||||
testcases = data_structure["testcases"]
|
||||
if not isinstance(testcases, list):
|
||||
return False
|
||||
|
||||
for item in testcases:
|
||||
if not is_testcase(item):
|
||||
return False
|
||||
|
||||
|
||||
@@ -1,5 +1,11 @@
|
||||
- api:
|
||||
def: get_token($user_agent, $device_sn, $os_platform, $app_version)
|
||||
id: get_token
|
||||
name: get token
|
||||
variables:
|
||||
- user_agent: XXX
|
||||
- device_sn: API_XXX
|
||||
- os_platform: XXX
|
||||
- app_version: XXX
|
||||
request:
|
||||
url: /api/get-token
|
||||
method: POST
|
||||
@@ -8,6 +14,8 @@
|
||||
device_sn: $device_sn
|
||||
os_platform: $os_platform
|
||||
app_version: $app_version
|
||||
Content-Type: "application/json"
|
||||
device_sn: $device_sn
|
||||
json:
|
||||
sign: ${get_sign($user_agent, $device_sn, $os_platform, $app_version)}
|
||||
validate:
|
||||
@@ -16,15 +24,18 @@
|
||||
- contains: [{"a": 1, "b": 2}, "a"]
|
||||
|
||||
- api:
|
||||
def: create_user($uid, $user_name, $user_password, $token)
|
||||
id: create_user
|
||||
variables:
|
||||
- user_name: user0
|
||||
- user_password: "000000"
|
||||
- uid: 2000
|
||||
- uid: 9000
|
||||
- token: XXX
|
||||
request:
|
||||
url: /api/users/$uid
|
||||
method: POST
|
||||
headers:
|
||||
Content-Type: "application/json"
|
||||
device_sn: $device_sn
|
||||
token: $token
|
||||
json:
|
||||
name: $user_name
|
||||
@@ -33,21 +44,33 @@
|
||||
- eq: ["status_code", 201]
|
||||
|
||||
- api:
|
||||
def: get_user($uid, $token)
|
||||
id: get_user
|
||||
variables:
|
||||
- uid: 9000
|
||||
- token: XXX
|
||||
request:
|
||||
url: /api/users/$uid
|
||||
method: GET
|
||||
headers:
|
||||
Content-Type: "application/json"
|
||||
device_sn: $device_sn
|
||||
token: $token
|
||||
validate:
|
||||
- eq: ["status_code", 200]
|
||||
|
||||
- api:
|
||||
def: update_user($uid, $user_name, $user_password, $token)
|
||||
id: update_user
|
||||
variables:
|
||||
- user_name: user0
|
||||
- user_password: "000000"
|
||||
- uid: 9000
|
||||
- token: XXX
|
||||
request:
|
||||
url: /api/users/$uid
|
||||
method: PUT
|
||||
headers:
|
||||
Content-Type: "application/json"
|
||||
device_sn: $device_sn
|
||||
token: $token
|
||||
json:
|
||||
name: $user_name
|
||||
@@ -56,40 +79,58 @@
|
||||
- eq: ["status_code", 200]
|
||||
|
||||
- api:
|
||||
def: delete_user($uid, $token)
|
||||
id: delete_user
|
||||
variables:
|
||||
- uid: 9000
|
||||
- token: XXX
|
||||
request:
|
||||
url: /api/users/$uid
|
||||
method: DELETE
|
||||
headers:
|
||||
Content-Type: "application/json"
|
||||
device_sn: $device_sn
|
||||
token: $token
|
||||
validate:
|
||||
- eq: ["status_code", 200]
|
||||
|
||||
- api:
|
||||
def: get_users($token)
|
||||
id: get_users
|
||||
variables:
|
||||
- token: XXX
|
||||
request:
|
||||
url: /api/users
|
||||
method: GET
|
||||
headers:
|
||||
Content-Type: "application/json"
|
||||
device_sn: $device_sn
|
||||
token: $token
|
||||
validate:
|
||||
- eq: ["status_code", 200]
|
||||
|
||||
- api:
|
||||
def: reset_all($token)
|
||||
id: reset_all
|
||||
variables:
|
||||
- token: XXX
|
||||
request:
|
||||
url: /api/reset-all
|
||||
method: GET
|
||||
headers:
|
||||
Content-Type: "application/json"
|
||||
device_sn: $device_sn
|
||||
token: $token
|
||||
validate:
|
||||
- eq: ["status_code", 200]
|
||||
- eq: ["content.success", true]
|
||||
|
||||
- api:
|
||||
def: get_headers($n_secs)
|
||||
id: get_headers
|
||||
variables:
|
||||
- n_secs: 1
|
||||
request:
|
||||
url: /headers
|
||||
headers:
|
||||
Content-Type: "application/json"
|
||||
device_sn: $device_sn
|
||||
method: GET
|
||||
setup_hooks:
|
||||
- ${setup_hook_add_kwargs($request)}
|
||||
|
||||
@@ -18,7 +18,7 @@
|
||||
|
||||
- test:
|
||||
name: get token with $user_agent, $app_version
|
||||
api: get_token($user_agent, $device_sn, $os_platform, $app_version)
|
||||
api: get_token
|
||||
extract:
|
||||
- token: content.token
|
||||
validate:
|
||||
|
||||
@@ -6,20 +6,18 @@
|
||||
- ["test1","111111"]
|
||||
- ["test2","222222"]
|
||||
variables:
|
||||
- username: test1
|
||||
- user_agent: "iOS/10.1"
|
||||
- device_sn: ${gen_random_string(15)}
|
||||
- os_platform: 'ios'
|
||||
- app_version: 2.8.5
|
||||
request:
|
||||
base_url: ${get_base_url()}
|
||||
headers:
|
||||
Content-Type: application/json
|
||||
device_sn: $device_sn
|
||||
base_url: ${get_base_url()}
|
||||
output:
|
||||
- token
|
||||
|
||||
- test:
|
||||
name: get token with $user_agent and $username
|
||||
api: get_token($user_agent, $device_sn, $os_platform, $app_version)
|
||||
api: get_token
|
||||
extract:
|
||||
- token: content.token
|
||||
validate:
|
||||
|
||||
@@ -4,12 +4,11 @@
|
||||
- var_a: 0
|
||||
- var_c: "${sum_two(1, 2)}"
|
||||
- PROJECT_KEY: ${ENV(PROJECT_KEY)}
|
||||
parameters:
|
||||
- "var_a-var_b":
|
||||
- [11, 21]
|
||||
- [12, 22]
|
||||
- "app_version": "${gen_app_version()}"
|
||||
request: ${get_default_request()}
|
||||
# parameters:
|
||||
# - "var_a-var_b":
|
||||
# - [11, 21]
|
||||
# - [12, 22]
|
||||
# - "app_version": "${gen_app_version()}"
|
||||
|
||||
- test:
|
||||
name: testcase1-$var_a
|
||||
|
||||
@@ -5,11 +5,7 @@
|
||||
- device_sn: ${gen_random_string(15)}
|
||||
- os_platform: 'ios'
|
||||
- app_version: '2.8.6'
|
||||
request:
|
||||
base_url: ${get_base_url()}
|
||||
headers:
|
||||
Content-Type: application/json
|
||||
device_sn: $device_sn
|
||||
base_url: ${get_base_url()}
|
||||
|
||||
- test:
|
||||
name: get token
|
||||
@@ -38,6 +34,8 @@
|
||||
url: /api/users/1000
|
||||
method: POST
|
||||
headers:
|
||||
Content-Type: application/json
|
||||
device_sn: $device_sn
|
||||
token: $token
|
||||
json:
|
||||
name: $user_name
|
||||
@@ -52,6 +50,8 @@
|
||||
url: /api/users/1000
|
||||
method: POST
|
||||
headers:
|
||||
Content-Type: application/json
|
||||
device_sn: $device_sn
|
||||
token: $token
|
||||
json:
|
||||
name: "user1"
|
||||
|
||||
@@ -5,17 +5,13 @@
|
||||
- device_sn: ${gen_random_string(15)}
|
||||
- os_platform: 'ios'
|
||||
- app_version: '2.8.6'
|
||||
request:
|
||||
base_url: ${get_base_url()}
|
||||
headers:
|
||||
Content-Type: application/json
|
||||
device_sn: $device_sn
|
||||
base_url: ${get_base_url()}
|
||||
output:
|
||||
- token
|
||||
|
||||
- test:
|
||||
name: get token with $user_agent, $app_version
|
||||
api: get_token($user_agent, $device_sn, $os_platform, $app_version)
|
||||
api: get_token
|
||||
extract:
|
||||
- token: content.token
|
||||
validate:
|
||||
@@ -25,14 +21,19 @@
|
||||
|
||||
- test:
|
||||
name: reset all users
|
||||
api: reset_all($token)
|
||||
api: reset_all
|
||||
variables:
|
||||
- token: $token
|
||||
validate:
|
||||
- {"check": "status_code", "expect": 200}
|
||||
- {"check": "content.success", "expect": true}
|
||||
|
||||
- test:
|
||||
name: get user that does not exist
|
||||
api: get_user(1000, $token)
|
||||
api: get_user
|
||||
variables:
|
||||
- uid: 1000
|
||||
- token: $token
|
||||
validate:
|
||||
- {"check": "status_code", "expect": 404}
|
||||
- {"check": "content.success", "expect": false}
|
||||
@@ -40,16 +41,21 @@
|
||||
- test:
|
||||
name: create user which does not exist
|
||||
variables:
|
||||
- uid: 1000
|
||||
- user_name: "user1"
|
||||
- user_password: "123456"
|
||||
api: create_user(1000, $user_name, $user_password, $token)
|
||||
- token: $token
|
||||
api: create_user
|
||||
validate:
|
||||
- {"check": "status_code", "expect": 201}
|
||||
- {"check": "content.success", "expect": true}
|
||||
|
||||
- test:
|
||||
name: get user that has been created
|
||||
api: get_user(1000, $token)
|
||||
api: get_user
|
||||
variables:
|
||||
- uid: 1000
|
||||
- token: $token
|
||||
validate:
|
||||
- {"check": "status_code", "expect": 200}
|
||||
- {"check": "content.success", "expect": true}
|
||||
@@ -58,9 +64,11 @@
|
||||
- test:
|
||||
name: create user which exists
|
||||
variables:
|
||||
- uid: 1000
|
||||
- user_name: "user1"
|
||||
- user_password: "123456"
|
||||
api: create_user(1000, $user_name, $user_password, $token)
|
||||
- token: $token
|
||||
api: create_user
|
||||
validate:
|
||||
- {"check": "status_code", "expect": 500}
|
||||
- {"check": "content.success", "expect": false}
|
||||
@@ -68,16 +76,21 @@
|
||||
- test:
|
||||
name: update user which exists
|
||||
variables:
|
||||
- uid: 1000
|
||||
- user_name: "user1"
|
||||
- user_password: "654321"
|
||||
api: update_user(1000, $user_name, $user_password, $token)
|
||||
- token: $token
|
||||
api: update_user
|
||||
validate:
|
||||
- {"check": "status_code", "expect": 200}
|
||||
- {"check": "content.success", "expect": true}
|
||||
|
||||
- test:
|
||||
name: get user that has been updated
|
||||
api: get_user(1000, $token)
|
||||
api: get_user
|
||||
variables:
|
||||
- uid: 1000
|
||||
- token: $token
|
||||
validate:
|
||||
- {"check": "status_code", "expect": 200}
|
||||
- {"check": "content.success", "expect": true}
|
||||
@@ -85,21 +98,28 @@
|
||||
|
||||
- test:
|
||||
name: get users
|
||||
api: get_users($token)
|
||||
api: get_users
|
||||
variables:
|
||||
- token: $token
|
||||
validate:
|
||||
- {"check": "status_code", "expect": 200}
|
||||
- {"check": "content.count", "expect": 1}
|
||||
|
||||
- test:
|
||||
name: delete user that exists
|
||||
api: delete_user(1000, $token)
|
||||
api: delete_user
|
||||
variables:
|
||||
- uid: 1000
|
||||
- token: $token
|
||||
validate:
|
||||
- {"check": "status_code", "expect": 200}
|
||||
- {"check": "content.success", "expect": true}
|
||||
|
||||
- test:
|
||||
name: get users
|
||||
api: get_users($token)
|
||||
api: get_users
|
||||
variables:
|
||||
- token: $token
|
||||
validate:
|
||||
- {"check": "status_code", "expect": 200}
|
||||
- {"check": "content.count", "expect": 0}
|
||||
@@ -107,16 +127,20 @@
|
||||
- test:
|
||||
name: create user which has been deleted
|
||||
variables:
|
||||
- uid: 1000
|
||||
- user_name: "user1"
|
||||
- user_password: "123456"
|
||||
api: create_user(1000, $user_name, $user_password, $token)
|
||||
- token: $token
|
||||
api: create_user
|
||||
validate:
|
||||
- {"check": "status_code", "expect": 201}
|
||||
- {"check": "content.success", "expect": true}
|
||||
|
||||
- test:
|
||||
name: get users
|
||||
api: get_users($token)
|
||||
api: get_users
|
||||
variables:
|
||||
- token: $token
|
||||
validate:
|
||||
- {"check": "status_code", "expect": 200}
|
||||
- {"check": "content.count", "expect": 1}
|
||||
|
||||
@@ -2,11 +2,7 @@
|
||||
name: "create user testcases."
|
||||
variables:
|
||||
- device_sn: 'HZfFBh6tU59EdXJ'
|
||||
request:
|
||||
base_url: ${get_base_url()}
|
||||
headers:
|
||||
Content-Type: application/json
|
||||
device_sn: $device_sn
|
||||
base_url: ${get_base_url()}
|
||||
|
||||
- test:
|
||||
name: get token
|
||||
@@ -41,6 +37,8 @@
|
||||
url: /api/users/1000
|
||||
method: POST
|
||||
headers:
|
||||
Content-Type: application/json
|
||||
device_sn: $device_sn
|
||||
token: $token
|
||||
json:
|
||||
name: $user_name
|
||||
@@ -55,6 +53,8 @@
|
||||
url: /api/users/1000
|
||||
method: POST
|
||||
headers:
|
||||
Content-Type: application/json
|
||||
device_sn: $device_sn
|
||||
token: $token
|
||||
json:
|
||||
name: "user1"
|
||||
|
||||
@@ -1,7 +1,6 @@
|
||||
- config:
|
||||
name: basic test with httpbin
|
||||
request:
|
||||
base_url: https://httpbin.org/
|
||||
base_url: https://httpbin.org/
|
||||
|
||||
- test:
|
||||
name: index
|
||||
|
||||
@@ -1,7 +1,6 @@
|
||||
- config:
|
||||
name: basic test with httpbin
|
||||
request:
|
||||
base_url: ${get_httpbin_server()}
|
||||
base_url: ${get_httpbin_server()}
|
||||
setup_hooks:
|
||||
- ${hook_print(setup)}
|
||||
teardown_hooks:
|
||||
|
||||
@@ -1,7 +1,6 @@
|
||||
- config:
|
||||
name: load images
|
||||
request:
|
||||
base_url: ${get_httpbin_server()}
|
||||
base_url: ${get_httpbin_server()}
|
||||
|
||||
- test:
|
||||
name: get png image
|
||||
|
||||
@@ -1,7 +1,6 @@
|
||||
- config:
|
||||
name: test upload file with httpbin
|
||||
request:
|
||||
base_url: ${get_httpbin_server()}
|
||||
base_url: ${get_httpbin_server()}
|
||||
|
||||
- test:
|
||||
name: upload file
|
||||
|
||||
@@ -18,7 +18,7 @@ class TestHttpRunner(ApiServerUnittest):
|
||||
os.path.join(
|
||||
os.getcwd(), 'tests/data/demo_testcase_hardcode.json')
|
||||
]
|
||||
self.testcases = [{
|
||||
testcases = [{
|
||||
'config': {
|
||||
'name': 'testcase description',
|
||||
'request': {
|
||||
@@ -61,6 +61,9 @@ class TestHttpRunner(ApiServerUnittest):
|
||||
}
|
||||
]
|
||||
}]
|
||||
self.tests_mapping = {
|
||||
"testcases": testcases
|
||||
}
|
||||
self.reset_all()
|
||||
|
||||
def reset_all(self):
|
||||
@@ -89,7 +92,7 @@ class TestHttpRunner(ApiServerUnittest):
|
||||
shutil.rmtree(report_save_dir)
|
||||
|
||||
def test_run_testcases(self):
|
||||
runner = HttpRunner().run(self.testcases)
|
||||
runner = HttpRunner().run(self.tests_mapping)
|
||||
summary = runner.summary
|
||||
self.assertTrue(summary["success"])
|
||||
self.assertEqual(summary["stat"]["testsRun"], 2)
|
||||
@@ -133,7 +136,10 @@ class TestHttpRunner(ApiServerUnittest):
|
||||
]
|
||||
}
|
||||
]
|
||||
runner = HttpRunner().run(testcases)
|
||||
tests_mapping = {
|
||||
"testcases": testcases
|
||||
}
|
||||
runner = HttpRunner().run(tests_mapping)
|
||||
summary = runner.summary
|
||||
self.assertTrue(summary["success"])
|
||||
self.assertEqual(summary["stat"]["testsRun"], 1)
|
||||
@@ -148,11 +154,18 @@ class TestHttpRunner(ApiServerUnittest):
|
||||
report_save_dir = os.path.join(os.getcwd(), 'reports', output_folder_name)
|
||||
shutil.rmtree(report_save_dir)
|
||||
|
||||
def test_testcase_layer(self):
|
||||
runner = HttpRunner(failfast=True).run("tests/testcases/smoketest.yml")
|
||||
def test_testcase_layer_with_api(self):
|
||||
runner = HttpRunner(failfast=True).run("tests/testcases/setup.yml")
|
||||
summary = runner.summary
|
||||
self.assertTrue(summary["success"])
|
||||
self.assertEqual(summary["stat"]["testsRun"], 8)
|
||||
self.assertEqual(summary["details"][0]["records"][0]["name"], "get token (setup)")
|
||||
self.assertEqual(summary["stat"]["testsRun"], 2)
|
||||
|
||||
def test_testcase_layer_with_testcase(self):
|
||||
runner = HttpRunner(failfast=True).run("tests/testsuites/create_users.yml")
|
||||
summary = runner.summary
|
||||
self.assertTrue(summary["success"])
|
||||
self.assertEqual(summary["stat"]["testsRun"], 2)
|
||||
|
||||
def test_run_httprunner_with_hooks(self):
|
||||
testcase_file_path = os.path.join(
|
||||
@@ -165,11 +178,9 @@ class TestHttpRunner(ApiServerUnittest):
|
||||
self.assertLess(end_time - start_time, 60)
|
||||
|
||||
def test_run_httprunner_with_teardown_hooks_alter_response(self):
|
||||
config = {"name": "test teardown hooks"}
|
||||
config.update(loader.load_project_tests("tests"))
|
||||
testcases = [
|
||||
{
|
||||
"config": config,
|
||||
"config": {"name": "test teardown hooks"},
|
||||
"teststeps": [
|
||||
{
|
||||
"name": "test teardown hooks",
|
||||
@@ -195,7 +206,12 @@ class TestHttpRunner(ApiServerUnittest):
|
||||
]
|
||||
}
|
||||
]
|
||||
runner = HttpRunner().run(testcases)
|
||||
loader.load_project_tests("tests")
|
||||
tests_mapping = {
|
||||
"project_mapping": loader.project_mapping,
|
||||
"testcases": testcases
|
||||
}
|
||||
runner = HttpRunner().run(tests_mapping)
|
||||
summary = runner.summary
|
||||
self.assertTrue(summary["success"])
|
||||
|
||||
@@ -223,7 +239,12 @@ class TestHttpRunner(ApiServerUnittest):
|
||||
]
|
||||
}
|
||||
]
|
||||
runner = HttpRunner().run(testcases)
|
||||
loader.load_project_tests("tests")
|
||||
tests_mapping = {
|
||||
"project_mapping": loader.project_mapping,
|
||||
"testcases": testcases
|
||||
}
|
||||
runner = HttpRunner().run(tests_mapping)
|
||||
summary = runner.summary
|
||||
self.assertFalse(summary["success"])
|
||||
self.assertEqual(summary["stat"]["errors"], 1)
|
||||
@@ -249,7 +270,12 @@ class TestHttpRunner(ApiServerUnittest):
|
||||
]
|
||||
}
|
||||
]
|
||||
runner = HttpRunner().run(testcases)
|
||||
loader.load_project_tests("tests")
|
||||
tests_mapping = {
|
||||
"project_mapping": loader.project_mapping,
|
||||
"testcases": testcases
|
||||
}
|
||||
runner = HttpRunner().run(tests_mapping)
|
||||
summary = runner.summary
|
||||
self.assertFalse(summary["success"])
|
||||
self.assertEqual(summary["stat"]["errors"], 1)
|
||||
@@ -262,14 +288,6 @@ class TestHttpRunner(ApiServerUnittest):
|
||||
self.assertEqual(summary["stat"]["testsRun"], 3)
|
||||
self.assertEqual(summary["stat"]["successes"], 3)
|
||||
|
||||
def test_run_testcases_hardcode(self):
|
||||
runner = HttpRunner().run(self.testcase_file_path_list)
|
||||
summary = runner.summary
|
||||
self.assertTrue(summary["success"])
|
||||
self.assertTrue(summary["success"])
|
||||
self.assertEqual(summary["stat"]["testsRun"], 6)
|
||||
self.assertEqual(summary["stat"]["successes"], 6)
|
||||
|
||||
def test_run_testcase_template_variables(self):
|
||||
testcase_file_path = os.path.join(
|
||||
os.getcwd(), 'tests/data/demo_testcase_variables.yml')
|
||||
@@ -299,7 +317,8 @@ class TestHttpRunner(ApiServerUnittest):
|
||||
summary = runner.summary
|
||||
self.assertTrue(summary["success"])
|
||||
self.assertIn("token", summary["details"][0]["in_out"]["out"])
|
||||
self.assertIn("user_agent", summary["details"][0]["in_out"]["in"])
|
||||
# TODO: add
|
||||
# self.assertIn("user_agent", summary["details"][0]["in_out"]["in"])
|
||||
|
||||
def test_run_testcase_with_variables_mapping(self):
|
||||
testcase_file_path = os.path.join(
|
||||
@@ -311,66 +330,238 @@ class TestHttpRunner(ApiServerUnittest):
|
||||
summary = runner.summary
|
||||
self.assertTrue(summary["success"])
|
||||
self.assertIn("token", summary["details"][0]["in_out"]["out"])
|
||||
self.assertGreater(len(summary["details"][0]["in_out"]["in"]), 3)
|
||||
# TODO: add
|
||||
# self.assertGreater(len(summary["details"][0]["in_out"]["in"]), 3)
|
||||
|
||||
def test_run_testcase_with_parameters(self):
|
||||
testcase_file_path = os.path.join(
|
||||
os.getcwd(), 'tests/data/demo_parameters.yml')
|
||||
runner = HttpRunner().run(testcase_file_path)
|
||||
summary = runner.summary
|
||||
self.assertEqual(
|
||||
summary["details"][0]["in_out"]["in"]["user_agent"],
|
||||
"iOS/10.1"
|
||||
)
|
||||
self.assertEqual(
|
||||
summary["details"][2]["in_out"]["in"]["user_agent"],
|
||||
"iOS/10.2"
|
||||
)
|
||||
self.assertEqual(
|
||||
summary["details"][4]["in_out"]["in"]["user_agent"],
|
||||
"iOS/10.3"
|
||||
)
|
||||
# TODO: add parameterize
|
||||
# self.assertEqual(
|
||||
# summary["details"][0]["in_out"]["in"]["user_agent"],
|
||||
# "iOS/10.1"
|
||||
# )
|
||||
# self.assertEqual(
|
||||
# summary["details"][2]["in_out"]["in"]["user_agent"],
|
||||
# "iOS/10.2"
|
||||
# )
|
||||
# self.assertEqual(
|
||||
# summary["details"][4]["in_out"]["in"]["user_agent"],
|
||||
# "iOS/10.3"
|
||||
# )
|
||||
self.assertTrue(summary["success"])
|
||||
self.assertEqual(len(summary["details"]), 3 * 2)
|
||||
self.assertEqual(summary["stat"]["testsRun"], 3 * 2)
|
||||
self.assertEqual(len(summary["details"]), 1)
|
||||
# self.assertEqual(len(summary["details"]), 3 * 2)
|
||||
# self.assertEqual(summary["stat"]["testsRun"], 3 * 2)
|
||||
self.assertIn("in", summary["details"][0]["in_out"])
|
||||
self.assertIn("out", summary["details"][0]["in_out"])
|
||||
|
||||
def test_run_testcase_with_parameters_name(self):
|
||||
testcase_file_path = os.path.join(
|
||||
os.getcwd(), 'tests/data/demo_parameters.yml')
|
||||
testcases = loader.load_tests(testcase_file_path)
|
||||
parsed_testcases = parser.parse_tests(testcases)
|
||||
tests_mapping = loader.load_tests(testcase_file_path)
|
||||
parser.parse_tests(tests_mapping)
|
||||
runner = HttpRunner()
|
||||
test_suite = runner._add_tests(parsed_testcases)
|
||||
test_suite = runner._add_tests(tests_mapping)
|
||||
|
||||
self.assertEqual(
|
||||
test_suite._tests[0].teststeps[0]['name'],
|
||||
'get token with iOS/10.1 and test1'
|
||||
)
|
||||
self.assertEqual(
|
||||
test_suite._tests[1].teststeps[0]['name'],
|
||||
'get token with iOS/10.1 and test2'
|
||||
)
|
||||
self.assertEqual(
|
||||
test_suite._tests[2].teststeps[0]['name'],
|
||||
'get token with iOS/10.2 and test1'
|
||||
)
|
||||
self.assertEqual(
|
||||
test_suite._tests[3].teststeps[0]['name'],
|
||||
'get token with iOS/10.2 and test2'
|
||||
)
|
||||
self.assertEqual(
|
||||
test_suite._tests[4].teststeps[0]['name'],
|
||||
'get token with iOS/10.3 and test1'
|
||||
)
|
||||
self.assertEqual(
|
||||
test_suite._tests[5].teststeps[0]['name'],
|
||||
'get token with iOS/10.3 and test2'
|
||||
)
|
||||
# TODO: add parameterize
|
||||
# self.assertEqual(
|
||||
# test_suite._tests[1].teststeps[0]['name'],
|
||||
# 'get token with iOS/10.1 and test2'
|
||||
# )
|
||||
# self.assertEqual(
|
||||
# test_suite._tests[2].teststeps[0]['name'],
|
||||
# 'get token with iOS/10.2 and test1'
|
||||
# )
|
||||
# self.assertEqual(
|
||||
# test_suite._tests[3].teststeps[0]['name'],
|
||||
# 'get token with iOS/10.2 and test2'
|
||||
# )
|
||||
# self.assertEqual(
|
||||
# test_suite._tests[4].teststeps[0]['name'],
|
||||
# 'get token with iOS/10.3 and test1'
|
||||
# )
|
||||
# self.assertEqual(
|
||||
# test_suite._tests[5].teststeps[0]['name'],
|
||||
# 'get token with iOS/10.3 and test2'
|
||||
# )
|
||||
|
||||
def test_validate_response_content(self):
|
||||
testcase_file_path = os.path.join(
|
||||
os.getcwd(), 'tests/httpbin/basic.yml')
|
||||
runner = HttpRunner().run(testcase_file_path)
|
||||
self.assertTrue(runner.summary["success"])
|
||||
|
||||
|
||||
class TestApi(ApiServerUnittest):
|
||||
|
||||
def test_testcase_loader(self):
|
||||
testcase_path = "tests/testcases/setup.yml"
|
||||
tests_mapping = loader.load_tests(testcase_path)
|
||||
|
||||
project_mapping = tests_mapping["project_mapping"]
|
||||
self.assertIsInstance(project_mapping, dict)
|
||||
self.assertIn("PWD", project_mapping)
|
||||
self.assertIn("functions", project_mapping)
|
||||
self.assertIn("env", project_mapping)
|
||||
|
||||
testcases = tests_mapping["testcases"]
|
||||
self.assertIsInstance(testcases, list)
|
||||
self.assertEqual(len(testcases), 1)
|
||||
testcase_config = testcases[0]["config"]
|
||||
self.assertEqual(testcase_config["name"], "setup and reset all.")
|
||||
self.assertIn("path", testcase_config)
|
||||
|
||||
testcase_teststeps = testcases[0]["teststeps"]
|
||||
self.assertEqual(len(testcase_teststeps), 2)
|
||||
self.assertIn("api", testcase_teststeps[0])
|
||||
self.assertEqual(testcase_teststeps[0]["name"], "get token (setup)")
|
||||
self.assertIsInstance(testcase_teststeps[0]["variables"], list)
|
||||
self.assertIn("api_def", testcase_teststeps[0])
|
||||
self.assertEqual(testcase_teststeps[0]["api_def"]["request"]["url"], "/api/get-token")
|
||||
|
||||
def test_testcase_parser(self):
|
||||
testcase_path = "tests/testcases/setup.yml"
|
||||
tests_mapping = loader.load_tests(testcase_path)
|
||||
|
||||
parser.parse_tests(tests_mapping)
|
||||
parsed_testcases = tests_mapping["testcases"]
|
||||
|
||||
self.assertEqual(len(parsed_testcases), 1)
|
||||
|
||||
self.assertNotIn("variables", parsed_testcases[0]["config"])
|
||||
self.assertEqual(len(parsed_testcases[0]["teststeps"]), 2)
|
||||
|
||||
teststep1 = parsed_testcases[0]["teststeps"][0]
|
||||
self.assertEqual(teststep1["name"], "get token (setup)")
|
||||
self.assertNotIn("api_def", teststep1)
|
||||
self.assertEqual(teststep1["variables"]["device_sn"], "TESTCASE_SETUP_XXX")
|
||||
self.assertEqual(teststep1["request"]["url"], "http://127.0.0.1:5000/api/get-token")
|
||||
|
||||
def test_testcase_add_tests(self):
|
||||
testcase_path = "tests/testcases/setup.yml"
|
||||
tests_mapping = loader.load_tests(testcase_path)
|
||||
|
||||
parser.parse_tests(tests_mapping)
|
||||
runner = HttpRunner()
|
||||
test_suite = runner._add_tests(tests_mapping)
|
||||
|
||||
self.assertEqual(len(test_suite._tests), 1)
|
||||
teststeps = test_suite._tests[0].teststeps
|
||||
self.assertEqual(teststeps[0]["name"], "get token (setup)")
|
||||
self.assertEqual(teststeps[0]["variables"]["device_sn"], "TESTCASE_SETUP_XXX")
|
||||
self.assertIn("api", teststeps[0])
|
||||
|
||||
def test_testcase_simple_run_suite(self):
|
||||
testcase_path = "tests/testcases/setup.yml"
|
||||
tests_mapping = loader.load_tests(testcase_path)
|
||||
parser.parse_tests(tests_mapping)
|
||||
runner = HttpRunner()
|
||||
test_suite = runner._add_tests(tests_mapping)
|
||||
tests_results = runner._run_suite(test_suite)
|
||||
self.assertEqual(len(tests_results[0][1].records), 2)
|
||||
|
||||
def test_testcase_complex_run_suite(self):
|
||||
testcase_path = "tests/testcases/create_and_check.yml"
|
||||
tests_mapping = loader.load_tests(testcase_path)
|
||||
parser.parse_tests(tests_mapping)
|
||||
runner = HttpRunner()
|
||||
test_suite = runner._add_tests(tests_mapping)
|
||||
tests_results = runner._run_suite(test_suite)
|
||||
self.assertEqual(len(tests_results[0][1].records), 4)
|
||||
|
||||
results = tests_results[0][1]
|
||||
self.assertEqual(
|
||||
results.records[0]["name"],
|
||||
"setup and reset all (override)."
|
||||
)
|
||||
self.assertEqual(
|
||||
results.records[1]["name"],
|
||||
"make sure user 9001 does not exist"
|
||||
)
|
||||
|
||||
def test_testsuite_loader(self):
|
||||
testcase_path = "tests/testsuites/create_users.yml"
|
||||
tests_mapping = loader.load_tests(testcase_path)
|
||||
|
||||
project_mapping = tests_mapping["project_mapping"]
|
||||
self.assertIsInstance(project_mapping, dict)
|
||||
self.assertIn("PWD", project_mapping)
|
||||
self.assertIn("functions", project_mapping)
|
||||
self.assertIn("env", project_mapping)
|
||||
|
||||
testcases = tests_mapping["testcases"]
|
||||
self.assertIsInstance(testcases, list)
|
||||
self.assertEqual(len(testcases), 1)
|
||||
testcase_config = testcases[0]["config"]
|
||||
self.assertEqual(testcase_config["name"], "create users with uid")
|
||||
self.assertIn("path", testcase_config)
|
||||
|
||||
testcase_teststeps = testcases[0]["teststeps"]
|
||||
self.assertEqual(len(testcase_teststeps), 2)
|
||||
self.assertIn("testcase_def", testcase_teststeps[0])
|
||||
self.assertEqual(testcase_teststeps[0]["name"], "create user 1000 and check result.")
|
||||
self.assertIsInstance(testcase_teststeps[0]["testcase_def"], dict)
|
||||
self.assertEqual(testcase_teststeps[0]["testcase_def"]["config"]["name"], "create user and check result.")
|
||||
self.assertEqual(len(testcase_teststeps[0]["testcase_def"]["teststeps"]), 4)
|
||||
self.assertEqual(testcase_teststeps[0]["testcase_def"]["teststeps"][0]["name"], "setup and reset all (override).")
|
||||
|
||||
def test_testsuite_parser(self):
|
||||
testcase_path = "tests/testsuites/create_users.yml"
|
||||
tests_mapping = loader.load_tests(testcase_path)
|
||||
|
||||
parser.parse_tests(tests_mapping)
|
||||
|
||||
parsed_testcases = tests_mapping["testcases"]
|
||||
self.assertEqual(len(parsed_testcases), 1)
|
||||
self.assertEqual(len(parsed_testcases[0]["teststeps"]), 2)
|
||||
|
||||
testcase1 = parsed_testcases[0]["teststeps"][0]
|
||||
self.assertEqual(testcase1["config"]["name"], "create user 1000 and check result.")
|
||||
self.assertNotIn("testcase_def", testcase1)
|
||||
self.assertEqual(len(testcase1["teststeps"]), 4)
|
||||
self.assertEqual(
|
||||
testcase1["teststeps"][0]["teststeps"][0]["request"]["url"],
|
||||
"http://127.0.0.1:5000/api/get-token"
|
||||
)
|
||||
self.assertEqual(len(testcase1["teststeps"][0]["teststeps"][0]["variables"]["device_sn"]), 15)
|
||||
|
||||
def test_testsuite_add_tests(self):
|
||||
testcase_path = "tests/testsuites/create_users.yml"
|
||||
tests_mapping = loader.load_tests(testcase_path)
|
||||
|
||||
parser.parse_tests(tests_mapping)
|
||||
runner = HttpRunner()
|
||||
test_suite = runner._add_tests(tests_mapping)
|
||||
|
||||
self.assertEqual(len(test_suite._tests), 1)
|
||||
teststeps = test_suite._tests[0].teststeps
|
||||
self.assertEqual(teststeps[0]["config"]["name"], "create user 1000 and check result.")
|
||||
|
||||
def test_testsuite_run_suite(self):
|
||||
testcase_path = "tests/testsuites/create_users.yml"
|
||||
tests_mapping = loader.load_tests(testcase_path)
|
||||
|
||||
parser.parse_tests(tests_mapping)
|
||||
|
||||
runner = HttpRunner()
|
||||
test_suite = runner._add_tests(tests_mapping)
|
||||
tests_results = runner._run_suite(test_suite)
|
||||
|
||||
self.assertEqual(len(tests_results[0][1].records), 2)
|
||||
|
||||
results = tests_results[0][1]
|
||||
self.assertEqual(
|
||||
results.records[0]["name"],
|
||||
"create user 1000 and check result."
|
||||
)
|
||||
self.assertEqual(
|
||||
results.records[1]["name"],
|
||||
"create user 1001 and check result."
|
||||
)
|
||||
@@ -2,64 +2,38 @@ import os
|
||||
import time
|
||||
|
||||
import requests
|
||||
from httprunner import context, exceptions, loader, response
|
||||
from httprunner import context, exceptions, loader, response, utils
|
||||
from tests.base import ApiServerUnittest
|
||||
|
||||
|
||||
class TestContext(ApiServerUnittest):
|
||||
|
||||
def setUp(self):
|
||||
project_mapping = loader.load_project_tests(os.path.join(os.getcwd(), "tests"))
|
||||
self.context = context.Context(
|
||||
variables={"SECRET_KEY": "DebugTalk"},
|
||||
functions=project_mapping["functions"]
|
||||
loader.load_project_tests(os.path.join(os.getcwd(), "tests"))
|
||||
project_mapping = loader.project_mapping
|
||||
self.context = context.SessionContext(
|
||||
functions=project_mapping["functions"],
|
||||
variables={"SECRET_KEY": "DebugTalk"}
|
||||
)
|
||||
testcase_file_path = os.path.join(os.getcwd(), 'tests/data/demo_binds.yml')
|
||||
self.testcases = loader.load_file(testcase_file_path)
|
||||
|
||||
def test_init_context_functions(self):
|
||||
context_functions = self.context.TESTCASE_SHARED_FUNCTIONS_MAPPING
|
||||
context_functions = self.context.FUNCTIONS_MAPPING
|
||||
self.assertIn("gen_md5", context_functions)
|
||||
|
||||
def test_init_context_variables(self):
|
||||
def test_init_teststep_variables(self):
|
||||
self.assertEqual(
|
||||
self.context.teststep_variables_mapping["SECRET_KEY"],
|
||||
"DebugTalk"
|
||||
)
|
||||
self.assertEqual(
|
||||
self.context.testcase_runtime_variables_mapping["SECRET_KEY"],
|
||||
"DebugTalk"
|
||||
self.context.teststep_variables_mapping,
|
||||
{'SECRET_KEY': 'DebugTalk'}
|
||||
)
|
||||
|
||||
def test_update_context_testcase_level(self):
|
||||
variables = [
|
||||
{"TOKEN": "debugtalk"},
|
||||
{"data": '{"name": "user", "password": "123456"}'}
|
||||
]
|
||||
self.context.update_context_variables(variables, "testcase")
|
||||
def test_update_seesion_variables(self):
|
||||
self.context.update_seesion_variables({"TOKEN": "debugtalk"})
|
||||
self.assertEqual(
|
||||
self.context.teststep_variables_mapping["TOKEN"],
|
||||
self.context.session_variables_mapping["TOKEN"],
|
||||
"debugtalk"
|
||||
)
|
||||
self.assertEqual(
|
||||
self.context.testcase_runtime_variables_mapping["TOKEN"],
|
||||
"debugtalk"
|
||||
)
|
||||
|
||||
def test_update_context_teststep_level(self):
|
||||
variables = [
|
||||
{"TOKEN": "debugtalk"},
|
||||
{"data": '{"name": "user", "password": "123456"}'}
|
||||
]
|
||||
self.context.update_context_variables(variables, "teststep")
|
||||
self.assertEqual(
|
||||
self.context.teststep_variables_mapping["TOKEN"],
|
||||
"debugtalk"
|
||||
)
|
||||
self.assertNotIn(
|
||||
"TOKEN",
|
||||
self.context.testcase_runtime_variables_mapping
|
||||
)
|
||||
|
||||
def test_eval_content_functions(self):
|
||||
content = "${sleep_N_secs(1)}"
|
||||
@@ -82,29 +56,6 @@ class TestContext(ApiServerUnittest):
|
||||
# "abcDebugTalkdef"
|
||||
# )
|
||||
|
||||
def test_update_testcase_runtime_variables_mapping(self):
|
||||
variables = {"abc": 123}
|
||||
self.context.update_testcase_runtime_variables_mapping(variables)
|
||||
self.assertEqual(
|
||||
self.context.testcase_runtime_variables_mapping["abc"],
|
||||
123
|
||||
)
|
||||
self.assertEqual(
|
||||
self.context.teststep_variables_mapping["abc"],
|
||||
123
|
||||
)
|
||||
|
||||
def test_update_teststep_variables_mapping(self):
|
||||
self.context.update_teststep_variables_mapping("abc", 123)
|
||||
self.assertEqual(
|
||||
self.context.teststep_variables_mapping["abc"],
|
||||
123
|
||||
)
|
||||
self.assertNotIn(
|
||||
"abc",
|
||||
self.context.testcase_runtime_variables_mapping
|
||||
)
|
||||
|
||||
def test_get_parsed_request(self):
|
||||
variables = [
|
||||
{"TOKEN": "debugtalk"},
|
||||
@@ -112,7 +63,7 @@ class TestContext(ApiServerUnittest):
|
||||
{"data": '{"name": "user", "password": "123456"}'},
|
||||
{"authorization": "${gen_md5($TOKEN, $data, $random)}"}
|
||||
]
|
||||
self.context.update_context_variables(variables, "teststep")
|
||||
self.context.init_teststep_variables(variables)
|
||||
|
||||
request = {
|
||||
"url": "http://127.0.0.1:5000/api/users/1000",
|
||||
@@ -125,7 +76,7 @@ class TestContext(ApiServerUnittest):
|
||||
},
|
||||
"data": "$data"
|
||||
}
|
||||
parsed_request = self.context.get_parsed_request(request, level="teststep")
|
||||
parsed_request = self.context.eval_content(request)
|
||||
self.assertIn("authorization", parsed_request["headers"])
|
||||
self.assertEqual(len(parsed_request["headers"]["authorization"]), 32)
|
||||
self.assertIn("random", parsed_request["headers"])
|
||||
@@ -159,7 +110,7 @@ class TestContext(ApiServerUnittest):
|
||||
{"resp_status_code": 200},
|
||||
{"resp_body_success": True}
|
||||
]
|
||||
self.context.update_context_variables(variables, "teststep")
|
||||
self.context.init_teststep_variables(variables)
|
||||
|
||||
with self.assertRaises(exceptions.ValidationFailure):
|
||||
self.context.validate(validators, resp_obj)
|
||||
@@ -174,7 +125,7 @@ class TestContext(ApiServerUnittest):
|
||||
{"resp_status_code": 201},
|
||||
{"resp_body_success": True}
|
||||
]
|
||||
self.context.update_context_variables(variables, "teststep")
|
||||
self.context.init_teststep_variables(variables)
|
||||
self.context.validate(validators, resp_obj)
|
||||
|
||||
def test_validate_exception(self):
|
||||
@@ -188,7 +139,7 @@ class TestContext(ApiServerUnittest):
|
||||
{"check": "$resp_status_code", "comparator": "eq", "expect": 201}
|
||||
]
|
||||
variables = []
|
||||
self.context.update_context_variables(variables, "teststep")
|
||||
self.context.init_teststep_variables(variables)
|
||||
|
||||
with self.assertRaises(exceptions.VariableNotFound):
|
||||
self.context.validate(validators, resp_obj)
|
||||
@@ -197,7 +148,7 @@ class TestContext(ApiServerUnittest):
|
||||
variables = [
|
||||
{"resp_status_code": 200}
|
||||
]
|
||||
self.context.update_context_variables(variables, "teststep")
|
||||
self.context.init_teststep_variables(variables)
|
||||
|
||||
with self.assertRaises(exceptions.ValidationFailure):
|
||||
self.context.validate(validators, resp_obj)
|
||||
|
||||
@@ -195,10 +195,12 @@ class TestModuleLoader(unittest.TestCase):
|
||||
self.assertNotIn("is_py3", module_functions)
|
||||
|
||||
def test_load_debugtalk_module(self):
|
||||
project_mapping = loader.load_project_tests(os.path.join(os.getcwd(), "httprunner"))
|
||||
loader.load_project_tests(os.path.join(os.getcwd(), "httprunner"))
|
||||
project_mapping = loader.project_mapping
|
||||
self.assertNotIn("alter_response", project_mapping["functions"])
|
||||
|
||||
project_mapping = loader.load_project_tests(os.path.join(os.getcwd(), "tests"))
|
||||
loader.load_project_tests(os.path.join(os.getcwd(), "tests"))
|
||||
project_mapping = loader.project_mapping
|
||||
self.assertIn("alter_response", project_mapping["functions"])
|
||||
|
||||
is_status_code_200 = project_mapping["functions"]["is_status_code_200"]
|
||||
@@ -230,16 +232,21 @@ class TestModuleLoader(unittest.TestCase):
|
||||
def test_load_tests(self):
|
||||
testcase_file_path = os.path.join(
|
||||
os.getcwd(), 'tests/data/demo_testcase.yml')
|
||||
testcases = loader.load_tests(testcase_file_path)
|
||||
tests_mapping = loader.load_tests(testcase_file_path)
|
||||
testcases = tests_mapping["testcases"]
|
||||
self.assertIsInstance(testcases, list)
|
||||
self.assertEqual(
|
||||
testcases[0]["config"]["request"],
|
||||
'${get_default_request()}'
|
||||
)
|
||||
self.assertEqual(testcases[0]["config"]["name"], '123$var_a')
|
||||
self.assertIn(
|
||||
"sum_two",
|
||||
testcases[0]["config"]["functions"]
|
||||
tests_mapping["project_mapping"]["functions"]
|
||||
)
|
||||
self.assertEqual(
|
||||
testcases[0]["config"]["variables"][1]["var_c"],
|
||||
"${sum_two(1, 2)}"
|
||||
)
|
||||
self.assertEqual(
|
||||
testcases[0]["config"]["variables"][2]["PROJECT_KEY"],
|
||||
"${ENV(PROJECT_KEY)}"
|
||||
)
|
||||
|
||||
|
||||
@@ -247,198 +254,98 @@ class TestSuiteLoader(unittest.TestCase):
|
||||
|
||||
@classmethod
|
||||
def setUpClass(cls):
|
||||
cls.project_mapping = loader.load_project_tests(os.path.join(os.getcwd(), "tests"))
|
||||
loader.load_project_tests(os.path.join(os.getcwd(), "tests"))
|
||||
cls.project_mapping = loader.project_mapping
|
||||
cls.tests_def_mapping = loader.tests_def_mapping
|
||||
|
||||
def test_load_teststeps(self):
|
||||
test_block = {
|
||||
def test_load_teststep_testcase(self):
|
||||
raw_teststep = {
|
||||
"name": "setup and reset all (override).",
|
||||
"suite": "setup_and_reset($device_sn)",
|
||||
"testcase": "testcases/setup.yml",
|
||||
"variables": [
|
||||
{"device_sn": "$device_sn"}
|
||||
],
|
||||
"output": ["token", "device_sn"]
|
||||
}
|
||||
teststeps = loader._load_teststeps(test_block, self.project_mapping)
|
||||
testcase = loader.load_teststep(raw_teststep)
|
||||
self.assertEqual(
|
||||
"setup and reset all (override).",
|
||||
testcase["name"]
|
||||
)
|
||||
teststeps = testcase["testcase_def"]["teststeps"]
|
||||
self.assertEqual(len(teststeps), 2)
|
||||
self.assertEqual(teststeps[0]["name"], "get token")
|
||||
self.assertEqual(teststeps[0]["name"], "get token (setup)")
|
||||
self.assertEqual(teststeps[1]["name"], "reset all users")
|
||||
|
||||
def test_load_testcase(self):
|
||||
raw_testcase = loader.load_file("tests/testcases/smoketest.yml")
|
||||
testcase = loader._load_testcase(raw_testcase, self.project_mapping)
|
||||
self.assertEqual(testcase["config"]["name"], "smoketest")
|
||||
raw_testcase = loader.load_file("tests/testsuites/create_users.yml")
|
||||
testcase = loader.load_testcase(raw_testcase)
|
||||
self.assertEqual(testcase["config"]["name"], "create users with uid")
|
||||
self.assertIn("device_sn", testcase["config"]["variables"][0])
|
||||
self.assertEqual(len(testcase["teststeps"]), 8)
|
||||
self.assertEqual(testcase["teststeps"][0]["name"], "get token")
|
||||
|
||||
def test_get_block_by_name(self):
|
||||
ref_call = "get_user($uid, $token)"
|
||||
block = loader._get_block_by_name(ref_call, "def-api", self.project_mapping)
|
||||
self.assertEqual(block["request"]["url"], "/api/users/$uid")
|
||||
self.assertEqual(block["function_meta"]["func_name"], "get_user")
|
||||
self.assertEqual(block["function_meta"]["args"], ['$uid', '$token'])
|
||||
|
||||
def test_get_block_by_name_args_mismatch(self):
|
||||
ref_call = "get_user($uid, $token, $var)"
|
||||
with self.assertRaises(exceptions.ParamsError):
|
||||
loader._get_block_by_name(ref_call, "def-api", self.project_mapping)
|
||||
|
||||
def test_override_block(self):
|
||||
def_block = loader._get_block_by_name(
|
||||
"get_token($user_agent, $device_sn, $os_platform, $app_version)",
|
||||
"def-api",
|
||||
self.project_mapping
|
||||
)
|
||||
test_block = {
|
||||
"name": "override block",
|
||||
"times": 3,
|
||||
"variables": [
|
||||
{"var": 123}
|
||||
],
|
||||
'request': {
|
||||
'url': '/api/get-token',
|
||||
'method': 'POST',
|
||||
'headers': {'user_agent': '$user_agent', 'device_sn': '$device_sn', 'os_platform': '$os_platform', 'app_version': '$app_version'},
|
||||
'json': {'sign': '${get_sign($user_agent, $device_sn, $os_platform, $app_version)}'}
|
||||
},
|
||||
'validate': [
|
||||
{'eq': ['status_code', 201]},
|
||||
{'len_eq': ['content.token', 32]}
|
||||
]
|
||||
}
|
||||
|
||||
extended_block = loader._extend_block(test_block, def_block)
|
||||
self.assertEqual(extended_block["name"], "override block")
|
||||
self.assertIn({'var': 123}, extended_block["variables"])
|
||||
self.assertIn({'check': 'status_code', 'expect': 201, 'comparator': 'eq'}, extended_block["validate"])
|
||||
self.assertIn({'check': 'content.token', 'comparator': 'len_eq', 'expect': 32}, extended_block["validate"])
|
||||
self.assertEqual(extended_block["times"], 3)
|
||||
|
||||
def test_get_test_definition_api(self):
|
||||
api_def = loader._get_test_definition("get_headers", "def-api", self.project_mapping)
|
||||
self.assertEqual(api_def["request"]["url"], "/headers")
|
||||
self.assertEqual(len(api_def["setup_hooks"]), 2)
|
||||
self.assertEqual(len(api_def["teardown_hooks"]), 1)
|
||||
|
||||
with self.assertRaises(exceptions.ApiNotFound):
|
||||
loader._get_test_definition("get_token_XXX", "def-api", self.project_mapping)
|
||||
|
||||
def test_get_test_definition_suite(self):
|
||||
api_def = loader._get_test_definition("create_and_check", "def-testcase", self.project_mapping)
|
||||
self.assertEqual(api_def["config"]["name"], "create user and check result.")
|
||||
|
||||
with self.assertRaises(exceptions.TestcaseNotFound):
|
||||
loader._get_test_definition("create_and_check_XXX", "def-testcase", self.project_mapping)
|
||||
|
||||
def test_extend_validators(self):
|
||||
def_validators = [
|
||||
{'eq': ['v1', 200]},
|
||||
{"check": "s2", "expect": 16, "comparator": "len_eq"}
|
||||
]
|
||||
current_validators = [
|
||||
{"check": "v1", "expect": 201},
|
||||
{'len_eq': ['s3', 12]}
|
||||
]
|
||||
|
||||
extended_validators = loader._extend_validators(def_validators, current_validators)
|
||||
self.assertIn(
|
||||
{"check": "v1", "expect": 201, "comparator": "eq"},
|
||||
extended_validators
|
||||
)
|
||||
self.assertIn(
|
||||
{"check": "s2", "expect": 16, "comparator": "len_eq"},
|
||||
extended_validators
|
||||
)
|
||||
self.assertIn(
|
||||
{"check": "s3", "expect": 12, "comparator": "len_eq"},
|
||||
extended_validators
|
||||
)
|
||||
|
||||
def test_extend_validators_with_dict(self):
|
||||
def_validators = [
|
||||
{'eq': ["a", {"v": 1}]},
|
||||
{'eq': [{"b": 1}, 200]}
|
||||
]
|
||||
current_validators = [
|
||||
{'len_eq': ['s3', 12]},
|
||||
{'eq': [{"b": 1}, 201]}
|
||||
]
|
||||
|
||||
extended_validators = loader._extend_validators(def_validators, current_validators)
|
||||
self.assertEqual(len(extended_validators), 3)
|
||||
self.assertIn({'check': {'b': 1}, 'expect': 201, 'comparator': 'eq'}, extended_validators)
|
||||
self.assertNotIn({'check': {'b': 1}, 'expect': 200, 'comparator': 'eq'}, extended_validators)
|
||||
|
||||
def test_extend_variables(self):
|
||||
def_variables = [{"var1": "val1"}, {"var2": "val2"}]
|
||||
ref_variables = [{"var1": "val111"}, {"var3": "val3"}]
|
||||
|
||||
extended_variables = loader._extend_variables(def_variables, ref_variables)
|
||||
self.assertIn(
|
||||
{"var1": "val111"},
|
||||
extended_variables
|
||||
)
|
||||
self.assertIn(
|
||||
{"var2": "val2"},
|
||||
extended_variables
|
||||
)
|
||||
self.assertIn(
|
||||
{"var3": "val3"},
|
||||
extended_variables
|
||||
)
|
||||
self.assertEqual(len(testcase["teststeps"]), 2)
|
||||
self.assertEqual(testcase["teststeps"][0]["name"], "create user 1000 and check result.")
|
||||
self.assertEqual(testcase["teststeps"][0]["testcase_def"]["config"]["name"], "create user and check result.")
|
||||
|
||||
def test_load_testcases_by_path_files(self):
|
||||
testcases_list = []
|
||||
|
||||
# absolute file path
|
||||
path = os.path.join(
|
||||
os.getcwd(), 'tests/data/demo_testcase_hardcode.json')
|
||||
testcases_list = loader.load_tests(path)
|
||||
tests_mapping = loader.load_tests(path)
|
||||
project_mapping = tests_mapping["project_mapping"]
|
||||
testcases_list = tests_mapping["testcases"]
|
||||
self.assertEqual(len(testcases_list), 1)
|
||||
self.assertEqual(len(testcases_list[0]["teststeps"]), 3)
|
||||
self.assertIn("get_sign", testcases_list[0]["config"]["functions"])
|
||||
self.assertIn("get_sign", project_mapping["functions"])
|
||||
|
||||
# relative file path
|
||||
path = 'tests/data/demo_testcase_hardcode.yml'
|
||||
testcases_list = loader.load_tests(path)
|
||||
tests_mapping = loader.load_tests(path)
|
||||
project_mapping = tests_mapping["project_mapping"]
|
||||
testcases_list = tests_mapping["testcases"]
|
||||
self.assertEqual(len(testcases_list), 1)
|
||||
self.assertEqual(len(testcases_list[0]["teststeps"]), 3)
|
||||
self.assertIn("get_sign", testcases_list[0]["config"]["functions"])
|
||||
self.assertIn("get_sign", project_mapping["functions"])
|
||||
|
||||
# list/set container with file(s)
|
||||
path = [
|
||||
os.path.join(os.getcwd(), 'tests/data/demo_testcase_hardcode.json'),
|
||||
'tests/data/demo_testcase_hardcode.yml'
|
||||
]
|
||||
testcases_list = loader.load_tests(path)
|
||||
self.assertEqual(len(testcases_list), 2)
|
||||
self.assertEqual(len(testcases_list[0]["teststeps"]), 3)
|
||||
self.assertEqual(len(testcases_list[1]["teststeps"]), 3)
|
||||
testcases_list.extend(testcases_list)
|
||||
self.assertEqual(len(testcases_list), 4)
|
||||
# TODO: list/set container with file(s)
|
||||
# path = [
|
||||
# os.path.join(os.getcwd(), 'tests/data/demo_testcase_hardcode.json'),
|
||||
# 'tests/data/demo_testcase_hardcode.yml'
|
||||
# ]
|
||||
# testcases_list = loader.load_tests(path)
|
||||
# self.assertEqual(len(testcases_list), 2)
|
||||
# self.assertEqual(len(testcases_list[0]["teststeps"]), 3)
|
||||
# self.assertEqual(len(testcases_list[1]["teststeps"]), 3)
|
||||
# testcases_list.extend(testcases_list)
|
||||
# self.assertEqual(len(testcases_list), 4)
|
||||
|
||||
for testcase in testcases_list:
|
||||
for teststep in testcase["teststeps"]:
|
||||
self.assertIn('name', teststep)
|
||||
self.assertIn('request', teststep)
|
||||
self.assertIn('url', teststep['request'])
|
||||
self.assertIn('method', teststep['request'])
|
||||
# for testcase in testcases_list:
|
||||
# for teststep in testcase["teststeps"]:
|
||||
# self.assertIn('name', teststep)
|
||||
# self.assertIn('request', teststep)
|
||||
# self.assertIn('url', teststep['request'])
|
||||
# self.assertIn('method', teststep['request'])
|
||||
|
||||
def test_load_testcases_by_path_folder(self):
|
||||
# absolute folder path
|
||||
path = os.path.join(os.getcwd(), 'tests/data')
|
||||
testcase_list_1 = loader.load_tests(path)
|
||||
tests_mapping = loader.load_tests(path)
|
||||
testcase_list_1 = tests_mapping["testcases"]
|
||||
self.assertGreater(len(testcase_list_1), 4)
|
||||
|
||||
# relative folder path
|
||||
path = 'tests/data/'
|
||||
testcase_list_2 = loader.load_tests(path)
|
||||
tests_mapping = loader.load_tests(path)
|
||||
testcase_list_2 = tests_mapping["testcases"]
|
||||
self.assertEqual(len(testcase_list_1), len(testcase_list_2))
|
||||
|
||||
# list/set container with file(s)
|
||||
path = [
|
||||
os.path.join(os.getcwd(), 'tests/data'),
|
||||
'tests/data/'
|
||||
]
|
||||
testcase_list_3 = loader.load_tests(path)
|
||||
self.assertEqual(len(testcase_list_3), 2 * len(testcase_list_1))
|
||||
# TODO: list/set container with file(s)
|
||||
# path = [
|
||||
# os.path.join(os.getcwd(), 'tests/data'),
|
||||
# 'tests/data/'
|
||||
# ]
|
||||
# tests_mapping = loader.load_tests(path)
|
||||
# testcase_list_3 = tests_mapping["testcases"]
|
||||
# self.assertEqual(len(testcase_list_3), 2 * len(testcase_list_1))
|
||||
|
||||
def test_load_testcases_by_path_not_exist(self):
|
||||
# absolute folder path
|
||||
@@ -451,34 +358,68 @@ class TestSuiteLoader(unittest.TestCase):
|
||||
with self.assertRaises(exceptions.FileNotFound):
|
||||
loader.load_tests(path)
|
||||
|
||||
# list/set container with file(s)
|
||||
path = [
|
||||
os.path.join(os.getcwd(), 'tests/data_not_exist'),
|
||||
'tests/data_not_exist/'
|
||||
]
|
||||
with self.assertRaises(exceptions.FileNotFound):
|
||||
loader.load_tests(path)
|
||||
# TODO: list/set container with file(s)
|
||||
# path = [
|
||||
# os.path.join(os.getcwd(), 'tests/data_not_exist'),
|
||||
# 'tests/data_not_exist/'
|
||||
# ]
|
||||
# with self.assertRaises(exceptions.FileNotFound):
|
||||
# loader.load_tests(path)
|
||||
|
||||
def test_load_testcases_by_path_layered(self):
|
||||
def test_load_testcases_with_api_ref(self):
|
||||
path = os.path.join(
|
||||
os.getcwd(), 'tests/data/demo_testcase_layer.yml')
|
||||
testcases_list = loader.load_tests(path)
|
||||
self.assertIn("variables", testcases_list[0]["config"])
|
||||
self.assertIn("request", testcases_list[0]["config"])
|
||||
|
||||
# variables in testcase teststep should override api's
|
||||
tests_mapping = loader.load_tests(path)
|
||||
project_mapping = tests_mapping["project_mapping"]
|
||||
testcases_list = tests_mapping["testcases"]
|
||||
self.assertIn({'device_sn': '${gen_random_string(15)}'}, testcases_list[0]["config"]["variables"])
|
||||
self.assertIn("gen_md5", project_mapping["functions"])
|
||||
self.assertIn("base_url", testcases_list[0]["config"])
|
||||
teststep0 = testcases_list[0]["teststeps"][0]
|
||||
self.assertEqual(
|
||||
testcases_list[0]["teststeps"][3]["variables"][0]["user_name"],
|
||||
"user1"
|
||||
"get token with $user_agent, $app_version",
|
||||
teststep0["name"]
|
||||
)
|
||||
self.assertEqual(
|
||||
testcases_list[0]["teststeps"][3]["variables"][2]["uid"],
|
||||
2000
|
||||
self.assertIn("/api/get-token", teststep0["api_def"]["request"]["url"])
|
||||
self.assertIn(
|
||||
{'eq': ['status_code', 200]},
|
||||
teststep0["validate"]
|
||||
)
|
||||
|
||||
self.assertIn("request", testcases_list[0]["teststeps"][0])
|
||||
self.assertIn("url", testcases_list[0]["teststeps"][0]["request"])
|
||||
self.assertIn("validate", testcases_list[0]["teststeps"][0])
|
||||
def test_load_testcases_with_testcase_ref(self):
|
||||
path = os.path.join(
|
||||
os.getcwd(), 'tests/testsuites/create_users.yml')
|
||||
tests_mapping = loader.load_tests(path)
|
||||
project_mapping = tests_mapping["project_mapping"]
|
||||
testcases_list = tests_mapping["testcases"]
|
||||
|
||||
self.assertEqual(
|
||||
"create users with uid",
|
||||
testcases_list[0]["config"]["name"]
|
||||
)
|
||||
self.assertEqual(
|
||||
{'device_sn': '${gen_random_string(15)}'},
|
||||
testcases_list[0]["config"]["variables"][0]
|
||||
)
|
||||
testcase0 = testcases_list[0]["teststeps"][0]
|
||||
self.assertEqual(
|
||||
"create user 1000 and check result.",
|
||||
testcase0["name"]
|
||||
)
|
||||
self.assertEqual(
|
||||
"create user and check result.",
|
||||
testcase0["testcase_def"]["config"]["name"]
|
||||
)
|
||||
|
||||
testcase1 = testcases_list[0]["teststeps"][1]
|
||||
self.assertEqual(
|
||||
"create user 1001 and check result.",
|
||||
testcase1["name"]
|
||||
)
|
||||
self.assertEqual(
|
||||
{'uid': 1001},
|
||||
testcase1["variables"][0]
|
||||
)
|
||||
|
||||
def test_load_folder_content(self):
|
||||
path = os.path.join(os.getcwd(), "tests", "api")
|
||||
@@ -492,47 +433,16 @@ class TestSuiteLoader(unittest.TestCase):
|
||||
api_definition_mapping = loader.load_api_folder(path)
|
||||
self.assertIn("get_token", api_definition_mapping)
|
||||
self.assertIn("request", api_definition_mapping["get_token"])
|
||||
self.assertIn("function_meta", api_definition_mapping["get_token"])
|
||||
|
||||
def test_load_testcases_folder(self):
|
||||
path = os.path.join(os.getcwd(), "tests", "suite")
|
||||
testcases_definition_mapping = loader.load_test_folder(path)
|
||||
|
||||
self.assertIn("setup_and_reset", testcases_definition_mapping)
|
||||
self.assertIn("create_and_check", testcases_definition_mapping)
|
||||
self.assertEqual(
|
||||
testcases_definition_mapping["setup_and_reset"]["config"]["name"],
|
||||
"setup and reset all."
|
||||
)
|
||||
self.assertEqual(
|
||||
testcases_definition_mapping["setup_and_reset"]["function_meta"]["func_name"],
|
||||
"setup_and_reset"
|
||||
)
|
||||
|
||||
def test_load_testsuites_folder(self):
|
||||
path = os.path.join(os.getcwd(), "tests", "testcases")
|
||||
testsuites_definition_mapping = loader.load_test_folder(path)
|
||||
|
||||
testsute_path = os.path.join(os.getcwd(), "tests", "testcases", "smoketest.yml")
|
||||
self.assertIn(
|
||||
testsute_path,
|
||||
testsuites_definition_mapping
|
||||
)
|
||||
self.assertEqual(
|
||||
testsuites_definition_mapping[testsute_path]["config"]["name"],
|
||||
"smoketest"
|
||||
)
|
||||
|
||||
def test_load_project_tests(self):
|
||||
project_mapping = loader.load_project_tests(os.path.join(os.getcwd(), "tests"))
|
||||
self.assertIn("get_token", project_mapping["def-api"])
|
||||
self.assertIn("setup_and_reset", project_mapping["def-testcase"])
|
||||
self.assertEqual(project_mapping["env"]["PROJECT_KEY"], "ABCDEFGH")
|
||||
loader.load_project_tests(os.path.join(os.getcwd(), "tests"))
|
||||
self.assertIn("get_token", self.tests_def_mapping["api"])
|
||||
self.assertEqual(self.project_mapping["env"]["PROJECT_KEY"], "ABCDEFGH")
|
||||
|
||||
def test_load_locust_tests(self):
|
||||
path = os.path.join(
|
||||
os.getcwd(), 'tests/data/demo_locust.yml')
|
||||
locust_tests = loader.load_locust_tests(path)
|
||||
self.assertEqual(len(locust_tests["tests"]), 10)
|
||||
self.assertEqual(locust_tests["tests"][0][0]["name"], "index")
|
||||
self.assertEqual(locust_tests["tests"][9][0]["name"], "user-agent")
|
||||
self.assertEqual(locust_tests["tests"][0]["name"], "index")
|
||||
self.assertEqual(locust_tests["tests"][9]["name"], "user-agent")
|
||||
|
||||
@@ -409,7 +409,8 @@ class TestParser(unittest.TestCase):
|
||||
)
|
||||
|
||||
def test_parse_parameters_mix(self):
|
||||
project_mapping = loader.load_project_tests(os.path.join(os.getcwd(), "tests"))
|
||||
loader.load_project_tests(os.path.join(os.getcwd(), "tests"))
|
||||
project_mapping = loader.project_mapping
|
||||
|
||||
parameters = [
|
||||
{"user_agent": ["iOS/10.1", "iOS/10.2", "iOS/10.3"]},
|
||||
@@ -431,7 +432,8 @@ class TestParser(unittest.TestCase):
|
||||
def test_parse_tests(self):
|
||||
testcase_file_path = os.path.join(
|
||||
os.getcwd(), 'tests/data/demo_testcase.yml')
|
||||
testcases = loader.load_tests(testcase_file_path)
|
||||
tests_mapping = loader.load_tests(testcase_file_path)
|
||||
testcases = tests_mapping["testcases"]
|
||||
self.assertEqual(
|
||||
testcases[0]["config"]["variables"][1]["var_c"],
|
||||
"${sum_two(1, 2)}"
|
||||
@@ -440,17 +442,15 @@ class TestParser(unittest.TestCase):
|
||||
testcases[0]["config"]["variables"][2]["PROJECT_KEY"],
|
||||
"${ENV(PROJECT_KEY)}"
|
||||
)
|
||||
|
||||
parsed_testcases = parser.parse_tests(testcases)
|
||||
self.assertEqual(parsed_testcases[0]["config"]["variables"]["var_c"], 3)
|
||||
self.assertEqual(parsed_testcases[0]["config"]["variables"]["PROJECT_KEY"], "ABCDEFGH")
|
||||
self.assertEqual(len(parsed_testcases), 2 * 2)
|
||||
self.assertEqual(
|
||||
parsed_testcases[0]["config"]["request"]["base_url"],
|
||||
"http://127.0.0.1:5000"
|
||||
)
|
||||
parser.parse_tests(tests_mapping)
|
||||
parsed_testcases = tests_mapping["testcases"]
|
||||
self.assertIsInstance(parsed_testcases, list)
|
||||
self.assertEqual(parsed_testcases[0]["config"]["name"], '12311')
|
||||
teststep1 = parsed_testcases[0]["teststeps"][0]
|
||||
self.assertEqual(teststep1["variables"]["var_c"], 3)
|
||||
self.assertEqual(teststep1["variables"]["PROJECT_KEY"], "ABCDEFGH")
|
||||
# TODO: parameters
|
||||
# self.assertEqual(len(parsed_testcases), 2 * 2)
|
||||
self.assertEqual(parsed_testcases[0]["config"]["name"], '1230')
|
||||
|
||||
def test_parse_environ(self):
|
||||
os.environ["PROJECT_KEY"] = "ABCDEFGH"
|
||||
@@ -476,3 +476,35 @@ class TestParser(unittest.TestCase):
|
||||
}
|
||||
with self.assertRaises(exceptions.ParamsError):
|
||||
parser.parse_data(content)
|
||||
|
||||
def test_extend_with_api(self):
|
||||
loader.load_project_tests(os.path.join(os.getcwd(), "tests"))
|
||||
raw_stepinfo = {
|
||||
"name": "get token",
|
||||
"api": "get_token",
|
||||
}
|
||||
api_def_dict = loader.load_teststep(raw_stepinfo)
|
||||
test_block = {
|
||||
"name": "override block",
|
||||
"times": 3,
|
||||
"variables": [
|
||||
{"var": 123}
|
||||
],
|
||||
'request': {
|
||||
'url': '/api/get-token',
|
||||
'method': 'POST',
|
||||
'headers': {'user_agent': '$user_agent', 'device_sn': '$device_sn', 'os_platform': '$os_platform', 'app_version': '$app_version'},
|
||||
'json': {'sign': '${get_sign($user_agent, $device_sn, $os_platform, $app_version)}'}
|
||||
},
|
||||
'validate': [
|
||||
{'eq': ['status_code', 201]},
|
||||
{'len_eq': ['content.token', 32]}
|
||||
]
|
||||
}
|
||||
|
||||
extended_block = parser._extend_with_api(test_block, api_def_dict)
|
||||
self.assertEqual(extended_block["name"], "override block")
|
||||
self.assertIn({'var': 123}, extended_block["variables"])
|
||||
self.assertIn({'check': 'status_code', 'expect': 201, 'comparator': 'eq'}, extended_block["validate"])
|
||||
self.assertIn({'check': 'content.token', 'comparator': 'len_eq', 'expect': 32}, extended_block["validate"])
|
||||
self.assertEqual(extended_block["times"], 3)
|
||||
|
||||
@@ -10,13 +10,16 @@ from tests.base import ApiServerUnittest
|
||||
class TestRunner(ApiServerUnittest):
|
||||
|
||||
def setUp(self):
|
||||
project_mapping = loader.load_project_tests(os.path.join(os.getcwd(), "tests"))
|
||||
loader.load_project_tests(os.path.join(os.getcwd(), "tests"))
|
||||
project_mapping = loader.project_mapping
|
||||
self.debugtalk_functions = project_mapping["functions"]
|
||||
config_dict = {
|
||||
"variables": {},
|
||||
"functions": self.debugtalk_functions
|
||||
|
||||
config = {
|
||||
"name": "XXX",
|
||||
"base_url": "http://127.0.0.1",
|
||||
"verify": False
|
||||
}
|
||||
self.test_runner = runner.Runner(config_dict)
|
||||
self.test_runner = runner.Runner(config, self.debugtalk_functions)
|
||||
self.reset_all()
|
||||
|
||||
def reset_all(self):
|
||||
@@ -35,11 +38,8 @@ class TestRunner(ApiServerUnittest):
|
||||
for testcase_file_path in testcase_file_path_list:
|
||||
testcases = loader.load_file(testcase_file_path)
|
||||
|
||||
config_dict = {
|
||||
"variables": {},
|
||||
"functions": self.debugtalk_functions
|
||||
}
|
||||
test_runner = runner.Runner(config_dict)
|
||||
config_dict = {}
|
||||
test_runner = runner.Runner(config_dict, self.debugtalk_functions)
|
||||
|
||||
test = testcases[0]["test"]
|
||||
test_runner.run_test(test)
|
||||
@@ -81,11 +81,7 @@ class TestRunner(ApiServerUnittest):
|
||||
|
||||
config_dict = {
|
||||
"name": "basic test with httpbin",
|
||||
"variables": {},
|
||||
"functions": self.debugtalk_functions,
|
||||
"request": {
|
||||
"base_url": HTTPBIN_SERVER
|
||||
},
|
||||
"base_url": HTTPBIN_SERVER,
|
||||
"setup_hooks": [
|
||||
"${sleep_N_secs(0.5)}"
|
||||
"${hook_print(setup)}"
|
||||
@@ -115,7 +111,7 @@ class TestRunner(ApiServerUnittest):
|
||||
{"check": "status_code", "expect": 200}
|
||||
]
|
||||
}
|
||||
test_runner = runner.Runner(config_dict)
|
||||
test_runner = runner.Runner(config_dict, self.debugtalk_functions)
|
||||
end_time = time.time()
|
||||
# check if testcase setup hook executed
|
||||
self.assertGreater(end_time - start_time, 0.5)
|
||||
@@ -130,11 +126,7 @@ class TestRunner(ApiServerUnittest):
|
||||
def test_run_testcase_with_hooks_modify_request(self):
|
||||
config_dict = {
|
||||
"name": "basic test with httpbin",
|
||||
"variables": {},
|
||||
"functions": self.debugtalk_functions,
|
||||
"request": {
|
||||
"base_url": HTTPBIN_SERVER
|
||||
}
|
||||
"base_url": HTTPBIN_SERVER
|
||||
}
|
||||
test = {
|
||||
"name": "modify request headers",
|
||||
@@ -158,7 +150,7 @@ class TestRunner(ApiServerUnittest):
|
||||
{"check": "content.headers.Os-Platform", "expect": "android"}
|
||||
]
|
||||
}
|
||||
test_runner = runner.Runner(config_dict)
|
||||
test_runner = runner.Runner(config_dict, self.debugtalk_functions)
|
||||
test_runner.run_test(test)
|
||||
|
||||
def test_run_testcase_with_teardown_hooks_success(self):
|
||||
@@ -183,9 +175,6 @@ class TestRunner(ApiServerUnittest):
|
||||
],
|
||||
"teardown_hooks": ["${teardown_hook_sleep_N_secs($response, 2)}"]
|
||||
}
|
||||
config_dict = {}
|
||||
self.test_runner.init_test(config_dict, "testcase")
|
||||
|
||||
start_time = time.time()
|
||||
self.test_runner.run_test(test)
|
||||
end_time = time.time()
|
||||
@@ -214,9 +203,6 @@ class TestRunner(ApiServerUnittest):
|
||||
],
|
||||
"teardown_hooks": ["${teardown_hook_sleep_N_secs($response, 2)}"]
|
||||
}
|
||||
config_dict = {}
|
||||
self.test_runner.init_test(config_dict, "testcase")
|
||||
|
||||
start_time = time.time()
|
||||
self.test_runner.run_test(test)
|
||||
end_time = time.time()
|
||||
@@ -226,8 +212,8 @@ class TestRunner(ApiServerUnittest):
|
||||
def test_run_testcase_with_empty_header(self):
|
||||
testcase_file_path = os.path.join(
|
||||
os.getcwd(), 'tests/data/test_bugfix.yml')
|
||||
testcases = loader.load_tests(testcase_file_path)
|
||||
testcase = testcases[0]
|
||||
tests_mapping = loader.load_tests(testcase_file_path)
|
||||
testcase = tests_mapping["testcases"][0]
|
||||
config_dict_headers = testcase["config"]["request"]["headers"]
|
||||
test_dict_headers = testcase["teststeps"][0]["request"]["headers"]
|
||||
headers = deep_update_dict(
|
||||
@@ -240,8 +226,6 @@ class TestRunner(ApiServerUnittest):
|
||||
testcase_file_path = os.path.join(
|
||||
os.getcwd(), 'tests/data/test_bugfix.yml')
|
||||
testcases = loader.load_file(testcase_file_path)
|
||||
config_dict = {}
|
||||
self.test_runner.init_test(config_dict, "testcase")
|
||||
|
||||
test = testcases[2]["test"]
|
||||
self.test_runner.run_test(test)
|
||||
|
||||
@@ -2,8 +2,7 @@ import io
|
||||
import os
|
||||
import shutil
|
||||
|
||||
from httprunner import exceptions, loader, utils
|
||||
from httprunner.compat import OrderedDict
|
||||
from httprunner import exceptions, loader, parser, utils
|
||||
from tests.base import ApiServerUnittest
|
||||
|
||||
|
||||
@@ -206,6 +205,69 @@ class TestUtils(ApiServerUnittest):
|
||||
self.assertIsInstance(ordered_dict, dict)
|
||||
self.assertIn("a", ordered_dict)
|
||||
|
||||
def test_extend_validators(self):
|
||||
def_validators = [
|
||||
{'eq': ['v1', 200]},
|
||||
{"check": "s2", "expect": 16, "comparator": "len_eq"}
|
||||
]
|
||||
current_validators = [
|
||||
{"check": "v1", "expect": 201},
|
||||
{'len_eq': ['s3', 12]}
|
||||
]
|
||||
def_validators = [
|
||||
parser.parse_validator(validator)
|
||||
for validator in def_validators
|
||||
]
|
||||
ref_validators = [
|
||||
parser.parse_validator(validator)
|
||||
for validator in current_validators
|
||||
]
|
||||
|
||||
extended_validators = utils.extend_validators(def_validators, ref_validators)
|
||||
self.assertIn(
|
||||
{"check": "v1", "expect": 201, "comparator": "eq"},
|
||||
extended_validators
|
||||
)
|
||||
self.assertIn(
|
||||
{"check": "s2", "expect": 16, "comparator": "len_eq"},
|
||||
extended_validators
|
||||
)
|
||||
self.assertIn(
|
||||
{"check": "s3", "expect": 12, "comparator": "len_eq"},
|
||||
extended_validators
|
||||
)
|
||||
|
||||
def test_extend_validators_with_dict(self):
|
||||
def_validators = [
|
||||
{'eq': ["a", {"v": 1}]},
|
||||
{'eq': [{"b": 1}, 200]}
|
||||
]
|
||||
current_validators = [
|
||||
{'len_eq': ['s3', 12]},
|
||||
{'eq': [{"b": 1}, 201]}
|
||||
]
|
||||
def_validators = [
|
||||
parser.parse_validator(validator)
|
||||
for validator in def_validators
|
||||
]
|
||||
ref_validators = [
|
||||
parser.parse_validator(validator)
|
||||
for validator in current_validators
|
||||
]
|
||||
|
||||
extended_validators = utils.extend_validators(def_validators, ref_validators)
|
||||
self.assertEqual(len(extended_validators), 3)
|
||||
self.assertIn({'check': {'b': 1}, 'expect': 201, 'comparator': 'eq'}, extended_validators)
|
||||
self.assertNotIn({'check': {'b': 1}, 'expect': 200, 'comparator': 'eq'}, extended_validators)
|
||||
|
||||
def test_extend_variables(self):
|
||||
raw_variables = [{"var1": "val1"}, {"var2": "val2"}]
|
||||
override_variables = [{"var1": "val111"}, {"var3": "val3"}]
|
||||
extended_variables_mapping = utils.extend_variables(raw_variables, override_variables)
|
||||
self.assertEqual(extended_variables_mapping["var1"], "val111")
|
||||
self.assertEqual(extended_variables_mapping["var2"], "val2")
|
||||
self.assertEqual(extended_variables_mapping["var3"], "val3")
|
||||
|
||||
def test_deepcopy_dict(self):
|
||||
data = {
|
||||
'a': 1,
|
||||
|
||||
@@ -12,12 +12,34 @@ class TestValidator(unittest.TestCase):
|
||||
self.assertFalse(validator.is_testcases(data_structure))
|
||||
|
||||
data_structure = {
|
||||
"name": "desc1",
|
||||
"config": {},
|
||||
"api": {},
|
||||
"testcases": ["testcase11", "testcase12"]
|
||||
"project_mapping": {
|
||||
"PWD": "XXXXX",
|
||||
"functions": {},
|
||||
"env": {}
|
||||
},
|
||||
"testcases": [
|
||||
{ # testcase data structure
|
||||
"config": {
|
||||
"name": "desc1",
|
||||
"path": "testcase1_path",
|
||||
"variables": [], # optional
|
||||
},
|
||||
"teststeps": [
|
||||
# teststep data structure
|
||||
{
|
||||
'name': 'test step desc1',
|
||||
'variables': [], # optional
|
||||
'extract': [], # optional
|
||||
'validate': [],
|
||||
'request': {}
|
||||
},
|
||||
# teststep2 # another teststep dict
|
||||
]
|
||||
},
|
||||
# testcase_dict_2 # another testcase dict
|
||||
]
|
||||
}
|
||||
self.assertTrue(data_structure)
|
||||
self.assertTrue(validator.is_testcases(data_structure))
|
||||
data_structure = [
|
||||
{
|
||||
"name": "desc1",
|
||||
|
||||
@@ -1,35 +1,47 @@
|
||||
|
||||
- config:
|
||||
name: "create user and check result."
|
||||
def: create_and_check($uid, $token)
|
||||
request:
|
||||
"base_url": "http://127.0.0.1:5000"
|
||||
"headers":
|
||||
"Content-Type": "application/json"
|
||||
"device_sn": "$device_sn"
|
||||
id: create_and_check
|
||||
variables:
|
||||
- uid: 9001
|
||||
- token: XXX
|
||||
- device_sn: "TESTCASE_CREATE_XXX"
|
||||
base_url: "http://127.0.0.1:5000"
|
||||
|
||||
- test:
|
||||
name: setup and reset all (override).
|
||||
testcase: testcases/setup.yml
|
||||
output:
|
||||
- token
|
||||
|
||||
- test:
|
||||
name: make sure user $uid does not exist
|
||||
api: get_user($uid, $token)
|
||||
api: get_user
|
||||
variables:
|
||||
- uid: $uid
|
||||
- token: $token
|
||||
validate:
|
||||
- eq: ["status_code", 404]
|
||||
- eq: ["content.success", false]
|
||||
|
||||
- test:
|
||||
name: create user $uid
|
||||
api: create_user
|
||||
variables:
|
||||
- user_name: "user1"
|
||||
- user_password: "123456"
|
||||
api: create_user($uid, $user_name, $user_password, $token)
|
||||
- uid: $uid
|
||||
- token: $token
|
||||
validate:
|
||||
- eq: ["status_code", 201]
|
||||
- eq: ["content.success", true]
|
||||
|
||||
- test:
|
||||
name: check if user $uid exists
|
||||
api: get_user($uid, $token)
|
||||
api: get_user
|
||||
variables:
|
||||
- uid: $uid
|
||||
- token: $token
|
||||
validate:
|
||||
- eq: ["status_code", 200]
|
||||
- eq: ["content.success", true]
|
||||
@@ -1,22 +1,19 @@
|
||||
- config:
|
||||
name: "setup and reset all."
|
||||
def: setup_and_reset($device_sn)
|
||||
id: setup_and_reset
|
||||
variables:
|
||||
- user_agent: 'iOS/10.3'
|
||||
- device_sn: ${gen_random_string(15)}
|
||||
- device_sn: "TESTCASE_SETUP_XXX"
|
||||
- os_platform: 'ios'
|
||||
- app_version: '2.8.6'
|
||||
request:
|
||||
"base_url": "http://127.0.0.1:5000"
|
||||
"headers":
|
||||
"Content-Type": "application/json"
|
||||
"device_sn": "$device_sn"
|
||||
base_url: "http://127.0.0.1:5000"
|
||||
verify: False
|
||||
output:
|
||||
- token
|
||||
|
||||
- test:
|
||||
name: get token
|
||||
api: get_token($user_agent, $device_sn, $os_platform, $app_version)
|
||||
name: get token (setup)
|
||||
api: get_token
|
||||
variables:
|
||||
- user_agent: 'iOS/10.3'
|
||||
- device_sn: $device_sn
|
||||
@@ -30,4 +27,6 @@
|
||||
|
||||
- test:
|
||||
name: reset all users
|
||||
api: reset_all($token)
|
||||
api: reset_all
|
||||
variables:
|
||||
- token: $token
|
||||
@@ -1,24 +0,0 @@
|
||||
- config:
|
||||
name: smoketest
|
||||
variables:
|
||||
- device_sn: ${gen_random_string(15)}
|
||||
request:
|
||||
"base_url": "http://127.0.0.1:5000"
|
||||
"headers":
|
||||
"Content-Type": "application/json"
|
||||
"device_sn": "$device_sn"
|
||||
|
||||
- test:
|
||||
name: setup and reset all.
|
||||
suite: setup_and_reset($device_sn)
|
||||
output:
|
||||
- token
|
||||
- device_sn
|
||||
|
||||
- test:
|
||||
name: create user 1000 and check result.
|
||||
suite: create_and_check(1000, $token)
|
||||
|
||||
- test:
|
||||
name: create user 1001 and check result.
|
||||
suite: create_and_check(1001, $token)
|
||||
17
tests/testsuites/create_users.yml
Normal file
17
tests/testsuites/create_users.yml
Normal file
@@ -0,0 +1,17 @@
|
||||
- config:
|
||||
name: create users with uid
|
||||
variables:
|
||||
- device_sn: ${gen_random_string(15)}
|
||||
base_url: "http://127.0.0.1:5000"
|
||||
|
||||
- test:
|
||||
name: create user 1000 and check result.
|
||||
testcase: testcases/create_and_check.yml
|
||||
variables:
|
||||
- uid: 1000
|
||||
|
||||
- test:
|
||||
name: create user 1001 and check result.
|
||||
testcase: testcases/create_and_check.yml
|
||||
variables:
|
||||
- uid: 1001
|
||||
Reference in New Issue
Block a user