• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

pantsbuild / pants / 20332790708

18 Dec 2025 09:48AM UTC coverage: 64.992% (-15.3%) from 80.295%
20332790708

Pull #22949

github

web-flow
Merge f730a56cd into 407284c67
Pull Request #22949: Add experimental uv resolver for Python lockfiles

54 of 97 new or added lines in 5 files covered. (55.67%)

8270 existing lines in 295 files now uncovered.

48990 of 75379 relevant lines covered (64.99%)

1.81 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

83.93
/src/python/pants/backend/python/subsystems/pytest.py
1
# Copyright 2016 Pants project contributors (see CONTRIBUTORS.md).
2
# Licensed under the Apache License, Version 2.0 (see LICENSE).
3

4
from __future__ import annotations
4✔
5

6
import os.path
4✔
7
from collections.abc import Iterable
4✔
8
from dataclasses import dataclass
4✔
9

10
from pants.backend.python.subsystems.python_tool_base import PythonToolBase
4✔
11
from pants.backend.python.target_types import (
4✔
12
    ConsoleScript,
13
    InterpreterConstraintsField,
14
    PythonResolveField,
15
    PythonTestsBatchCompatibilityTagField,
16
    PythonTestsExtraEnvVarsField,
17
    PythonTestSourceField,
18
    PythonTestsTimeoutField,
19
    PythonTestsXdistConcurrencyField,
20
    SkipPythonTestsField,
21
)
22
from pants.core.environments.target_types import EnvironmentField
4✔
23
from pants.core.goals.resolves import ExportableTool
4✔
24
from pants.core.goals.test import RuntimePackageDependenciesField, TestFieldSet
4✔
25
from pants.core.util_rules.config_files import ConfigFilesRequest
4✔
26
from pants.engine.rules import collect_rules
4✔
27
from pants.engine.target import Target
4✔
28
from pants.engine.unions import UnionRule
4✔
29
from pants.option.option_types import ArgsListOption, BoolOption, FileOption, SkipOption, StrOption
4✔
30
from pants.util.strutil import softwrap
4✔
31

32

33
@dataclass(frozen=True)
4✔
34
class PythonTestFieldSet(TestFieldSet):
4✔
35
    required_fields = (PythonTestSourceField,)
4✔
36

37
    source: PythonTestSourceField
4✔
38
    interpreter_constraints: InterpreterConstraintsField
4✔
39
    timeout: PythonTestsTimeoutField
4✔
40
    runtime_package_dependencies: RuntimePackageDependenciesField
4✔
41
    extra_env_vars: PythonTestsExtraEnvVarsField
4✔
42
    xdist_concurrency: PythonTestsXdistConcurrencyField
4✔
43
    batch_compatibility_tag: PythonTestsBatchCompatibilityTagField
4✔
44
    resolve: PythonResolveField
4✔
45
    environment: EnvironmentField
4✔
46

47
    @classmethod
4✔
48
    def opt_out(cls, tgt: Target) -> bool:
4✔
UNCOV
49
        return tgt.get(SkipPythonTestsField).value
×
50

51

52
class PyTest(PythonToolBase):
4✔
53
    options_scope = "pytest"
4✔
54
    name = "Pytest"
4✔
55
    help_short = "The pytest Python test framework (https://docs.pytest.org/)."
4✔
56

57
    default_requirements = [
4✔
58
        "pytest>=7,<9,!=7.1.0,!=7.1.1",
59
        "pytest-cov>=5,<7",
60
        "pytest-xdist>=3.6.1,<4",
61
    ]
62

63
    default_main = ConsoleScript("pytest")
4✔
64

65
    default_lockfile_resource = ("pants.backend.python.subsystems", "pytest.lock")
4✔
66

67
    args = ArgsListOption(example="-k test_foo --quiet", passthrough=True)
