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

Provide module extensions with a way to reference extension repos #19055

Closed
fmeum opened this issue Jul 25, 2023 · 16 comments
Closed

Provide module extensions with a way to reference extension repos #19055

fmeum opened this issue Jul 25, 2023 · 16 comments
Assignees
Labels
area-Bzlmod Bzlmod-specific PRs, issues, and feature requests P1 I'll work on this now. (Assignee required) team-ExternalDeps External dependency handling, remote repositiories, WORKSPACE file. type: feature request

Comments

@fmeum
Copy link
Collaborator

fmeum commented Jul 25, 2023

What underlying problem are you trying to solve with this feature?

The repository mapping that applies to a module extension does not include the repositories it generates. This makes it very tricky to pass a reference to some file in such a repo to another repo rule instantiated by the same extension, e.g.:

def _ext_impl(module_ctx):
    my_lang_toolchain_repo(
        name = "lang_sdk",
    )
    my_toolchain_repo(
        name = "lang_toolchain",
        # This does *not* work as `lang_sdk` is not in the current file's repo mapping.
        sdk_root = "@lang_sdk//:root",
    )

This is a very common pattern, but requires major hacks to pull of without additional API:

  1. Observe that the canonical name of an extension repo is <common prefix>~<apparent name> and rely on that implementation detail to construct the canonical name of lang_sdk in lang_toolchain as ctx.attr.name.removesuffix("lang_toolchain") + "lang_sdk".
  2. Make use of the fact that lang_sdk is in the repo mapping of the repositories generated by the extension. By making sdk_root an attr.string instead of an attr.label and generating a .bzl file in some other ("hub") repo in the the same extension that calls Label on a label string, lang_toolchain can pass that literal through the correct repo mapping. This requires the module to use_repo the hub repo and is overall pretty complex.

Description of the feature request:

Provide a function on module_ctx that converts the name of a repository generated by the extension and a repo-relative label into an instance of Label referencing the intended target.

This could look as follows:

# In rules_foo v1.2.3, extensions.bzl
def _ext_impl(module_ctx):
    repo(name = "foo", ...)
    repo(name = "bar", ...)

    # Returns `Label("@rules_foo~1.2.3~ext~foo//dir:name")`
    module_ctx.extension_repo_label("@foo//dir:name")
    # Returns `Label("@rules_foo~1.2.3~ext~bar//:name")`
    module_ctx.extension_repo_label("@bar//:name")

    # Fails immediately, argument has to be absolute
    module_ctx.extension_repo_label("//:name")
    # Fails immediately, label must not be canonical
    module_ctx.extension_repo_label("@@rules_foo~1.2.3~ext~foo//dir:name")
    # Fails immediately, repo name must be a valid "user-specified" name
    module_ctx.extension_repo_label("@rules_foo~1.2.3~ext~foo//dir:name")
    # Fails after the module extension returns as `not_foo` is not generated by the extension.
    module_ctx.extension_repo_label("@not_foo//dir:name")

Alternative ideas for names:

  • label_in_repo

Alternative signatures:

  • Separate arguments for apparent repository name and repo-relative label.

Alternative approaches:

  • Hook into the repo rule creation and defer the conversion of label strings into Label instances until after the module extension has fully evaluated, at which point the names of all the repos it generates are known. Then use this information to convert label strings implicitly. This has the advantage that the "obvious" approach just works and the downside that its implementation is probably more involved.

Which operating system are you running Bazel on?

N/A

What is the output of bazel info release?

6.3.0

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

No response

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

No response

Have you found anything relevant by searching the web?

No response

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

No response

@fmeum
Copy link
Collaborator Author

fmeum commented Jul 25, 2023

CC @Wyverald @meteorcloudy, I am especially interested in your opinion on the alternative approach. If it is feasible, I could see this being better than introducing new API.

@sgowroji sgowroji added the team-OSS Issues for the Bazel OSS team: installation, release processBazel packaging, website label Jul 25, 2023
@fmeum fmeum added team-ExternalDeps External dependency handling, remote repositiories, WORKSPACE file. and removed team-OSS Issues for the Bazel OSS team: installation, release processBazel packaging, website labels Jul 25, 2023
@Wyverald Wyverald added P1 I'll work on this now. (Assignee required) area-Bzlmod Bzlmod-specific PRs, issues, and feature requests and removed type: feature request labels Jul 25, 2023
@Wyverald Wyverald added this to Bzlmod Jul 25, 2023
@github-project-automation github-project-automation bot moved this to Todo in Bzlmod Jul 25, 2023
@Wyverald
Copy link
Member

