rules_python icon indicating copy to clipboard operation
rules_python copied to clipboard

Can Gazelle's 'resolve' directive be smarter? Or am I just using it wrong?

Open dougthor42 opened this issue 2 months ago • 1 comments

:question: question | possible 🚀 feature request

Relevant Rules

  • gazelle

Description

Can Gazelle's 'resolve' directive be smarter? Or am I just using it wrong?

Example:

# gazelle:resolve py foo @other//foo
Import Expected Dep Actual Dep
import foo.bar.baz "@other//foo/bar:baz" "@other//foo"
from foo import bar "@other//foo:bar" "@other//foo"
import foo.bar as bar "@other//foo/bar" "@other//foo"

This is coming about because we have a nested project with a dir structure like:

./main_project_repo
+ BUILD.bazel
+ MODULE.bazel  # module(name = "main_project_repo")
+ src/
    + BUILD.bazel  # sets python_root
    + main_project/
        + BUILD.bazel
        + foo.py      # imports other_project_repo/src/other_project/bar.py
                      # via `import other_project.bar` thanks to PYTHONPATH
        + private.py
+ other_project_repo/
    + BUILD.bazel
    + MODULE.bazel  # module(name = "other_project_repo")
    + src/
        + BUILD.bazel  # sets python_root
        + other_project/
            + BUILD.bazel
            + bar.py

And ./main_project_repo/src/main_project/BUILD.bazel looks like so to work:

# //src/main_project:foo
py_library(
    name = "foo",
    srcs = ["foo.py"],
    deps = [
        ":private",
        "@other_project_repo//src/other_project:bar"
    ],
)

With the current functionality, I'd have to write out resolve directives for every import used

# gazelle:resolve py other_project @other_project_repo//src/other_project
# gazelle:resolve py other_project.bar @other_project_repo//src/other_project:bar
# gazelle:resolve py other_project.foobar @other_project_repo//src/other_project/foobar
# gazelle:resolve py other_project.foobar.hello @other_project_repo//src/other_project/foobar:hello
# gazelle:resolve py other_project.foobar.goodbye @other_project_repo//src/other_project/foobar:goodbye
...

Sadly this is not feasible given the size of our codebases.

Describe the solution you'd like

I may be in the wrong, but it seems like the directive should match the start of the import and then follow the rest of the import path. I'm not exactly sure how, or even if, this would work though (see example 2).

Example 1

# gazelle:resolve py foo @other//foo
# import foo      -->  @other//foo
# import foo.bar  -->  @other//foo:bar

Example 2

# gazelle resolve py hello.world //goodbye:world
# import hello              -->  //hello   # no change because it's not matched by resolve.
# import hello.world        -->  //goodbye:world  # no change from current functionality
# import hello.world.again  -->  ... I donno. Can't be //goodbye:world:again haha

Describe alternatives you've considered

We might be able to run buildozer and modify all the deps in ./main_project_repo/src after Gazelle generates them, but I haven't looked into how feasible that actually is.

dougthor42 avatar May 06 '24 03:05 dougthor42