|
1 | 1 | # mypy: allow-untyped-defs |
2 | 2 | from __future__ import annotations |
3 | 3 |
|
| 4 | +from collections.abc import Sequence |
4 | 5 | import os |
5 | 6 | from pathlib import Path |
6 | 7 | from pathlib import PurePath |
|
20 | 21 | from _pytest.pathlib import symlink_or_skip |
21 | 22 | from _pytest.pytester import HookRecorder |
22 | 23 | from _pytest.pytester import Pytester |
| 24 | +from _pytest.pytester import RunResult |
23 | 25 | import pytest |
24 | 26 |
|
25 | 27 |
|
@@ -2702,3 +2704,123 @@ def test_1(): pass |
2702 | 2704 | ], |
2703 | 2705 | consecutive=True, |
2704 | 2706 | ) |
| 2707 | + |
| 2708 | + |
| 2709 | +class TestRequireUniqueParametrizationtIds: |
| 2710 | + CASES = [ |
| 2711 | + ([(1, 1), (1, 1)], {"1-1": [0, 1]}), |
| 2712 | + ([(1, 1), (1, 2), (1, 1)], {"1-1": [0, 2]}), |
| 2713 | + ([(1, 1), (2, 2), (1, 1)], {"1-1": [0, 2]}), |
| 2714 | + ([(1, 1), (2, 2), (1, 2), (2, 1), (1, 1)], {"1-1": [0, 4]}), |
| 2715 | + ] |
| 2716 | + |
| 2717 | + @pytest.mark.parametrize(("parametrize_args", "expected_indices"), CASES) |
| 2718 | + def test_cli_enables( |
| 2719 | + self, |
| 2720 | + pytester: Pytester, |
| 2721 | + parametrize_args: Sequence[tuple[int, int]], |
| 2722 | + expected_indices: dict[str, Sequence[int]], |
| 2723 | + ) -> None: |
| 2724 | + pytester.makepyfile( |
| 2725 | + f""" |
| 2726 | + import pytest |
| 2727 | +
|
| 2728 | + @pytest.mark.parametrize('x, y', {parametrize_args}) |
| 2729 | + def test1(y, x): |
| 2730 | + pass |
| 2731 | + """ |
| 2732 | + ) |
| 2733 | + result = pytester.runpytest("--require-unique-parametrization-ids") |
| 2734 | + resolved_ids = self._convert_to_resolved_ids(parametrize_args) |
| 2735 | + argument_values = self._convert_to_argument_values(parametrize_args) |
| 2736 | + self._assert_duplicate_msg( |
| 2737 | + result, expected_indices, argument_values, resolved_ids |
| 2738 | + ) |
| 2739 | + |
| 2740 | + @pytest.mark.parametrize("parametrize_args, expected_indices", CASES) |
| 2741 | + def test_ini_enables( |
| 2742 | + self, |
| 2743 | + pytester: Pytester, |
| 2744 | + parametrize_args: Sequence[tuple[int, int]], |
| 2745 | + expected_indices: dict[str, Sequence[int]], |
| 2746 | + ) -> None: |
| 2747 | + pytester.makeini( |
| 2748 | + """ |
| 2749 | + [pytest] |
| 2750 | + require_unique_parametrization_ids = true |
| 2751 | + """ |
| 2752 | + ) |
| 2753 | + pytester.makepyfile( |
| 2754 | + f""" |
| 2755 | + import pytest |
| 2756 | +
|
| 2757 | + @pytest.mark.parametrize('x, y', {parametrize_args}) |
| 2758 | + def test1(y, x): |
| 2759 | + pass |
| 2760 | + """ |
| 2761 | + ) |
| 2762 | + result = pytester.runpytest() |
| 2763 | + resolved_ids = self._convert_to_resolved_ids(parametrize_args) |
| 2764 | + argument_values = self._convert_to_argument_values(parametrize_args) |
| 2765 | + self._assert_duplicate_msg( |
| 2766 | + result, expected_indices, argument_values, resolved_ids |
| 2767 | + ) |
| 2768 | + |
| 2769 | + def test_cli_overrides_ini_false(self, pytester: Pytester) -> None: |
| 2770 | + """CLI True should override ini False.""" |
| 2771 | + pytester.makeini( |
| 2772 | + """ |
| 2773 | + [pytest] |
| 2774 | + require_unique_parametrization_ids = false |
| 2775 | + """ |
| 2776 | + ) |
| 2777 | + parametrization_args = [(1, 1), (1, 1)] |
| 2778 | + pytester.makepyfile( |
| 2779 | + f""" |
| 2780 | + import pytest |
| 2781 | +
|
| 2782 | + @pytest.mark.parametrize('x, y', {parametrization_args}) |
| 2783 | + def test1(y, x): |
| 2784 | + pass |
| 2785 | + """ |
| 2786 | + ) |
| 2787 | + result = pytester.runpytest("--require-unique-parametrization-ids") |
| 2788 | + resolved_ids = self._convert_to_resolved_ids(parametrization_args) |
| 2789 | + argument_values = self._convert_to_argument_values(parametrization_args) |
| 2790 | + self._assert_duplicate_msg( |
| 2791 | + result, {"1-1": [0, 1]}, argument_values, resolved_ids |
| 2792 | + ) |
| 2793 | + |
| 2794 | + def _assert_duplicate_msg( |
| 2795 | + self, |
| 2796 | + result: RunResult, |
| 2797 | + expected_indices: dict[str, Sequence[int]], |
| 2798 | + argument_values: Sequence[str], |
| 2799 | + resolved_ids: Sequence[str], |
| 2800 | + ) -> None: |
| 2801 | + stream = result.stdout |
| 2802 | + stream.fnmatch_lines( |
| 2803 | + [ |
| 2804 | + "Duplicate parametrization IDs detected, but --require-unique-parametrization-ids is set.", |
| 2805 | + "Test name: *::test1", |
| 2806 | + "Argument names: [[]'x', 'y'[]]", |
| 2807 | + f"Argument values: {argument_values}", |
| 2808 | + f"Resolved IDs: {resolved_ids}", |
| 2809 | + f"Duplicates: {expected_indices}", |
| 2810 | + "You can fix this problem using:", |
| 2811 | + "the `ids` argument of `@pytest.mark.parametrize()` or the `id` argument of `pytest.param()`.", |
| 2812 | + ] |
| 2813 | + ) |
| 2814 | + assert result.ret != 0 |
| 2815 | + |
| 2816 | + def _convert_to_resolved_ids( |
| 2817 | + self, parametrize_args: Sequence[tuple[int, int]] |
| 2818 | + ) -> Sequence[str]: |
| 2819 | + return [ |
| 2820 | + f"{argument1}-{argument2}" for (argument1, argument2) in parametrize_args |
| 2821 | + ] |
| 2822 | + |
| 2823 | + def _convert_to_argument_values( |
| 2824 | + self, parametrize_args: Sequence[tuple[int, int]] |
| 2825 | + ) -> Sequence[str]: |
| 2826 | + return [str(argument) for argument in parametrize_args] |
0 commit comments