kopia lustrzana https://github.com/espressif/esp-idf
232 wiersze
9.1 KiB
Python
232 wiersze
9.1 KiB
Python
# SPDX-FileCopyrightText: 2015-2022 Espressif Systems (Shanghai) CO LTD
|
|
# SPDX-License-Identifier: Apache-2.0
|
|
|
|
""" Interface for test cases. """
|
|
import functools
|
|
import os
|
|
import socket
|
|
import time
|
|
from datetime import datetime
|
|
|
|
import junit_xml
|
|
|
|
from . import DUT, App, Env, Utility
|
|
from .Utility import format_case_id
|
|
|
|
|
|
class TestCaseFailed(AssertionError):
|
|
def __init__(self, *cases):
|
|
"""
|
|
Raise this exception if one or more test cases fail in a 'normal' way (ie the test runs but fails, no unexpected exceptions)
|
|
|
|
This will avoid dumping the Python stack trace, because the assumption is the junit error info and full job log already has
|
|
enough information for a developer to debug.
|
|
|
|
'cases' argument is the names of one or more test cases
|
|
"""
|
|
message = 'Test case{} failed: {}'.format('s' if len(cases) > 1 else '', ', '.join(str(c) for c in cases))
|
|
super(TestCaseFailed, self).__init__(self, message)
|
|
|
|
|
|
class DefaultEnvConfig(object):
|
|
"""
|
|
default test configs. There're 3 places to set configs, priority is (high -> low):
|
|
|
|
1. overwrite set by caller of test method
|
|
2. values set by test_method decorator
|
|
3. default env config get from this class
|
|
"""
|
|
DEFAULT_CONFIG = {
|
|
'app': App.BaseApp,
|
|
'dut': DUT.BaseDUT,
|
|
'env_tag': 'default',
|
|
'env_config_file': None,
|
|
'test_suite_name': None,
|
|
}
|
|
|
|
@classmethod
|
|
def set_default_config(cls, **kwargs):
|
|
"""
|
|
:param kwargs: configs need to be updated
|
|
:return: None
|
|
"""
|
|
cls.DEFAULT_CONFIG.update(kwargs)
|
|
|
|
@classmethod
|
|
def get_default_config(cls):
|
|
"""
|
|
:return: current default config
|
|
"""
|
|
return cls.DEFAULT_CONFIG.copy()
|
|
|
|
|
|
set_default_config = DefaultEnvConfig.set_default_config
|
|
get_default_config = DefaultEnvConfig.get_default_config
|
|
|
|
|
|
MANDATORY_INFO = {
|
|
'execution_time': 1,
|
|
'env_tag': 'default',
|
|
'category': 'function',
|
|
'ignore': False,
|
|
}
|
|
|
|
|
|
class JunitReport(object):
|
|
# wrapper for junit test report
|
|
# TODO: JunitReport methods are not thread safe (although not likely to be used this way).
|
|
|
|
JUNIT_FILE_NAME = 'XUNIT_RESULT.xml'
|
|
JUNIT_DEFAULT_TEST_SUITE = 'test-suite'
|
|
JUNIT_TEST_SUITE = junit_xml.TestSuite(JUNIT_DEFAULT_TEST_SUITE,
|
|
hostname=socket.gethostname(),
|
|
timestamp=datetime.utcnow().isoformat())
|
|
JUNIT_CURRENT_TEST_CASE = None
|
|
_TEST_CASE_CREATED_TS = 0
|
|
|
|
@classmethod
|
|
def output_report(cls, junit_file_path):
|
|
""" Output current test result to file. """
|
|
with open(os.path.join(junit_file_path, cls.JUNIT_FILE_NAME), 'w') as f:
|
|
junit_xml.to_xml_report_file(f, [cls.JUNIT_TEST_SUITE], prettyprint=False)
|
|
|
|
@classmethod
|
|
def get_current_test_case(cls):
|
|
"""
|
|
By default, the test framework will handle junit test report automatically.
|
|
While some test case might want to update some info to test report.
|
|
They can use this method to get current test case created by test framework.
|
|
|
|
:return: current junit test case instance created by ``JunitTestReport.create_test_case``
|
|
"""
|
|
return cls.JUNIT_CURRENT_TEST_CASE
|
|
|
|
@classmethod
|
|
def test_case_finish(cls, test_case):
|
|
"""
|
|
Append the test case to test suite so it can be output to file.
|
|
Execution time will be automatically updated (compared to ``create_test_case``).
|
|
"""
|
|
test_case.elapsed_sec = time.time() - cls._TEST_CASE_CREATED_TS
|
|
cls.JUNIT_TEST_SUITE.test_cases.append(test_case)
|
|
|
|
@classmethod
|
|
def create_test_case(cls, name):
|
|
"""
|
|
Extend ``junit_xml.TestCase`` with:
|
|
|
|
1. save create test case so it can be get by ``get_current_test_case``
|
|
2. log create timestamp, so ``elapsed_sec`` can be auto updated in ``test_case_finish``.
|
|
|
|
:param name: test case name
|
|
:return: instance of ``junit_xml.TestCase``
|
|
"""
|
|
# set stdout to empty string, so we can always append string to stdout.
|
|
# It won't affect output logic. If stdout is empty, it won't be put to report.
|
|
test_case = junit_xml.TestCase(name, stdout='')
|
|
cls.JUNIT_CURRENT_TEST_CASE = test_case
|
|
cls._TEST_CASE_CREATED_TS = time.time()
|
|
return test_case
|
|
|
|
@classmethod
|
|
def update_performance(cls, performance_items):
|
|
"""
|
|
Update performance results to ``stdout`` of current test case.
|
|
|
|
:param performance_items: a list of performance items. each performance item is a key-value pair.
|
|
"""
|
|
assert cls.JUNIT_CURRENT_TEST_CASE
|
|
|
|
for item in performance_items:
|
|
cls.JUNIT_CURRENT_TEST_CASE.stdout += '[Performance][{}]: {}\n'.format(item[0], item[1])
|
|
|
|
|
|
def test_method(**kwargs):
|
|
"""
|
|
decorator for test case function.
|
|
The following keyword arguments are pre-defined.
|
|
Any other keyword arguments will be regarded as filter for the test case,
|
|
able to access them by ``case_info`` attribute of test method.
|
|
|
|
:keyword app: class for test app. see :doc:`App <App>` for details
|
|
:keyword dut: class for current dut. see :doc:`DUT <DUT>` for details
|
|
:keyword env_tag: name for test environment, used to select configs from config file
|
|
:keyword env_config_file: test env config file. usually will not set this keyword when define case
|
|
:keyword test_suite_name: test suite name, used for generating log folder name and adding xunit format test result.
|
|
usually will not set this keyword when define case
|
|
:keyword junit_report_by_case: By default the test fw will handle junit report generation.
|
|
In some cases, one test function might test many test cases.
|
|
If this flag is set, test case can update junit report by its own.
|
|
"""
|
|
def test(test_func):
|
|
|
|
case_info = MANDATORY_INFO.copy()
|
|
case_info['name'] = case_info['ID'] = test_func.__name__
|
|
case_info['junit_report_by_case'] = False
|
|
case_info.update(kwargs)
|
|
|
|
@functools.wraps(test_func)
|
|
def handle_test(extra_data=None, **overwrite):
|
|
"""
|
|
create env, run test and record test results
|
|
|
|
:param extra_data: extra data that runner or main passed to test case
|
|
:param overwrite: args that runner or main want to overwrite
|
|
:return: None
|
|
"""
|
|
# create env instance
|
|
env_config = DefaultEnvConfig.get_default_config()
|
|
for key in kwargs:
|
|
if key in env_config:
|
|
env_config[key] = kwargs[key]
|
|
|
|
env_config.update(overwrite)
|
|
env_inst = Env.Env(**env_config)
|
|
|
|
# prepare for xunit test results
|
|
junit_file_path = env_inst.app_cls.get_log_folder(env_config['test_suite_name'])
|
|
junit_test_case = JunitReport.create_test_case(format_case_id(case_info['ID'],
|
|
target=env_inst.default_dut_cls.TARGET))
|
|
result = False
|
|
unexpected_error = False
|
|
try:
|
|
Utility.console_log('starting running test: ' + test_func.__name__, color='green')
|
|
# execute test function
|
|
test_func(env_inst, extra_data)
|
|
# if finish without exception, test result is True
|
|
result = True
|
|
except TestCaseFailed as e:
|
|
junit_test_case.add_failure_info(str(e))
|
|
except Exception as e:
|
|
Utility.handle_unexpected_exception(junit_test_case, e)
|
|
unexpected_error = True
|
|
finally:
|
|
# do close all DUTs, if result is False then print DUT debug info
|
|
close_errors = env_inst.close(dut_debug=(not result))
|
|
# We have a hook in DUT close, allow DUT to raise error to fail test case.
|
|
# For example, we don't allow DUT exception (reset) during test execution.
|
|
# We don't want to implement in exception detection in test function logic,
|
|
# as we need to add it to every test case.
|
|
# We can implement it in DUT receive thread,
|
|
# and raise exception in DUT close to fail test case if reset detected.
|
|
if close_errors:
|
|
for error in close_errors:
|
|
junit_test_case.add_failure_info('env close error: {}'.format(error))
|
|
result = False
|
|
if not case_info['junit_report_by_case'] or unexpected_error:
|
|
JunitReport.test_case_finish(junit_test_case)
|
|
|
|
# end case and output result
|
|
JunitReport.output_report(junit_file_path)
|
|
|
|
if result:
|
|
Utility.console_log('Test Succeed: ' + test_func.__name__, color='green')
|
|
else:
|
|
Utility.console_log(('Test Fail: ' + test_func.__name__), color='red')
|
|
return result
|
|
|
|
handle_test.case_info = case_info
|
|
handle_test.test_method = True
|
|
return handle_test
|
|
return test
|