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

feat: Allow custom bootstrap template in python_repository #2032

Open
wants to merge 4 commits into
base: main
Choose a base branch
from

Conversation

tgeng
Copy link
Contributor

@tgeng tgeng commented Jul 3, 2024

Before, users need to reimplement the entire
python_repository repo rule in order to pass different bootstrap templates to the underlying py_runtime.

After, users can simply pass custom templates to bootstrap_template and stage2_bootstrap_template when registering python toolchains using python_repository or python_register_toolchains.

This PR allows users to set custom `bootstrap_template` and
`stage2_bootstrap_template` when registering python toolchains using
`python_repository` and `python_register_toolchains`.

Otherwise users need to reimplement the entire
`python_repository` repo rule in order to pass different bootstrap
templates to the underlying `py_runtime`.
@tgeng tgeng requested review from rickeylev and aignas as code owners July 3, 2024 22:17
@aignas
Copy link
Collaborator

aignas commented Jul 4, 2024

Thanks for the PR. What is the usecase for overriding these templates?

@tgeng
Copy link
Contributor Author

tgeng commented Jul 4, 2024

A couple of things:
We are trying to adopt bazel for our internal python code base that has many native dependencies.

  1. We need to set LD_LIBRARY_PATH to absolute forms and apparently there is no way to do it inside starlark since the exec root is only available during the execution phase
  2. We have many tests that import a lot of modules and hence running individual py_test becomes very slow. Hence we are thinking about implementing a test daemon that batches test dynamically by delegating python calls to a few worker processes (similar to persistent workers, to reduce module import time). And we plan to use this daemon with our internal remote execution service (afaik persistent workers don't with with rbe out of box). (There will be test pollution issues but we are running tests in dynamic batches already out of Bazel so we hope adopting Bazel this way won't be a problem.)

@rickeylev
Copy link
Collaborator

Hm, I'm not sure how I feel about this feature. I'm not sure our repo rules having large APIs has really aged very well. This API would also need to be ported over to the bzlmod extension and have the appropriate is_root checks added to it.

Going through the repo-rule also means the values have to be re-specified for each registration. Is that a good thing, or bad thing? IDK; answer doesn't seem obvious to me.

An alternative would be to use a flag; you would set it in your bazelrc or command line.

We need to set LD_LIBRARY_PATH ...

Can you tell a bit more about your use case? This sounds like something py_binary et al should handle for you (e.g., if you depend on libfoo, the rules do the needful to ensure libfoo is on the ld search path).

implementing a test daemon

Oh, wow, yeah, you'll probably need to be customizing something about the way a binary is run to do that. You might have a hard time, though; AFICT, Bazel doesn't have any support for a persistent test process. It looks like there used to be experimental support for it, but it was removed a couple years ago due to being buggy. Good luck :)

@tgeng
Copy link
Contributor Author

tgeng commented Jul 11, 2024

Thanks for the detailed response! Setting bootstrap script through flags will work for us too.

I am not super clear about how depending on cc_library would work. Our python tests depend on many native libraries at runtime (libtorch, libcudnn, libcudart, etc). Some of these so files are loaded through in-house repo rules that downloads and unpacks Debian packages; some other so files are loaded via custom python wheel builder rules (based on rules_pycross). All of these currently are passed as data dependency of the py_test and we set LD_LIBRARY_PATH to point to these so files. But since bazel's make variable expansion only expands to relative path, we need to expand the paths to absolute form in the bootstrapper in order for various python packages to find the so files (since cwd may change in the long chain of calls, sometimes across process boundaries).

The test daemon service would behave just like normal python interpreter, except it delegates computation to another shared process. Afaik the biggest trouble with bazel's persistent test worker is related to test state pollution. But since we are doing batch already we are optimistic about this issue.

@tgeng
Copy link
Contributor Author

tgeng commented Jul 11, 2024

BTW, I noticed an issue with stage2_bootstrap_template at

sys.path.extend(python_path_entries)

Because the extra path entries are appended, it seems impossible to overwrite packages provided by python itself, for example, a newer version of pip. Also, it also seems that python would honor the user packages under ~/.local/lib/python3.10/site-packages. I wonder if it makes sense to prepend all the path entries. This is another reason why we are using our own bootstrap template

@rickeylev
Copy link
Collaborator

Because the extra path entries are appended, it seems impossible to overwrite packages provided by python itself

This is mostly WAI, for two reasons:

The first is it better matches the sys.path of a regular python invocation. The order of sys.path is: PWD, PYTHONPATH, stdlib, site-packages. Things like pypi dependencies are, conceptually, in the position of site-packages.

The second is it avoids issues with other directories masking stdlib directories, which results in errors that are very confusing. The main cause of this is repo directories being added to sys.path (which is still the unfortunate default behavior of rules_python)

overriding pip

I don't think pip is actually part of the stdlib? But it is part of site-packages. I think what's happening is, because all the pypi deps are appended, they end up after site-packages. Which, yeah, that would mean a pypi dependency on a newer pip wouldn't be respected.

Hm, yeah, maybe we should change that ordering, so long as we keep them after the stdlib entry. I filed #2064 about this

@aignas
Copy link
Collaborator

aignas commented Jul 15, 2024 via email

@rickeylev
Copy link
Collaborator

shared libs via data dependency and setting LD_LIBRARY_PATH

Thanks for explaining. Yeah, using cc_library with a prebuilt shared library is part of the solution (this gets meta-data about the library into Bazel). The other half is generating linker paths (rpath, ld_library_path) that point to where those shared libraries end up in runfiles. Making that work is a bit of a ways out, though

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants