ci(pytest): fix multi dut wrongly be picked issue

This issue will happen to multi-dut test cases
- without `target` defined in param
- and with `app_path` defined in param
- and with `pytest.mark.target` markers
This commit is contained in:
Fu Hanxi 2024-01-30 11:18:41 +01:00
parent 03d6b092c0
commit 85372fb1ce
No known key found for this signature in database
GPG Key ID: 19399699CF3C4B16
3 changed files with 72 additions and 5 deletions

View File

@ -5,6 +5,7 @@ Pytest Related Constants. Don't import third-party packages here.
""" """
import os import os
import typing as t import typing as t
import warnings
from dataclasses import dataclass from dataclasses import dataclass
from enum import Enum from enum import Enum
from functools import cached_property from functools import cached_property
@ -174,6 +175,7 @@ class PytestCase:
apps: t.List[PytestApp] apps: t.List[PytestApp]
item: Function item: Function
multi_dut_without_param: bool
def __hash__(self) -> int: def __hash__(self) -> int:
return hash((self.path, self.name, self.apps, self.all_markers)) return hash((self.path, self.name, self.apps, self.all_markers))
@ -188,7 +190,22 @@ class PytestCase:
@cached_property @cached_property
def targets(self) -> t.List[str]: def targets(self) -> t.List[str]:
return [app.target for app in self.apps] if not self.multi_dut_without_param:
return [app.target for app in self.apps]
# multi-dut test cases without parametrize
skip = True
for _t in [app.target for app in self.apps]:
if _t in self.target_markers:
skip = False
warnings.warn(f'`pytest.mark.[TARGET]` defined in parametrize for multi-dut test cases is deprecated. '
f'Please use parametrize instead for test case {self.item.nodeid}')
break
if not skip:
return [app.target for app in self.apps]
return [''] * len(self.apps) # this will help to filter these cases out later
@cached_property @cached_property
def is_single_dut_test_case(self) -> bool: def is_single_dut_test_case(self) -> bool:

View File

@ -104,7 +104,7 @@ class IdfPytestEmbedded:
return item.callspec.params.get(key, default) or default return item.callspec.params.get(key, default) or default
def item_to_pytest_case(self, item: Function) -> PytestCase: def item_to_pytest_case(self, item: Function) -> t.Optional[PytestCase]:
""" """
Turn pytest item to PytestCase Turn pytest item to PytestCase
""" """
@ -113,10 +113,23 @@ class IdfPytestEmbedded:
# default app_path is where the test script locates # default app_path is where the test script locates
app_paths = to_list(parse_multi_dut_args(count, self.get_param(item, 'app_path', os.path.dirname(item.path)))) app_paths = to_list(parse_multi_dut_args(count, self.get_param(item, 'app_path', os.path.dirname(item.path))))
configs = to_list(parse_multi_dut_args(count, self.get_param(item, 'config', DEFAULT_SDKCONFIG))) configs = to_list(parse_multi_dut_args(count, self.get_param(item, 'config', DEFAULT_SDKCONFIG)))
targets = to_list(parse_multi_dut_args(count, self.get_param(item, 'target', self.target[0]))) targets = to_list(parse_multi_dut_args(count, self.get_param(item, 'target')))
multi_dut_without_param = False
if count > 1 and targets == [None] * count:
multi_dut_without_param = True
try:
targets = to_list(parse_multi_dut_args(count, '|'.join(self.target))) # check later while collecting
except ValueError: # count doesn't match
return None
elif targets is None:
targets = self.target
return PytestCase( return PytestCase(
[PytestApp(app_paths[i], targets[i], configs[i]) for i in range(count)], item apps=[PytestApp(app_paths[i], targets[i], configs[i]) for i in range(count)],
item=item,
multi_dut_without_param=multi_dut_without_param
) )
@pytest.hookimpl(tryfirst=True) @pytest.hookimpl(tryfirst=True)
@ -167,7 +180,11 @@ class IdfPytestEmbedded:
# 2. Add markers according to special markers # 2. Add markers according to special markers
item_to_case_dict: t.Dict[Function, PytestCase] = {} item_to_case_dict: t.Dict[Function, PytestCase] = {}
for item in items: for item in items:
item.stash[ITEM_PYTEST_CASE_KEY] = item_to_case_dict[item] = self.item_to_pytest_case(item) case = self.item_to_pytest_case(item)
if case is None:
continue
item.stash[ITEM_PYTEST_CASE_KEY] = item_to_case_dict[item] = case
if 'supported_targets' in item.keywords: if 'supported_targets' in item.keywords:
for _target in SUPPORTED_TARGETS: for _target in SUPPORTED_TARGETS:
item.add_marker(_target) item.add_marker(_target)
@ -177,6 +194,7 @@ class IdfPytestEmbedded:
if 'all_targets' in item.keywords: if 'all_targets' in item.keywords:
for _target in [*SUPPORTED_TARGETS, *PREVIEW_TARGETS]: for _target in [*SUPPORTED_TARGETS, *PREVIEW_TARGETS]:
item.add_marker(_target) item.add_marker(_target)
items[:] = [_item for _item in items if _item in item_to_case_dict]
# 3.1. CollectMode.SINGLE_SPECIFIC, like `pytest --target esp32` # 3.1. CollectMode.SINGLE_SPECIFIC, like `pytest --target esp32`
if self.collect_mode == CollectMode.SINGLE_SPECIFIC: if self.collect_mode == CollectMode.SINGLE_SPECIFIC:

View File

@ -1,5 +1,6 @@
# SPDX-FileCopyrightText: 2023-2024 Espressif Systems (Shanghai) CO LTD # SPDX-FileCopyrightText: 2023-2024 Espressif Systems (Shanghai) CO LTD
# SPDX-License-Identifier: Apache-2.0 # SPDX-License-Identifier: Apache-2.0
import textwrap
from pathlib import Path from pathlib import Path
from idf_pytest.constants import CollectMode from idf_pytest.constants import CollectMode
@ -86,3 +87,34 @@ def test_get_pytest_cases_all(work_dirpath: Path) -> None:
assert cases[5].targets == ['esp32s2'] assert cases[5].targets == ['esp32s2']
assert cases[5].name == 'test_foo_single' assert cases[5].name == 'test_foo_single'
def test_multi_with_marker_and_app_path(work_dirpath: Path) -> None:
script = work_dirpath / 'pytest_multi_with_marker_and_app_path.py'
script.write_text(
textwrap.dedent(
'''
import pytest
@pytest.mark.esp32c2
@pytest.mark.parametrize(
'count,app_path', [
(2, 'foo|bar'),
(3, 'foo|bar|baz'),
], indirect=True
)
def test_foo_multi_with_marker_and_app_path(dut):
pass
'''
)
)
cases = get_pytest_cases([str(work_dirpath)], 'esp32c3,esp32c3')
assert len(cases) == 0
cases = get_pytest_cases([str(work_dirpath)], 'esp32c2,esp32c2')
assert len(cases) == 1
assert cases[0].targets == ['esp32c2', 'esp32c2']
cases = get_pytest_cases([str(work_dirpath)], 'esp32c2,esp32c2,esp32c2')
assert len(cases) == 1
assert cases[0].targets == ['esp32c2', 'esp32c2', 'esp32c2']