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

pantsbuild / pants / 23074067894

13 Mar 2026 11:06PM UTC coverage: 64.165% (-28.8%) from 92.932%
23074067894

Pull #23171

github

web-flow
Merge 17d8ea7d8 into f07276df6
Pull Request #23171: Debug reapi test cache misses

42163 of 65710 relevant lines covered (64.17%)

0.99 hits per line

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

91.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
2✔
5
from dataclasses import dataclass
2✔
6

7
from pants.core.util_rules.source_files import SourceFiles
2✔
8
from pants.core.util_rules.source_files import rules as source_files_rules
2✔
9
from pants.engine.collection import Collection
2✔
10
from pants.engine.engine_aware import EngineAwareParameter
2✔
11
from pants.engine.fs import DigestSubset, MergeDigests, PathGlobs, RemovePrefix, Snapshot
2✔
12
from pants.engine.intrinsics import digest_subset_to_digest, digest_to_snapshot, remove_prefix
2✔
13
from pants.engine.rules import collect_rules, concurrently, implicitly, rule
2✔
14
from pants.engine.target import SourcesPaths
2✔
15
from pants.source.source_root import (
2✔
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
2✔
22
from pants.util.dirutil import fast_relpath
2✔
23

24

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

29
    snapshot: Snapshot
30

31

32
@rule
2✔
33
async def strip_source_roots(source_files: SourceFiles) -> StrippedSourceFiles:
2✔
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:
2✔
39
        return StrippedSourceFiles(source_files.snapshot)
2✔
40

41
    if source_files.unrooted_files:
2✔
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
2✔
48

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

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

57
    if len(source_roots_to_files) == 1:
2✔
58
        source_root = next(iter(source_roots_to_files.keys()))
2✔
59
        if source_root == ".":
2✔
60
            resulting_snapshot = rooted_files_snapshot
2✔
61
        else:
62
            resulting_snapshot = await digest_to_snapshot(
2✔
63
                **implicitly(RemovePrefix(rooted_files_snapshot.digest, source_root))
64
            )
65
    else:
66
        digest_subsets = await concurrently(
1✔
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(
1✔
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)))
1✔
75

76
    # Add the unrooted files back in.
77
    if source_files.unrooted_files:
2✔
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)
2✔
86

87

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

92

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

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

100

101
@rule
2✔
102
async def strip_file_name(request: StrippedFileNameRequest) -> StrippedFileName:
2✔
103
    source_root = await get_source_root(SourceRootRequest.for_file(request.file_path))
2✔
104
    return StrippedFileName(
2✔
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]):
2✔
112
    """The file names from a target's `sources` field, with source roots stripped."""
113

114

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

124

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

© 2026 Coveralls, Inc