2024-01-30 16:53:12 +01:00
|
|
|
# SPDX-FileCopyrightText: 2022-2024 Espressif Systems (Shanghai) CO LTD
|
2022-08-09 02:11:57 +02:00
|
|
|
# SPDX-License-Identifier: Apache-2.0
|
|
|
|
import datetime
|
|
|
|
import logging
|
|
|
|
import os
|
|
|
|
import shutil
|
|
|
|
import subprocess
|
|
|
|
import typing
|
|
|
|
from pathlib import Path
|
|
|
|
from tempfile import mkdtemp
|
|
|
|
|
|
|
|
import pytest
|
|
|
|
from _pytest.fixtures import FixtureRequest
|
2024-01-30 16:53:12 +01:00
|
|
|
from test_build_system_helpers import EnvDict
|
|
|
|
from test_build_system_helpers import EXT_IDF_PATH
|
|
|
|
from test_build_system_helpers import get_idf_build_env
|
|
|
|
from test_build_system_helpers import IdfPyFunc
|
|
|
|
from test_build_system_helpers import run_idf_py
|
2022-08-09 02:11:57 +02:00
|
|
|
|
|
|
|
|
|
|
|
# Pytest hook used to check if the test has passed or failed, from a fixture.
|
|
|
|
# Based on https://docs.pytest.org/en/latest/example/simple.html#making-test-result-information-available-in-fixtures
|
|
|
|
@pytest.hookimpl(tryfirst=True, hookwrapper=True)
|
|
|
|
def pytest_runtest_makereport(item: typing.Any, call: typing.Any) -> typing.Generator[None, pytest.TestReport, None]: # pylint: disable=unused-argument
|
|
|
|
outcome = yield # Execute all other hooks to obtain the report object
|
|
|
|
report = outcome.get_result()
|
|
|
|
if report.when == 'call' and report.passed:
|
|
|
|
# set an attribute which can be checked using 'should_clean_test_dir' function below
|
|
|
|
setattr(item, 'passed', True)
|
|
|
|
|
|
|
|
|
|
|
|
def should_clean_test_dir(request: FixtureRequest) -> bool:
|
|
|
|
# Only remove the test directory if the test has passed
|
2024-04-16 14:11:31 +02:00
|
|
|
return getattr(request.node, 'passed', False) or request.config.getoption('cleanup_idf_copy', False)
|
2022-08-09 02:11:57 +02:00
|
|
|
|
|
|
|
|
|
|
|
def pytest_addoption(parser: pytest.Parser) -> None:
|
|
|
|
parser.addoption(
|
|
|
|
'--work-dir', action='store', default=None,
|
|
|
|
help='Directory for temporary files. If not specified, an OS-specific '
|
|
|
|
'temporary directory will be used.'
|
|
|
|
)
|
2024-04-16 14:11:31 +02:00
|
|
|
parser.addoption(
|
|
|
|
'--cleanup-idf-copy', action='store_true',
|
|
|
|
help='Always clean up the IDF copy after the test. By default, the copy is cleaned up only if the test passes.'
|
|
|
|
)
|
2022-08-09 02:11:57 +02:00
|
|
|
|
|
|
|
|
2023-10-25 22:05:31 +02:00
|
|
|
@pytest.fixture(scope='session')
|
|
|
|
def _session_work_dir(request: FixtureRequest) -> typing.Generator[typing.Tuple[Path, bool], None, None]:
|
2022-08-09 02:11:57 +02:00
|
|
|
work_dir = request.config.getoption('--work-dir')
|
2023-10-25 22:05:31 +02:00
|
|
|
|
2022-08-09 02:11:57 +02:00
|
|
|
if work_dir:
|
|
|
|
work_dir = os.path.join(work_dir, datetime.datetime.utcnow().strftime('%Y-%m-%d_%H-%M-%S'))
|
|
|
|
logging.debug(f'using work directory: {work_dir}')
|
|
|
|
os.makedirs(work_dir, exist_ok=True)
|
|
|
|
clean_dir = None
|
2023-10-25 22:05:31 +02:00
|
|
|
is_temp_dir = False
|
2022-08-09 02:11:57 +02:00
|
|
|
else:
|
|
|
|
work_dir = mkdtemp()
|
|
|
|
logging.debug(f'created temporary work directory: {work_dir}')
|
|
|
|
clean_dir = work_dir
|
2023-10-25 22:05:31 +02:00
|
|
|
is_temp_dir = True
|
|
|
|
|
|
|
|
# resolve allows using relative paths with --work-dir option
|
|
|
|
yield Path(work_dir).resolve(), is_temp_dir
|
|
|
|
|
|
|
|
if clean_dir:
|
|
|
|
logging.debug(f'cleaning up {clean_dir}')
|
|
|
|
shutil.rmtree(clean_dir, ignore_errors=True)
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture(name='func_work_dir', autouse=True)
|
|
|
|
def work_dir(request: FixtureRequest, _session_work_dir: typing.Tuple[Path, bool]) -> typing.Generator[Path, None, None]:
|
|
|
|
session_work_dir, is_temp_dir = _session_work_dir
|
|
|
|
|
|
|
|
if request._pyfuncitem.keywords.get('force_temp_work_dir') and not is_temp_dir:
|
|
|
|
work_dir = Path(mkdtemp()).resolve()
|
|
|
|
logging.debug('Force using temporary work directory')
|
|
|
|
clean_dir = work_dir
|
|
|
|
else:
|
|
|
|
work_dir = session_work_dir
|
|
|
|
clean_dir = None
|
2022-08-09 02:11:57 +02:00
|
|
|
|
2023-10-25 22:05:31 +02:00
|
|
|
# resolve allows using relative paths with --work-dir option
|
|
|
|
yield work_dir
|
2022-08-09 02:11:57 +02:00
|
|
|
|
|
|
|
if clean_dir:
|
|
|
|
logging.debug(f'cleaning up {clean_dir}')
|
|
|
|
shutil.rmtree(clean_dir, ignore_errors=True)
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
2023-10-25 22:05:31 +02:00
|
|
|
def test_app_copy(func_work_dir: Path, request: FixtureRequest) -> typing.Generator[Path, None, None]:
|
2022-08-09 02:11:57 +02:00
|
|
|
# by default, use hello_world app and copy it to a temporary directory with
|
|
|
|
# the name resembling that of the test
|
|
|
|
copy_from = 'tools/test_build_system/build_test_app'
|
2024-08-09 13:26:42 +02:00
|
|
|
# sanitize test name in case pytest.mark.parametrize was used
|
|
|
|
test_name_sanitized = request.node.name.replace('[', '_').replace(']', '')
|
|
|
|
copy_to = test_name_sanitized + '_app'
|
2022-08-09 02:11:57 +02:00
|
|
|
|
|
|
|
# allow overriding source and destination via pytest.mark.test_app_copy()
|
|
|
|
mark = request.node.get_closest_marker('test_app_copy')
|
|
|
|
if mark:
|
|
|
|
copy_from = mark.args[0]
|
|
|
|
if len(mark.args) > 1:
|
|
|
|
copy_to = mark.args[1]
|
|
|
|
|
|
|
|
path_from = Path(os.environ['IDF_PATH']) / copy_from
|
2023-10-25 22:05:31 +02:00
|
|
|
path_to = func_work_dir / copy_to
|
2022-08-09 02:11:57 +02:00
|
|
|
|
|
|
|
# if the new directory inside the original directory,
|
|
|
|
# make sure not to go into recursion.
|
|
|
|
ignore = shutil.ignore_patterns(
|
|
|
|
path_to.name,
|
|
|
|
# also ignore files which may be present in the work directory
|
|
|
|
'build', 'sdkconfig')
|
|
|
|
|
|
|
|
logging.debug(f'copying {path_from} to {path_to}')
|
|
|
|
shutil.copytree(path_from, path_to, ignore=ignore, symlinks=True)
|
|
|
|
|
|
|
|
old_cwd = Path.cwd()
|
|
|
|
os.chdir(path_to)
|
|
|
|
|
|
|
|
yield Path(path_to)
|
|
|
|
|
|
|
|
os.chdir(old_cwd)
|
|
|
|
|
|
|
|
if should_clean_test_dir(request):
|
|
|
|
logging.debug('cleaning up work directory after a successful test: {}'.format(path_to))
|
|
|
|
shutil.rmtree(path_to, ignore_errors=True)
|
|
|
|
|
|
|
|
|
2023-05-19 18:06:31 +02:00
|
|
|
@pytest.fixture
|
2023-10-25 22:05:31 +02:00
|
|
|
def test_git_template_app(func_work_dir: Path, request: FixtureRequest) -> typing.Generator[Path, None, None]:
|
2024-08-16 15:31:30 +02:00
|
|
|
# sanitize test name in case pytest.mark.parametrize was used
|
|
|
|
test_name_sanitized = request.node.name.replace('[', '_').replace(']', '')
|
|
|
|
copy_to = test_name_sanitized + '_app'
|
2023-10-25 22:05:31 +02:00
|
|
|
path_to = func_work_dir / copy_to
|
2023-05-19 18:06:31 +02:00
|
|
|
|
2023-10-25 22:05:31 +02:00
|
|
|
logging.debug(f'cloning git-template app to {path_to}')
|
2023-05-19 18:06:31 +02:00
|
|
|
path_to.mkdir()
|
2023-10-25 22:05:31 +02:00
|
|
|
# No need to clone full repository, just a single master branch
|
2023-05-19 18:06:31 +02:00
|
|
|
subprocess.run(['git', 'clone', '--single-branch', '-b', 'master', '--depth', '1', 'https://github.com/espressif/esp-idf-template.git', '.'],
|
|
|
|
cwd=path_to, stdout=subprocess.PIPE, stderr=subprocess.PIPE)
|
|
|
|
|
|
|
|
old_cwd = Path.cwd()
|
|
|
|
os.chdir(path_to)
|
|
|
|
|
|
|
|
yield Path(path_to)
|
|
|
|
|
|
|
|
os.chdir(old_cwd)
|
|
|
|
|
|
|
|
if should_clean_test_dir(request):
|
|
|
|
logging.debug('cleaning up work directory after a successful test: {}'.format(path_to))
|
|
|
|
shutil.rmtree(path_to, ignore_errors=True)
|
|
|
|
|
|
|
|
|
2022-08-09 02:11:57 +02:00
|
|
|
@pytest.fixture
|
2023-10-25 22:05:31 +02:00
|
|
|
def idf_copy(func_work_dir: Path, request: FixtureRequest) -> typing.Generator[Path, None, None]:
|
2024-08-16 15:31:30 +02:00
|
|
|
# sanitize test name in case pytest.mark.parametrize was used
|
|
|
|
test_name_sanitized = request.node.name.replace('[', '_').replace(']', '')
|
|
|
|
copy_to = test_name_sanitized + '_idf'
|
2024-08-15 11:53:48 +02:00
|
|
|
# allow overriding the destination via pytest.mark.idf_copy_with_space so the destination contain space
|
|
|
|
mark_with_space = request.node.get_closest_marker('idf_copy_with_space')
|
|
|
|
if mark_with_space:
|
2024-08-16 15:31:30 +02:00
|
|
|
copy_to = test_name_sanitized + ' idf'
|
2022-08-09 02:11:57 +02:00
|
|
|
|
|
|
|
# allow overriding the destination via pytest.mark.idf_copy()
|
|
|
|
mark = request.node.get_closest_marker('idf_copy')
|
|
|
|
if mark:
|
|
|
|
copy_to = mark.args[0]
|
|
|
|
|
|
|
|
path_from = EXT_IDF_PATH
|
2023-10-25 22:05:31 +02:00
|
|
|
path_to = func_work_dir / copy_to
|
2022-08-09 02:11:57 +02:00
|
|
|
|
|
|
|
# if the new directory inside the original directory,
|
|
|
|
# make sure not to go into recursion.
|
|
|
|
ignore = shutil.ignore_patterns(
|
|
|
|
path_to.name,
|
|
|
|
# also ignore the build directories which may be quite large
|
2024-01-30 16:53:12 +01:00
|
|
|
# plus ignore .git since it is causing trouble when removing on Windows
|
|
|
|
'**/build', '.git')
|
2022-08-09 02:11:57 +02:00
|
|
|
|
|
|
|
logging.debug(f'copying {path_from} to {path_to}')
|
|
|
|
shutil.copytree(path_from, path_to, ignore=ignore, symlinks=True)
|
|
|
|
|
|
|
|
orig_idf_path = os.environ['IDF_PATH']
|
2022-11-01 19:57:03 +01:00
|
|
|
os.environ['IDF_PATH'] = str(path_to)
|
2022-08-09 02:11:57 +02:00
|
|
|
|
|
|
|
yield Path(path_to)
|
|
|
|
|
|
|
|
os.environ['IDF_PATH'] = orig_idf_path
|
|
|
|
|
|
|
|
if should_clean_test_dir(request):
|
|
|
|
logging.debug('cleaning up work directory after a successful test: {}'.format(path_to))
|
|
|
|
shutil.rmtree(path_to, ignore_errors=True)
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture(name='default_idf_env')
|
|
|
|
def fixture_default_idf_env() -> EnvDict:
|
|
|
|
return get_idf_build_env(os.environ['IDF_PATH']) # type: ignore
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.fixture
|
|
|
|
def idf_py(default_idf_env: EnvDict) -> IdfPyFunc:
|
2023-03-29 17:22:15 +02:00
|
|
|
def result(*args: str, check: bool = True, input_str: typing.Optional[str] = None) -> subprocess.CompletedProcess:
|
|
|
|
return run_idf_py(*args, env=default_idf_env, workdir=os.getcwd(), check=check, input_str=input_str) # type: ignore
|
2022-08-09 02:11:57 +02:00
|
|
|
return result
|