Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Incompatible Target Skipping ignores visibility restrictions #16044

Closed
philsc opened this issue Aug 4, 2022 · 11 comments
Closed

Incompatible Target Skipping ignores visibility restrictions #16044

philsc opened this issue Aug 4, 2022 · 11 comments
Labels
P3 We're not considering working on this, but happy to review a PR. (No assignee) stale Issues or PRs that are stale (no activity for 30 days) team-Configurability platforms, toolchains, cquery, select(), config transitions type: bug

Comments

@philsc
Copy link
Contributor

philsc commented Aug 4, 2022

Description of the bug:

With #14096, we're fixing a few usability issues with Incompatible Target Skipping. That is accomplished by moving the code to a much earlier stage of the analysis phase. Unfortunately, that means it runs before visibility is checked.

This is not inherently a problem as it doesn't allow you to bypass visibility restrictions for building code. Compatible targets still obey visibility restrictions. And incompatible targets cannot be built.

What's the simplest, easiest way to reproduce this bug? Please provide a minimal example if possible.

Create an incompatible target in package //a.

# a/BUILD
filegroup(
    name = "private_filegroup",
    visibility = ["//visibility:private"],
    srcs = [...],
)
filegroup(
    name = "private_incompatible_filegroup",
    visibility = ["//visibility:private"],
    target_compatible_with = ["@platforms//:incompatible"],
    srcs = [...],
)

Create a target that is either directly incompatible or indirectly incompatible in a different package that depends on one of the private targets.

# b/BUILD
filegroup(
    name = "directly_incompatible",
    srcs = [
        "//a:private_filegroup",
    ],
    target_compatible_with = ["@platforms//:incompatible"],
)
filegroup(
    name = "indirectly_incompatible",
    srcs = [
        "//a:private_incompatible_filegroup",
    ],
)

Neither //b:directly_incompatible nor //b:indirectly_incompatible will generate errors about visibility restrictions.

One possible course of action is to move the visibility checking earlier. #14096 (comment)

Which operating system are you running Bazel on?

x86_64 Debian 11

What is the output of bazel info release?

No response

If bazel info release returns development version or (@non-git), tell us how you built Bazel.

This is cbe2532 but with #14096 merged in.

I'm filing the bug report preemptively.

What's the output of git remote get-url origin; git rev-parse master; git rev-parse HEAD ?

$ git remote get-url upstream; git rev-parse master; git rev-parse HEAD
https://github.com/bazelbuild/bazel
e860453a7be19cc557d02b4edec13819358945fe
21a5fd1b5e091d4fb26ba683bfb3ff31d2f16621

Have you found anything relevant by searching the web?

No response

Any other information, logs, or outputs that you want to share?

The hope is to merge #14096 and separately work on this bug.

@sgowroji sgowroji added team-Configurability platforms, toolchains, cquery, select(), config transitions untriaged type: bug labels Aug 4, 2022
@philsc
Copy link
Contributor Author

philsc commented Aug 6, 2022

To clarify, I think that a directly incompatible target should be allowed to depend on private targets without error. The point here is that directly incompatible targets don't have their dependencies analysed. This also means avoiding visibility checking.

Indirectly incompatible targets, however, should honor visibility restrictions.

@gregestren , does that sound right to you?

@gregestren gregestren added P3 We're not considering working on this, but happy to review a PR. (No assignee) and removed untriaged labels Aug 8, 2022
@gregestren
Copy link
Contributor

gregestren commented Aug 8, 2022

i.e. if target A depends on B and B is both private and incompatible, we'd want a build of A to report a visibility error?

@philsc
Copy link
Contributor Author

philsc commented Aug 8, 2022

Yes, that's correct. Does that sound reasonable?

@gregestren
Copy link
Contributor

Yep!

copybara-service bot pushed a commit that referenced this issue Aug 9, 2022
This patch reworks how Incompatible Target Skipping is implemented.

The biggest change is that incompatibility is now checked earlier. This
allows us to avoid evaluating dependencies (such as toolchains) in
situations where a target is "directly" incompatible. "Direct incompatibility"
refers to incompatibility due to a target's `target_compatible_with` value.