Wyverald commented Jul 25, 2023

The alternative approach might not be possible since you could do this:

def _extension_impl(mctx):
  repo_rule(name="foo")
  print(str(Label("@foo//:abc")))
  # or with the API
  print(mctx.extension_repo_label("@foo//:abc"))

To clarify, we can't really "smartly" handle the first print. It's true that we may be able to smartly handle something like repo_rule(name="bar", some_label_attr="@foo//:abc"), but I'd say that if we can't smartly handle everything, we shouldn't smartly handle anything.

@fmeum
Copy link
Collaborator Author

fmeum commented Jul 25, 2023

Do we really want users to construct Labels into extension repos though? If they feed them to module_ctx.path, that would probably always result in cycles or other kinds of failures. Supporting label attributes of repo rules is what this is really about and I have a hard time coming up with any other usage that would make sense. But I agree that if we can think of just one then we should just expose a function that is guaranteed to handle all cases.

@Wyverald
Copy link
Member

I'm undecided. The decision probably needs more input data. Design doc time :)

@meteorcloudy
Copy link
Member

It's interesting to see how rules_java works around this:
https://github.com/bazelbuild/rules_java/blob/0878192f194292421788d35bea53533b0c730aec/toolchains/remote_java_repository.bzl#L86

It basically passes the label ("@{repo}//:jdk".format(repo = name)) as a plain string to generate the BUILD file, so that the label actually got resolved during build time. This works if the label isn't needed in the repo rule implementation function of my_toolchain_repo, which I don't know is often true or false.

@meteorcloudy meteorcloudy added P2 We'll consider working on this in future. (Assignee optional) and removed P1 I'll work on this now. (Assignee required) labels Aug 1, 2023
@meteorcloudy meteorcloudy added this to the 7.0.0 branch cut milestone Sep 19, 2023
@fmeum fmeum added P1 I'll work on this now. (Assignee required) and removed P2 We'll consider working on this in future. (Assignee optional) labels Aug 20, 2024
copybara-service bot pushed a commit that referenced this issue Aug 20, 2024
Since `Rule` logic drops a call stack entry, errors during repo rule instantiation in a module extension weren't reported with a useful `Location`.

Work towards #19055

Closes #23363.

PiperOrigin-RevId: 665468188
Change-Id: If67b5b6f290ac4e33b5743e9c1c361a93e1217ef
copybara-service bot pushed a commit that referenced this issue Aug 21, 2024
Eager fetching of all labels listed in repo rule attributes was introduced as a performance optimization to avoid costly restarts.

Now that restarts are gone by default, this is no longer a benefit as it can cause unnecessary fetches and also create cycles where there wouldn't be any without this behavior (e.g. when two repos write each others labels into a file without resolving them).

Work towards #19055

Closes #23371.

PiperOrigin-RevId: 665744319
Change-Id: Ia27f207793a2da3fb8e37743b328483f9d45192c
fmeum added a commit to fmeum/bazel that referenced this issue Aug 22, 2024
Eager fetching of all labels listed in repo rule attributes was introduced as a performance optimization to avoid costly restarts.

Now that restarts are gone by default, this is no longer a benefit as it can cause unnecessary fetches and also create cycles where there wouldn't be any without this behavior (e.g. when two repos write each others labels into a file without resolving them).

Work towards bazelbuild#19055

Closes bazelbuild#23371.

PiperOrigin-RevId: 665744319
Change-Id: Ia27f207793a2da3fb8e37743b328483f9d45192c
fmeum added a commit to fmeum/bazel that referenced this issue Aug 22, 2024
Since `Rule` logic drops a call stack entry, errors during repo rule instantiation in a module extension weren't reported with a useful `Location`.

Work towards bazelbuild#19055

Closes bazelbuild#23363.

