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

pantsbuild / pants / 19015773527

02 Nov 2025 05:33PM UTC coverage: 17.872% (-62.4%) from 80.3%
19015773527

Pull #22816

github

web-flow
Merge a12d75757 into 6c024e162
Pull Request #22816: Update Pants internal Python to 3.14

4 of 5 new or added lines in 3 files covered. (80.0%)

28452 existing lines in 683 files now uncovered.

9831 of 55007 relevant lines covered (17.87%)

0.18 hits per line

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

46.67
/src/python/pants/core/util_rules/stripped_source_files.py
1
# Copyright 2019 Pants project contributors (see CONTRIBUTORS.md).
2
# Licensed under the Apache License, Version 2.0 (see LICENSE).
3

4
from collections import defaultdict
1✔
5
from dataclasses import dataclass
1✔
6

7
from pants.core.util_rules.source_files import SourceFiles
1✔
8
from pants.core.util_rules.source_files import rules as source_files_rules
1✔
9
from pants.engine.collection import Collection
1✔
10
from pants.engine.engine_aware import EngineAwareParameter
1✔
11
from pants.engine.fs import DigestSubset, MergeDigests, PathGlobs, RemovePrefix, Snapshot
1✔
12
from pants.engine.intrinsics import digest_subset_to_digest, digest_to_snapshot, remove_prefix
1✔
13
from pants.engine.rules import collect_rules, concurrently, implicitly, rule
1✔
14
from pants.engine.target import SourcesPaths
1✔
15
from pants.source.source_root import (
1✔
16
    SourceRootRequest,
17
    SourceRootsRequest,
18
    get_source_root,
19
    get_source_roots,
20
)
21
from pants.source.source_root import rules as source_root_rules
1✔
22
from pants.util.dirutil import fast_relpath
1✔
23

24

25
@dataclass(frozen=True)
1✔
26
class StrippedSourceFiles:
1✔
27
    """Wrapper for a snapshot of files whose source roots have been stripped."""
28

29
    snapshot: Snapshot
30

31

32
@rule
1✔
33
async def strip_source_roots(source_files: SourceFiles) -> StrippedSourceFiles:
1✔
34
    """Removes source roots from a snapshot.
35

36
    E.g. `src/python/pants/util/strutil.py` -> `pants/util/strutil.py`.
37
    """
38
    if not source_files.snapshot.files:
×
39
        return StrippedSourceFiles(source_files.snapshot)
×
40

41
    if source_files.unrooted_files:
×
42
        rooted_files = set(source_files.snapshot.files) - set(source_files.unrooted_files)
×
43
        rooted_files_snapshot = await digest_to_snapshot(
×
44
            **implicitly(DigestSubset(source_files.snapshot.digest, PathGlobs(rooted_files)))
45
        )
46
    else:
47
        rooted_files_snapshot = source_files.snapshot
×
48

49
    source_roots_result = await get_source_roots(
×
50
        SourceRootsRequest.for_files(rooted_files_snapshot.files)
51
    )
52

53
    source_roots_to_files = defaultdict(set)
×
54
    for f, root in source_roots_result.path_to_root.items():
×
55
        source_roots_to_files[root.path].add(str(f))
×
56

57
    if len(source_roots_to_files) == 1:
×
58
        source_root = next(iter(source_roots_to_files.keys()))
×
59
        if source_root == ".":
×
60
            resulting_snapshot = rooted_files_snapshot
×
61
        else:
62
            resulting_snapshot = await digest_to_snapshot(
×
63
                **implicitly(RemovePrefix(rooted_files_snapshot.digest, source_root))
64
            )
65
    else:
66
        digest_subsets = await concurrently(
×
67
            digest_subset_to_digest(DigestSubset(rooted_files_snapshot.digest, PathGlobs(files)))
68
            for files in source_roots_to_files.values()
69
        )
70
        resulting_digests = await concurrently(
×
71
            remove_prefix(RemovePrefix(digest, source_root))
72
            for digest, source_root in zip(digest_subsets, source_roots_to_files.keys())
73
        )
74
        resulting_snapshot = await digest_to_snapshot(**implicitly(MergeDigests(resulting_digests)))
×
75

76
    # Add the unrooted files back in.
77
    if source_files.unrooted_files:
×
78
        unrooted_files_digest = await digest_subset_to_digest(
×
79
            DigestSubset(source_files.snapshot.digest, PathGlobs(source_files.unrooted_files))
80
        )
81
        resulting_snapshot = await digest_to_snapshot(
×
82
            **implicitly(MergeDigests((resulting_snapshot.digest, unrooted_files_digest)))
83
        )
84

85
    return StrippedSourceFiles(resulting_snapshot)
×
86

87

88
@dataclass(frozen=True)
1✔
89
class StrippedFileName:
1✔
90
    value: str
91

92

93
@dataclass(frozen=True)
1✔
94
class StrippedFileNameRequest(EngineAwareParameter):
1✔
95
    file_path: str
96

97
    def debug_hint(self) -> str:
1✔
98
        return self.file_path
×
99

100

101
@rule
1✔
102
async def strip_file_name(request: StrippedFileNameRequest) -> StrippedFileName:
1✔
103
    source_root = await get_source_root(SourceRootRequest.for_file(request.file_path))
×
104
    return StrippedFileName(
×
105
        request.file_path
106
        if source_root.path == "."
107
        else fast_relpath(request.file_path, source_root.path)
108
    )
109

110

111
class StrippedSourceFileNames(Collection[str]):
1✔
112
    """The file names from a target's `sources` field, with source roots stripped."""
113

114

115
@rule
1✔
116
async def strip_sources_paths(sources_paths: SourcesPaths) -> StrippedSourceFileNames:
1✔
117
    if not sources_paths.files:
×
118
        return StrippedSourceFileNames()
×
119
    source_root = await get_source_root(SourceRootRequest.for_file(sources_paths.files[0]))
×
120
    if source_root.path == ".":
×
121
        return StrippedSourceFileNames(sources_paths.files)
×
122
    return StrippedSourceFileNames(fast_relpath(f, source_root.path) for f in sources_paths.files)
×
123

124

125
def rules():
1✔
UNCOV
126
    return (*collect_rules(), *source_root_rules(), *source_files_rules())
×
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