4✔
68
    junit_family = StrOption(
4✔
69
        default="xunit2",
70
        advanced=True,
71
        help=softwrap(
72
            """
73
            The format of generated junit XML files. See
74
            https://docs.pytest.org/en/latest/reference.html#confval-junit_family.
75
            """
76
        ),
77
    )
78
    execution_slot_var = StrOption(
4✔
79
        default=None,
80
        advanced=True,
81
        help=softwrap(
82
            """
83
            If a non-empty string, the process execution slot id (an integer) will be exposed
84
            to tests under this environment variable name.
85
            """
86
        ),
87
    )
88
    config = FileOption(
4✔
89
        default=None,
90
        advanced=True,
91
        help=lambda cls: softwrap(
92
            f"""
93
            Path to a config file understood by Pytest
94
            (https://docs.pytest.org/en/latest/reference/customize.html#configuration-file-formats).
95
            Setting this option will disable `[{cls.options_scope}].config_discovery`. Use
96
            this option if the config is located in a non-standard location.
97
            """
98
        ),
99
    )
100
    config_discovery = BoolOption(
4✔
101
        default=True,
102
        advanced=True,
103
        help=lambda cls: softwrap(
104
            f"""
105
            If true, Pants will include all relevant Pytest config files (e.g. `pytest.ini`)
106
            during runs. See
107
            https://docs.pytest.org/en/stable/customize.html#finding-the-rootdir for where
108
            config files should be located for Pytest to discover them.
109

110
            Use `[{cls.options_scope}].config` instead if your config is in a
111
            non-standard location.
112
            """
113
        ),
114
    )
115
    xdist_enabled = BoolOption(
4✔
116
        default=False,
117
        advanced=False,
118
        help=softwrap(
119
            """
120
            If true, Pants will use `pytest-xdist` (https://pytest-xdist.readthedocs.io/en/latest/)
121
            to parallelize tests within each `python_test` target.
122

123
            NOTE: Enabling `pytest-xdist` can cause high-level scoped fixtures (for example `session`)
124
            to execute more than once. See the `pytest-xdist` docs for more info:
125
            https://pypi.org/project/pytest-xdist/#making-session-scoped-fixtures-execute-only-once
126
            """
127
        ),
128
    )
129
    allow_empty_test_collection = BoolOption(
4✔
130
        default=False,
131
        help=softwrap(
132
            """
133
            If true, treat pytest exit code 5 ("No tests were collected") as success.
134

135
            Pytest returns exit code 5 when no tests are collected, e.g., when all tests
136
            in a file are skipped via markers or deselected via `-k`. By default, Pants
137
            treats this as a test failure. Enable this option to instead treat it as
138
            "no tests found", which will be skipped in the test summary.
139

140
            See https://docs.pytest.org/en/stable/reference/exit-codes.html
141
            """
142
        ),
143
    )
144

145
    skip = SkipOption("test")
4✔
146

147
    def config_request(self, dirs: Iterable[str]) -> ConfigFilesRequest:
4✔
148
        # Refer to https://docs.pytest.org/en/stable/customize.html#finding-the-rootdir for how
149
        # config files are discovered.
150
        check_existence = []
×
151
        check_content = {}
×
152
        for d in ("", *dirs):
×
153
            check_existence.append(os.path.join(d, "pytest.ini"))
×
154
            check_content[os.path.join(d, "pyproject.toml")] = b"[tool.pytest.ini_options]"
×
155
            check_content[os.path.join(d, "tox.ini")] = b"[pytest]"
×
156
            check_content[os.path.join(d, "setup.cfg")] = b"[tool:pytest]"
×
157

158
        return ConfigFilesRequest(
×
159
            specified=self.config,
160
            specified_option_name=f"[{self.options_scope}].config",
161
            discovery=self.config_discovery,
162
            check_existence=check_existence,
163
            check_content=check_content,
164
        )
165

166

167
def rules():
4✔
168
    return (
2✔
169
        *collect_rules(),
170
        UnionRule(ExportableTool, PyTest),
171
    )
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2025 Coveralls, Inc