Moving the incompatibility checking earlier had the undesired effect of
visibility restrictions being ignored on incompatible targets. This is
tracked in #16044. As per
#14096 (comment),
we will fix it in a separate patch.

Fixes #12897.

Closes #14096.

PiperOrigin-RevId: 466407887
Change-Id: I3014390ddb95c7ff6bfaaf497a32b60c8a6e8fc9
philsc added a commit to philsc/bazel that referenced this issue Sep 14, 2022
The latest refactor unintentionally made it so you can list the same
constraint multiple times in the `target_compatible_with` attribute.

It was unintentional, but actually greatly simplifies a discussion
point on bazelbuild/bazel-skylib#381. That skylib patch aims to make
it easier to write non-trivial `target_compatible_with` expressions.
For example, to express that something is compatible with everything
except for Windows, one can use the following:

    foo_binary(
        name = "bin",
        target_compatible_with = select({
            "@platforms//os:windows": ["@platforms//:incomaptible"],
            "//conditions:default: [],
        }),
    )

The skylib patch aims to reduce that to:

    foo_binary(
        name = "bin",
        target_compatible_with = compatibility.none_of("@platforms//os:windows"),
    )

This works fine on its own, but a problem arises when these
expressions are composed. For example, a macro author might write the
following:

    def foo_wrapped_binary(name, target_compatible_with = [], **kwargs):
        foo_binary(
            name = name,
            target_compatible_with = target_compatible_with + select({
                "@platforms//os:none": ["@platforms//:incompatible"],
                "//conditions:default": [],
            }),
        )

A macro author should be able to express their own incompatibility
while also honouring user-defined incompatibility.

