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

pantsbuild / pants / 21552830208

31 Jan 2026 11:40PM UTC coverage: 80.277% (-0.05%) from 80.324%
21552830208

Pull #23062

github

web-flow
Merge 808a9786c into 2c4dcf9cf
Pull Request #23062: Remove support for Get

18 of 25 new or added lines in 4 files covered. (72.0%)

17119 existing lines in 541 files now uncovered.

78278 of 97510 relevant lines covered (80.28%)

3.36 hits per line

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

63.83
/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
4✔
5

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

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

40

41
@dataclass(frozen=True)
4✔
42
class PackageShellCommandFieldSet(PackageFieldSet):
4✔
43
    required_fields = (
4✔
44
        ShellCommandCommandField,
45
        ShellCommandPackageDependenciesField,
46
    )
47

48
    environment: EnvironmentField
4✔
49
    output_path: OutputPathField
4✔
50

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

55

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

66
    shell_process = await prepare_process_request_from_target(
×
67
        ShellCommandProcessFromTargetRequest(target), **implicitly()
68
    )
69

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

75
    result = await run_prepared_adhoc_process(**implicitly({shell_process: AdhocProcessRequest}))
×
76

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

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

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

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

103

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