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

pantsbuild / pants / 19381742489

15 Nov 2025 12:52AM UTC coverage: 49.706% (-30.6%) from 80.29%
19381742489

Pull #22890

github

web-flow
Merge d961abf79 into 42e1ebd41
Pull Request #22890: Updated all python subsystem constraints to 3.14

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

14659 existing lines in 485 files now uncovered.

31583 of 63540 relevant lines covered (49.71%)

0.79 hits per line

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

62.5
/src/python/pants/backend/shell/goals/package.py
1
# Copyright 2025 Pants project contributors (see CONTRIBUTORS.md).
2
# Licensed under the Apache License, Version 2.0 (see LICENSE).
3

4
from __future__ import annotations
1✔
5

6
import dataclasses
1✔
7
from dataclasses import dataclass
1✔
8

9
from pants.backend.shell.target_types import (
1✔
10
    ShellCommandCommandField,
11
    ShellCommandPackageDependenciesField,
12
    SkipShellCommandPackageField,
13
)
14
from pants.backend.shell.util_rules import shell_command
1✔
15
from pants.backend.shell.util_rules.shell_command import (
1✔
16
    ShellCommandProcessFromTargetRequest,
17
    prepare_process_request_from_target,
18
)
19
from pants.core.environments.target_types import EnvironmentField
1✔
20
from pants.core.goals.package import (
1✔
21
    BuiltPackage,
22
    BuiltPackageArtifact,
23
    OutputPathField,
24
    PackageFieldSet,
25
)
26
from pants.core.util_rules.adhoc_process_support import AdhocProcessRequest
1✔
27
from pants.core.util_rules.adhoc_process_support import rules as adhoc_process_support_rules
1✔
28
from pants.core.util_rules.adhoc_process_support import run_prepared_adhoc_process
1✔
29
from pants.engine.fs import EMPTY_DIGEST, AddPrefix, Digest
1✔
30
from pants.engine.internals.graph import resolve_target
1✔
31
from pants.engine.intrinsics import add_prefix, get_digest_entries
1✔
32
from pants.engine.process import ProcessCacheScope
1✔
33
from pants.engine.rules import collect_rules, implicitly, rule
1✔
34
from pants.engine.target import Target, WrappedTargetRequest
1✔
35
from pants.engine.unions import UnionRule
1✔
36
from pants.util.logging import LogLevel
1✔
37

38

39
@dataclass(frozen=True)
1✔
40
class PackageShellCommandFieldSet(PackageFieldSet):
1✔
41
    required_fields = (
1✔
42
        ShellCommandCommandField,
43
        ShellCommandPackageDependenciesField,
44
    )
45

46
    environment: EnvironmentField
1✔
47
    output_path: OutputPathField
1✔
48

49
    @classmethod
1✔
50
    def opt_out(cls, tgt: Target) -> bool:
1✔
UNCOV
51
        return tgt.get(SkipShellCommandPackageField).value
×
52

53

54
@rule(desc="Package with shell command", level=LogLevel.DEBUG)
1✔
55
async def package_shell_command(
1✔
56
    field_set: PackageShellCommandFieldSet,
57
) -> BuiltPackage:
58
    wrapped_tgt = await resolve_target(
×
59
        WrappedTargetRequest(field_set.address, description_of_origin="<infallible>"),
60
        **implicitly(),
61
    )
62
    target = wrapped_tgt.target
×
63

64
    shell_process = await prepare_process_request_from_target(
×
65
        ShellCommandProcessFromTargetRequest(target), **implicitly()
66
    )
67

68
    shell_process = dataclasses.replace(
×
69
        shell_process,
70
        cache_scope=shell_process.cache_scope or ProcessCacheScope.SUCCESSFUL,
71
    )
72

73
    result = await run_prepared_adhoc_process(**implicitly({shell_process: AdhocProcessRequest}))
×
74

75
    if result.process_result.exit_code != 0:
×
76
        raise ValueError(
×
77
            f"The `{target.alias}` at `{field_set.address}` failed with exit code {result.process_result.exit_code}.\n\n"
78
            f"stdout:\n\n{result.process_result.stdout.decode(errors='ignore')}\n\n"
79
            f"stderr:\n\n{result.process_result.stderr.decode(errors='ignore')}"
80
        )
81

82
    if result.adjusted_digest == EMPTY_DIGEST:
×
83
        raise ValueError(
×
84
            f"The `{target.alias}` at `{field_set.address}` did not produce or capture any output. "
85
            "The `package` goal is expected to produce output."
86
        )
87

88
    # Apply the output path for the artifacts.
89
    output_path = field_set.output_path.value_or_default(file_ending=None)
×
90
    output_digest: Digest
91
    if output_path:
×
92
        output_digest = await add_prefix(AddPrefix(result.adjusted_digest, output_path))
×
93
    else:
94
        output_digest = result.adjusted_digest
×
95

96
    output_digest_entries = await get_digest_entries(result.adjusted_digest)
×
97
    file_paths = sorted(e.path for e in output_digest_entries)
×
98
    artifacts = tuple(BuiltPackageArtifact(relpath=path) for path in file_paths)
×
99
    return BuiltPackage(output_digest, artifacts)
×
100

101

102
def rules():
1✔
103
    return (
1✔
104
        *collect_rules(),
105
        *shell_command.rules(),
106
        *adhoc_process_support_rules(),
107
        UnionRule(PackageFieldSet, PackageShellCommandFieldSet),
108
    )
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