It turns out that my latest refactor (bazelbuild#14096) unintentionally made
this work. This happened because we now check for incompatibility
before we perform a lot of sanity checks. That's also one of the
reasons that visibility restrictions are not honoured for incomaptible
targets at the moment (bazelbuild#16044).

Without the deduplicating behaviour, macro authors are stuck with
either not allowing composition, or having to create unique
incompatible constraints for every piece in a composed
`target_compatible_with` expression.

This patch adds a test to validate the deduplicating behaviour to
cement it as a feature.

Unfortunately, this means that `target_compatible_with` behaves
differently from other label list attributes. For other label list
attributes, bazel complains when labels are duplicated. For example,
the following BUILD file:

    py_library(
        name = "lib",
    )

    py_binary(
        name = "bin",
        srcs = ["bin.py"],
        deps = [
        |   ":lib",
        |   ":lib",
        ],
    )

will result in the following error:

    $ bazel build :bin
    INFO: Invocation ID: 3d8345a4-2e76-493e-8343-c96a36185b52
    ERROR: /home/jenkins/repos/test_repo/BUILD:41:10: Label '//:lib' is duplicated in the 'deps' attribute of rule 'bin'
    ERROR: error loading package '': Package '' contains errors
    INFO: Elapsed time: 2.514s
    INFO: 0 processes.
    FAILED: Build did NOT complete successfully (1 packages loaded)
copybara-service bot pushed a commit that referenced this issue Sep 16, 2022
The latest refactor unintentionally made it so you can list the same
incompatible constraint multiple times in the `target_compatible_with`
attribute.

It was unintentional, but actually greatly simplifies a discussion
point on bazelbuild/bazel-skylib#381. That skylib patch aims to make
it easier to write non-trivial `target_compatible_with` expressions.
For example, to express that something is compatible with everything
except for Windows, one can use the following:

    foo_binary(
        name = "bin",
        target_compatible_with = select({
            "@platforms//os:windows": ["@platforms//:incomaptible"],
            "//conditions:default: [],
        }),
    )

The skylib patch aims to reduce that to:

    foo_binary(
        name = "bin",
        target_compatible_with = compatibility.none_of("@platforms//os:windows"),
    )

This works fine on its own, but a problem arises when these
expressions are composed. For example, a macro author might write the
following:

    def foo_wrapped_binary(name, target_compatible_with = [], **kwargs):
        foo_binary(
            name = name,
            target_compatible_with = target_compatible_with + select({
                "@platforms//os:none": ["@platforms//:incompatible"],
                "//conditions:default": [],
            }),
        )

A macro author should be able to express their own incompatibility
while also honouring user-defined incompatibility.

It turns out that my latest refactor (#14096) unintentionally made
this work. This happened because we now check for incompatibility
before we perform a lot of sanity checks. That's also one of the
reasons that visibility restrictions are not honoured for incomaptible
targets at the moment (#16044).

Without the deduplicating behaviour, macro authors are stuck with
either not allowing composition, or having to create unique
incompatible constraints for every piece in a composed
`target_compatible_with` expression.

This patch adds a test to validate the deduplicating behaviour to
cement it as a feature.

Unfortunately, this means that `target_compatible_with` behaves
differently from other label list attributes. For other label list
attributes, bazel complains when labels are duplicated. For example,
the following BUILD file:

    py_library(
        name = "lib",
    )

    py_binary(
        name = "bin",
        srcs = ["bin.py"],
        deps = [
        |   ":lib",
        |   ":lib",
        ],
    )

will result in the following error:

    $ bazel build :bin
    INFO: Invocation ID: 3d8345a4-2e76-493e-8343-c96a36185b52
    ERROR: /home/jenkins/repos/test_repo/BUILD:41:10: Label '//:lib' is duplicated in the 'deps' attribute of rule 'bin'
    ERROR: error loading package '': Package '' contains errors
    INFO: Elapsed time: 2.514s
    INFO: 0 processes.
    FAILED: Build did NOT complete successfully (1 packages loaded)

Closes #16274.

PiperOrigin-RevId: 474747867
Change-Id: Iacc4de12ce90176d803e0bc9e695d4ec14603449
aiuto pushed a commit to aiuto/bazel that referenced this issue Oct 12, 2022
The latest refactor unintentionally made it so you can list the same
incompatible constraint multiple times in the `target_compatible_with`
attribute.

It was unintentional, but actually greatly simplifies a discussion
point on bazelbuild/bazel-skylib#381. That skylib patch aims to make
it easier to write non-trivial `target_compatible_with` expressions.
For example, to express that something is compatible with everything
except for Windows, one can use the following:

    foo_binary(
        name = "bin",
        target_compatible_with = select({
            "@platforms//os:windows": ["@platforms//:incomaptible"],
            "//conditions:default: [],
        }),
    )

The skylib patch aims to reduce that to:

    foo_binary(
        name = "bin",
        target_compatible_with = compatibility.none_of("@platforms//os:windows"),
    )

This works fine on its own, but a problem arises when these
expressions are composed. For example, a macro author might write the
following:

    def foo_wrapped_binary(name, target_compatible_with = [], **kwargs):
        foo_binary(
            name = name,
            target_compatible_with = target_compatible_with + select({
                "@platforms//os:none": ["@platforms//:incompatible"],
                "//conditions:default": [],
            }),
        )

A macro author should be able to express their own incompatibility
while also honouring user-defined incompatibility.

It turns out that my latest refactor (bazelbuild#14096) unintentionally made
this work. This happened because we now check for incompatibility
before we perform a lot of sanity checks. That's also one of the
reasons that visibility restrictions are not honoured for incomaptible
targets at the moment (bazelbuild#16044).

Without the deduplicating behaviour, macro authors are stuck with
either not allowing composition, or having to create unique
incompatible constraints for every piece in a composed
`target_compatible_with` expression.

This patch adds a test to validate the deduplicating behaviour to
cement it as a feature.

Unfortunately, this means that `target_compatible_with` behaves
differently from other label list attributes. For other label list
attributes, bazel complains when labels are duplicated. For example,
the following BUILD file:

    py_library(
        name = "lib",
    )

    py_binary(
        name = "bin",
        srcs = ["bin.py"],
        deps = [
        |   ":lib",
        |   ":lib",
        ],
    )

will result in the following error:

    $ bazel build :bin
    INFO: Invocation ID: 3d8345a4-2e76-493e-8343-c96a36185b52
    ERROR: /home/jenkins/repos/test_repo/BUILD:41:10: Label '//:lib' is duplicated in the 'deps' attribute of rule 'bin'
    ERROR: error loading package '': Package '' contains errors
    INFO: Elapsed time: 2.514s
    INFO: 0 processes.
    FAILED: Build did NOT complete successfully (1 packages loaded)

Closes bazelbuild#16274.

PiperOrigin-RevId: 474747867
Change-Id: Iacc4de12ce90176d803e0bc9e695d4ec14603449
@philsc
Copy link
Contributor Author

philsc commented Nov 1, 2022

@gregestren , do you happen to have some advice for me on fixing this?

I think part of the simplest solution is to call ConfiguredTargetFactory::convertVisibility() from IncompatibleTargetChecker instead of the convertVisibility() stub that's currently there
Unfortunately I can't figure out how to pass a function as an argument to another function. Everything I find online makes it look tedious or overly complicated.

I considered moving ConfiguredTargetFactory::convertVisibility() into a separate java_library, but that's proving difficult because it would introduce circular dependencies with :analysis_cluster. So I think passing a function pointer around is simplest. But maybe I'm not thinking "java-like" enough.

What's your thought on this? My Java-fu is not good enough to make a decision.

@gregestren
Copy link
Contributor

How do you imagine passing the function pointer (i.e. where is the pointer defined and where does it get passed into IncompatibleTargetChecker)?

Would a direct function call from IncompatibleTargetChecker::convertVisibility() into ConfiguredTargetFactory::convertVisibility() trigger the circular dependency?

@philsc
Copy link
Contributor Author

philsc commented Nov 3, 2022

I am embarrassed. I got ConfiguredTargetFactory and ConfiguredTargetFunction mixed up in my head. I don't think there's a problem. I should be able to call ConfiguredTargetFactory::convertVisibility() from IncompatibleTargetChecker::convertVisibility().

Sorry about the noise.

Copy link

github-actions bot commented Jan 8, 2024

Thank you for contributing to the Bazel repository! This issue has been marked as stale since it has not had any activity in the last 1+ years. It will be closed in the next 90 days unless any other activity occurs or one of the following labels is added: "not stale", "awaiting-bazeler". Please reach out to the triage team (@bazelbuild/triage) if you think this issue is still relevant or you are interested in getting the issue resolved.

@github-actions github-actions bot added the stale Issues or PRs that are stale (no activity for 30 days) label Jan 8, 2024
Copy link

github-actions bot commented Apr 7, 2024

This issue has been automatically closed due to inactivity. If you're still interested in pursuing this, please post @bazelbuild/triage in a comment here and we'll take a look. Thanks!

@github-actions github-actions bot closed this as not planned Won't fix, can't repro, duplicate, stale Apr 7, 2024
@SamB
Copy link

SamB commented Jan 7, 2025

@bazelbuild/triage

This issue is still linked from the docs: https://bazel.build/extending/platforms#known_issues

It's not clear if it is fixed to the extent that it can be, or exactly what that means. If the code isn't going to get better, the documentation should probably be clarified a bit, perhaps with an example of how this can happen?

@gregestren
Copy link
Contributor

Thanks for catching the doc reference, @SamB .

I just checked with Bazel 8.0 and looks like behavior is the same:

$  USE_BAZEL_VERSION=latest bazelisk build --nobuild //b:indirectly_incompatible
ERROR: Target //b:indirectly_incompatible is incompatible and cannot be built, but was explicitly requested.
Dependency chain:
    //b:indirectly_incompatible (ae2685)
    //a:private_incompatible_filegroup (ae2685)   <-- target platform (@@platforms//host:host) didn't satisfy constraint @@platforms//:incompatible
INFO: Elapsed time: 0.084s
INFO: 0 processes.
ERROR: Build did NOT complete successfully

It's not clear if it is fixed to the extent that it can be, or exactly what that means. If the code isn't going to get better, the documentation should probably be clarified a bit, perhaps with an example of how this can happen?

Do you find the issue description confusing? Or just that given its history you can no longer tell how current it is?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
P3 We're not considering working on this, but happy to review a PR. (No assignee) stale Issues or PRs that are stale (no activity for 30 days) team-Configurability platforms, toolchains, cquery, select(), config transitions type: bug
Projects
None yet
Development

No branches or pull requests

4 participants