PiperOrigin-RevId: 665468188
Change-Id: If67b5b6f290ac4e33b5743e9c1c361a93e1217ef
@github-project-automation github-project-automation bot moved this from Todo to Done in Bzlmod Aug 23, 2024
github-merge-queue bot pushed a commit that referenced this issue Aug 26, 2024
…extensions (#23390)

Since `Rule` logic drops a call stack entry, errors during repo rule
instantiation in a module extension weren't reported with a useful
`Location`.

Work towards #19055

Closes #23363.

PiperOrigin-RevId: 665468188
Change-Id: If67b5b6f290ac4e33b5743e9c1c361a93e1217ef

Closes #23364
fmeum added a commit to fmeum/bazel that referenced this issue Sep 5, 2024
fmeum added a commit to fmeum/bazel that referenced this issue Sep 5, 2024
fmeum added a commit to fmeum/bazel that referenced this issue Sep 5, 2024
copybara-service bot pushed a commit that referenced this issue Sep 9, 2024
This was missed in #19055.

Closes #23527.

PiperOrigin-RevId: 672480631
Change-Id: Ib2ef0edb6b6c90a7336085da20e2794a93763182
iancha1992 pushed a commit to iancha1992/bazel that referenced this issue Sep 9, 2024
This was missed in bazelbuild#19055.

Closes bazelbuild#23527.

PiperOrigin-RevId: 672480631
Change-Id: Ib2ef0edb6b6c90a7336085da20e2794a93763182
@Wyverald
Copy link
Member

@bazel-io fork 7.4.0

Wyverald pushed a commit that referenced this issue Sep 10, 2024
Fixes #19055

RELNOTES: Repository rules instantiated in the same module extensions can now refer to each other by their extension-specified names in label attributes.

Closes #23369.

PiperOrigin-RevId: 666893202
Change-Id: Ib2eaa55fcb563adc86e16dc4a357ac808228ebda
github-merge-queue bot pushed a commit that referenced this issue Sep 10, 2024
…23585)

Fixes #19055

RELNOTES: Repository rules instantiated in the same module extensions
can now refer to each other by their extension-specified names in label
attributes.

Closes #23369.

PiperOrigin-RevId: 666893202
Change-Id: Ib2eaa55fcb563adc86e16dc4a357ac808228ebda

---------

Co-authored-by: Fabian Meumertzheim <[email protected]>
iancha1992 pushed a commit to iancha1992/bazel that referenced this issue Sep 11, 2024
This was missed in bazelbuild#19055.

Closes bazelbuild#23527.

PiperOrigin-RevId: 672480631
Change-Id: Ib2ef0edb6b6c90a7336085da20e2794a93763182
iancha1992 pushed a commit to iancha1992/bazel that referenced this issue Sep 11, 2024
This was missed in bazelbuild#19055.

Closes bazelbuild#23527.

PiperOrigin-RevId: 672480631
Change-Id: Ib2ef0edb6b6c90a7336085da20e2794a93763182
iancha1992 pushed a commit to iancha1992/bazel that referenced this issue Sep 11, 2024
This was missed in bazelbuild#19055.

Closes bazelbuild#23527.

PiperOrigin-RevId: 672480631
Change-Id: Ib2ef0edb6b6c90a7336085da20e2794a93763182
github-merge-queue bot pushed a commit that referenced this issue Sep 11, 2024
…ges (#23564)

This was missed in #19055.

Closes #23527.

PiperOrigin-RevId: 672480631
Change-Id: Ib2ef0edb6b6c90a7336085da20e2794a93763182

Commit
926b574

---------

Co-authored-by: Fabian Meumertzheim <[email protected]>
Co-authored-by: Xùdōng Yáng <[email protected]>
@iancha1992
Copy link
Member

A fix for this issue has been included in Bazel 7.4.0 RC1. Please test out the release candidate and report any issues as soon as possible.
If you're using Bazelisk, you can point to the latest RC by setting USE_BAZEL_VERSION=7.4.0rc1. Thanks!

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
area-Bzlmod Bzlmod-specific PRs, issues, and feature requests P1 I'll work on this now. (Assignee required) team-ExternalDeps External dependency handling, remote repositiories, WORKSPACE file. type: feature request
Projects
Status: Done
Development

Successfully merging a pull request may close this issue.

6 participants