diff --git a/src/main/java/com/google/devtools/build/lib/bazel/rules/python/BazelPyRuleClasses.java b/src/main/java/com/google/devtools/build/lib/bazel/rules/python/BazelPyRuleClasses.java index fedc65cc494133..3282c86bbe6f50 100644 --- a/src/main/java/com/google/devtools/build/lib/bazel/rules/python/BazelPyRuleClasses.java +++ b/src/main/java/com/google/devtools/build/lib/bazel/rules/python/BazelPyRuleClasses.java @@ -36,6 +36,7 @@ import com.google.devtools.build.lib.rules.python.PyCommon; import com.google.devtools.build.lib.rules.python.PyInfo; import com.google.devtools.build.lib.rules.python.PyRuleClasses; +import com.google.devtools.build.lib.rules.python.PyRuntimeInfo; import com.google.devtools.build.lib.rules.python.PythonVersion; /** @@ -221,6 +222,11 @@ responsible for creating (possibly empty) __init__.py files and adding them to t .add( attr("$py_toolchain_type", NODEP_LABEL) .value(env.getToolsLabel("//tools/python:toolchain_type"))) + /* Only used when no py_runtime() is available. See #7901 + */ + .add( + attr("$default_bootstrap_template", LABEL) + .value(env.getToolsLabel(PyRuntimeInfo.DEFAULT_BOOTSTRAP_TEMPLATE))) .addToolchainTypes( ToolchainTypeRequirement.builder(env.getToolsLabel("//tools/python:toolchain_type")) .mandatory(true) diff --git a/src/main/java/com/google/devtools/build/lib/bazel/rules/python/BazelPythonSemantics.java b/src/main/java/com/google/devtools/build/lib/bazel/rules/python/BazelPythonSemantics.java index 44272182ac5171..dce0cdacb5ae41 100644 --- a/src/main/java/com/google/devtools/build/lib/bazel/rules/python/BazelPythonSemantics.java +++ b/src/main/java/com/google/devtools/build/lib/bazel/rules/python/BazelPythonSemantics.java @@ -36,7 +36,6 @@ import com.google.devtools.build.lib.analysis.actions.LauncherFileWriteAction.LaunchInfo; import com.google.devtools.build.lib.analysis.actions.SpawnAction; import com.google.devtools.build.lib.analysis.actions.Substitution; -import com.google.devtools.build.lib.analysis.actions.Template; import com.google.devtools.build.lib.analysis.actions.TemplateExpansionAction; import com.google.devtools.build.lib.cmdline.LabelConstants; import com.google.devtools.build.lib.collect.nestedset.NestedSetBuilder; @@ -62,8 +61,6 @@ public class BazelPythonSemantics implements PythonSemantics { public static final Runfiles.EmptyFilesSupplier GET_INIT_PY_FILES = new PythonUtils.GetInitPyFiles((Predicate & Serializable) source -> false); - private static final Template STUB_TEMPLATE = - Template.forResource(BazelPythonSemantics.class, "python_stub_template.txt"); public static final PathFragment ZIP_RUNFILES_DIRECTORY_NAME = PathFragment.create("runfiles"); @@ -166,12 +163,14 @@ private static void createStubFile( attrVersion = config.getDefaultPythonVersion(); } + Artifact bootstrapTemplate = getBootstrapTemplate(ruleContext, common); + // Create the stub file. ruleContext.registerAction( new TemplateExpansionAction( ruleContext.getActionOwner(), + bootstrapTemplate, stubOutput, - STUB_TEMPLATE, ImmutableList.of( Substitution.of("%shebang%", getStubShebang(ruleContext, common)), Substitution.of("%main%", common.determineMainExecutableSource()), @@ -285,8 +284,7 @@ public void createExecutable( /** Registers an action to create a Windows Python launcher at {@code pythonLauncher}. */ private static void createWindowsExeLauncher( - RuleContext ruleContext, String pythonBinary, Artifact pythonLauncher, boolean useZipFile) - throws InterruptedException { + RuleContext ruleContext, String pythonBinary, Artifact pythonLauncher, boolean useZipFile) { LaunchInfo launchInfo = LaunchInfo.builder() .addKeyValuePair("binary_type", "Python") @@ -337,7 +335,7 @@ private static String getZipRunfilesPath( } // We put the whole runfiles tree under the ZIP_RUNFILES_DIRECTORY_NAME directory, by doing this // , we avoid the conflict between default workspace name "__main__" and __main__.py file. - // Note: This name has to be the same with the one in python_stub_template.txt. + // Note: This name has to be the same with the one in python_bootstrap_template.txt. return ZIP_RUNFILES_DIRECTORY_NAME.getRelative(zipRunfilesPath).toString(); } @@ -427,6 +425,17 @@ private static PyRuntimeInfo getRuntime(RuleContext ruleContext, PyCommon common : ruleContext.getPrerequisite(":py_interpreter", PyRuntimeInfo.PROVIDER); } + private static Artifact getBootstrapTemplate(RuleContext ruleContext, PyCommon common) { + PyRuntimeInfo provider = getRuntime(ruleContext, common); + if (provider != null) { + Artifact bootstrapTemplate = provider.getBootstrapTemplate(); + if (bootstrapTemplate != null) { + return bootstrapTemplate; + } + } + return ruleContext.getPrerequisiteArtifact("$default_bootstrap_template"); + } + private static void addRuntime( RuleContext ruleContext, PyCommon common, Runfiles.Builder builder) { PyRuntimeInfo provider = getRuntime(ruleContext, common); diff --git a/src/main/java/com/google/devtools/build/lib/rules/python/PyRuntime.java b/src/main/java/com/google/devtools/build/lib/rules/python/PyRuntime.java index 5046be6ea52368..a01b384b6e53b9 100644 --- a/src/main/java/com/google/devtools/build/lib/rules/python/PyRuntime.java +++ b/src/main/java/com/google/devtools/build/lib/rules/python/PyRuntime.java @@ -110,13 +110,25 @@ public ConfiguredTarget create(RuleContext ruleContext) return null; } Preconditions.checkState(pythonVersion.isTargetValue()); + Artifact bootstrapTemplate = ruleContext.getPrerequisiteArtifact("bootstrap_template"); PyRuntimeInfo provider = hermetic ? PyRuntimeInfo.createForInBuildRuntime( - interpreter, files, coverageTool, coverageFiles, pythonVersion, stubShebang) + interpreter, + files, + coverageTool, + coverageFiles, + pythonVersion, + stubShebang, + bootstrapTemplate) : PyRuntimeInfo.createForPlatformRuntime( - interpreterPath, coverageTool, coverageFiles, pythonVersion, stubShebang); + interpreterPath, + coverageTool, + coverageFiles, + pythonVersion, + stubShebang, + bootstrapTemplate); return new RuleConfiguredTargetBuilder(ruleContext) .setFilesToBuild(files) diff --git a/src/main/java/com/google/devtools/build/lib/rules/python/PyRuntimeInfo.java b/src/main/java/com/google/devtools/build/lib/rules/python/PyRuntimeInfo.java index 5d812965b67075..a3d3be07baeb0e 100644 --- a/src/main/java/com/google/devtools/build/lib/rules/python/PyRuntimeInfo.java +++ b/src/main/java/com/google/devtools/build/lib/rules/python/PyRuntimeInfo.java @@ -62,6 +62,7 @@ public final class PyRuntimeInfo implements Info, PyRuntimeInfoApi { private final PythonVersion pythonVersion; private final String stubShebang; + @Nullable private final Artifact bootstrapTemplate; private PyRuntimeInfo( @Nullable Location location, @@ -71,7 +72,8 @@ private PyRuntimeInfo( @Nullable Artifact coverageTool, @Nullable Depset coverageFiles, PythonVersion pythonVersion, - @Nullable String stubShebang) { + @Nullable String stubShebang, + @Nullable Artifact bootstrapTemplate) { Preconditions.checkArgument((interpreterPath == null) != (interpreter == null)); Preconditions.checkArgument((interpreter == null) == (files == null)); Preconditions.checkArgument((coverageTool == null) == (coverageFiles == null)); @@ -88,6 +90,7 @@ private PyRuntimeInfo( } else { this.stubShebang = PyRuntimeInfoApi.DEFAULT_STUB_SHEBANG; } + this.bootstrapTemplate = bootstrapTemplate; } @Override @@ -107,16 +110,18 @@ public static PyRuntimeInfo createForInBuildRuntime( @Nullable Artifact coverageTool, @Nullable NestedSet coverageFiles, PythonVersion pythonVersion, - @Nullable String stubShebang) { + @Nullable String stubShebang, + @Nullable Artifact bootstrapTemplate) { return new PyRuntimeInfo( - /*location=*/ null, - /*interpreterPath=*/ null, + /* location= */ null, + /* interpreterPath= */ null, interpreter, Depset.of(Artifact.TYPE, files), coverageTool, coverageFiles == null ? null : Depset.of(Artifact.TYPE, coverageFiles), pythonVersion, - stubShebang); + stubShebang, + bootstrapTemplate); } /** Constructs an instance from native rule logic (built-in location) for a platform runtime. */ @@ -125,16 +130,18 @@ public static PyRuntimeInfo createForPlatformRuntime( @Nullable Artifact coverageTool, @Nullable NestedSet coverageFiles, PythonVersion pythonVersion, - @Nullable String stubShebang) { + @Nullable String stubShebang, + @Nullable Artifact bootstrapTemplate) { return new PyRuntimeInfo( - /*location=*/ null, + /* location= */ null, interpreterPath, - /*interpreter=*/ null, - /*files=*/ null, + /* interpreter= */ null, + /* files= */ null, coverageTool, coverageFiles == null ? null : Depset.of(Artifact.TYPE, coverageFiles), pythonVersion, - stubShebang); + stubShebang, + bootstrapTemplate); } @Override @@ -202,6 +209,12 @@ public String getStubShebang() { return stubShebang; } + @Override + @Nullable + public Artifact getBootstrapTemplate() { + return bootstrapTemplate; + } + @Nullable public NestedSet getFiles() { try { @@ -264,11 +277,16 @@ public PyRuntimeInfo constructor( Object coverageFilesUncast, String pythonVersion, String stubShebang, + Object bootstrapTemplateUncast, StarlarkThread thread) throws EvalException { String interpreterPath = interpreterPathUncast == NONE ? null : (String) interpreterPathUncast; Artifact interpreter = interpreterUncast == NONE ? null : (Artifact) interpreterUncast; + Artifact bootstrapTemplate = null; + if (bootstrapTemplateUncast != NONE) { + bootstrapTemplate = (Artifact) bootstrapTemplateUncast; + } Depset filesDepset = null; if (filesUncast != NONE) { // Validate type of filesDepset. @@ -306,23 +324,25 @@ public PyRuntimeInfo constructor( } return new PyRuntimeInfo( loc, - /*interpreterPath=*/ null, + /* interpreterPath= */ null, interpreter, filesDepset, coverageTool, coverageDepset, parsedPythonVersion, - stubShebang); + stubShebang, + bootstrapTemplate); } else { return new PyRuntimeInfo( loc, PathFragment.create(interpreterPath), - /*interpreter=*/ null, - /*files=*/ null, + /* interpreter= */ null, + /* files= */ null, coverageTool, coverageDepset, parsedPythonVersion, - stubShebang); + stubShebang, + bootstrapTemplate); } } } diff --git a/src/main/java/com/google/devtools/build/lib/rules/python/PyRuntimeRule.java b/src/main/java/com/google/devtools/build/lib/rules/python/PyRuntimeRule.java index 1ec90325df7965..46a5337219bd41 100644 --- a/src/main/java/com/google/devtools/build/lib/rules/python/PyRuntimeRule.java +++ b/src/main/java/com/google/devtools/build/lib/rules/python/PyRuntimeRule.java @@ -84,7 +84,7 @@

The entry point for the tool must be loadable by a python interpreter (e.g. a .allowedValues(PyRuleClasses.TARGET_PYTHON_ATTR_VALUE_SET)) /* - "Shebang" expression prepended to the bootstrapping Python stub script + "Shebang" expression prepended to the bootstrapping Python script used when executing py_binary targets.

See issue 8685 for @@ -93,6 +93,16 @@

The entry point for the tool must be loadable by a python interpreter (e.g. a

Does not apply to Windows. */ .add(attr("stub_shebang", STRING).value(PyRuntimeInfo.DEFAULT_STUB_SHEBANG)) + + /* + Previously referred to as the "Python stub script", this is the + entrypoint to every Python executable target. + */ + .add( + attr("bootstrap_template", LABEL) + .value(env.getToolsLabel(PyRuntimeInfo.DEFAULT_BOOTSTRAP_TEMPLATE)) + .allowedFileTypes(FileTypeSet.ANY_FILE) + .singleArtifact()) .add(attr("output_licenses", LICENSE)) .build(); } diff --git a/src/main/java/com/google/devtools/build/lib/starlarkbuildapi/python/PyRuntimeInfoApi.java b/src/main/java/com/google/devtools/build/lib/starlarkbuildapi/python/PyRuntimeInfoApi.java index 3aa47673d0ea3c..b10b47799400f3 100644 --- a/src/main/java/com/google/devtools/build/lib/starlarkbuildapi/python/PyRuntimeInfoApi.java +++ b/src/main/java/com/google/devtools/build/lib/starlarkbuildapi/python/PyRuntimeInfoApi.java @@ -46,6 +46,8 @@ public interface PyRuntimeInfoApi extends StarlarkValue { static final String DEFAULT_STUB_SHEBANG = "#!/usr/bin/env python3"; + // Must call getToolsLabel() when using this. + static final String DEFAULT_BOOTSTRAP_TEMPLATE = "//tools/python:python_bootstrap_template.txt"; @StarlarkMethod( name = "interpreter_path", @@ -119,6 +121,15 @@ public interface PyRuntimeInfoApi extends StarlarkValue { + "to Windows.") String getStubShebang(); + @StarlarkMethod( + name = "bootstrap_template", + structField = true, + doc = + "The stub script template file to use. Should have %python_binary%, " + + "%workspace_name%, %main%, and %imports%. See " + + "@bazel_tools//tools/python:python_bootstrap_template.txt for more variables.") + FileT getBootstrapTemplate(); + /** Provider type for {@link PyRuntimeInfoApi} objects. */ @StarlarkBuiltin(name = "Provider", documented = false, doc = "") interface PyRuntimeInfoProviderApi extends ProviderApi { @@ -204,6 +215,16 @@ interface PyRuntimeInfoProviderApi extends ProviderApi { + "Default is " + DEFAULT_STUB_SHEBANG + "."), + @Param( + name = "bootstrap_template", + allowedTypes = { + @ParamType(type = FileApi.class), + @ParamType(type = NoneType.class), + }, + positional = false, + named = true, + defaultValue = "None", + doc = ""), }, useStarlarkThread = true, selfCall = true) @@ -216,6 +237,7 @@ PyRuntimeInfoApi constructor( Object coverageFilesUncast, String pythonVersion, String stubShebang, + Object bootstrapTemplate, StarlarkThread thread) throws EvalException; } diff --git a/src/main/starlark/builtins_bzl/common/python/providers.bzl b/src/main/starlark/builtins_bzl/common/python/providers.bzl index 2e9bad6891ed42..813bc595f118fd 100644 --- a/src/main/starlark/builtins_bzl/common/python/providers.bzl +++ b/src/main/starlark/builtins_bzl/common/python/providers.bzl @@ -13,7 +13,10 @@ # limitations under the License. """Providers for Python rules.""" +load(":common/python/semantics.bzl", "TOOLS_REPO") + DEFAULT_STUB_SHEBANG = "#!/usr/bin/env python3" +DEFAULT_BOOTSTRAP_TEMPLATE = "@" + TOOLS_REPO + "//tools/python:python_bootstrap_template.txt" _PYTHON_VERSION_VALUES = ["PY2", "PY3"] def _PyRuntimeInfo_init( @@ -24,7 +27,8 @@ def _PyRuntimeInfo_init( coverage_tool = None, coverage_files = None, python_version, - stub_shebang = None): + stub_shebang = None, + bootstrap_template = None): if (interpreter_path == None) == (interpreter == None): fail("exactly one of interpreter_path or interpreter must be set") if (interpreter == None) != (files == None): @@ -46,6 +50,7 @@ def _PyRuntimeInfo_init( "coverage_files": coverage_files, "python_version": python_version, "stub_shebang": stub_shebang, + "bootstrap_template": bootstrap_template, } # TODO(#15897): Rename this to PyRuntimeInfo when we're ready to replace the Java @@ -98,6 +103,9 @@ the same conventions as the standard CPython interpreter. "script used when executing `py_binary` targets. Does not " + "apply to Windows." ), + "bootstrap_template": ( + "See py_runtime_rule.bzl%py_runtime.bootstrap_template for docs." + ), }, ) diff --git a/src/main/starlark/builtins_bzl/common/python/py_runtime_rule.bzl b/src/main/starlark/builtins_bzl/common/python/py_runtime_rule.bzl index ec293019d56513..e50103b0b12475 100644 --- a/src/main/starlark/builtins_bzl/common/python/py_runtime_rule.bzl +++ b/src/main/starlark/builtins_bzl/common/python/py_runtime_rule.bzl @@ -14,7 +14,7 @@ """Implementation of py_runtime rule.""" load(":common/paths.bzl", "paths") -load(":common/python/providers.bzl", "DEFAULT_STUB_SHEBANG", _PyRuntimeInfo = "PyRuntimeInfo") +load(":common/python/providers.bzl", "DEFAULT_BOOTSTRAP_TEMPLATE", "DEFAULT_STUB_SHEBANG", _PyRuntimeInfo = "PyRuntimeInfo") _py_builtins = _builtins.internal.py_builtins @@ -75,6 +75,7 @@ def _py_runtime_impl(ctx): coverage_files = coverage_files, python_version = python_version, stub_shebang = ctx.attr.stub_shebang, + bootstrap_template = ctx.file.bootstrap_template, ), DefaultInfo( files = runtime_files, @@ -177,6 +178,28 @@ See https://github.com/bazelbuild/bazel/issues/8685 for motivation. Does not apply to Windows. +""", + ), + "bootstrap_template": attr.label( + allow_single_file = True, + default = DEFAULT_BOOTSTRAP_TEMPLATE, + doc = """ +The bootstrap script template file to use. Should have %python_binary%, +%workspace_name%, %main%, and %imports%. + +This template, after expansion, becomes the executable file used to start the +process, so it is responsible for initial bootstrapping actions such as finding +the Python interpreter, runfiles, and constructing an environment to run the +intended Python application. + +While this attribute is currently optional, it will become required when the +Python rules are moved out of Bazel itself. + +The exact variable names expanded is an unstable API and is subject to change. +The API will become more stable when the Python rules are moved out of Bazel +itself. + +See @bazel_tools//tools/python:python_bootstrap_template.txt for more variables. """, ), }, diff --git a/src/test/java/com/google/devtools/build/lib/packages/util/BazelMockPythonSupport.java b/src/test/java/com/google/devtools/build/lib/packages/util/BazelMockPythonSupport.java index a95f2464a0c25e..ab4e3ef8a4559c 100644 --- a/src/test/java/com/google/devtools/build/lib/packages/util/BazelMockPythonSupport.java +++ b/src/test/java/com/google/devtools/build/lib/packages/util/BazelMockPythonSupport.java @@ -39,6 +39,7 @@ public void setup(MockToolsConfig config) throws IOException { addTool(config, "tools/python/toolchain.bzl"); addTool(config, "tools/python/utils.bzl"); addTool(config, "tools/python/private/defs.bzl"); + addTool(config, "tools/python/python_bootstrap_template.txt"); config.create( TestConstants.TOOLS_REPOSITORY_SCRATCH + "tools/python/BUILD", diff --git a/src/test/java/com/google/devtools/build/lib/rules/python/PyRuntimeInfoTest.java b/src/test/java/com/google/devtools/build/lib/rules/python/PyRuntimeInfoTest.java index 3d51090fbb171c..d93cdf6e954a9b 100644 --- a/src/test/java/com/google/devtools/build/lib/rules/python/PyRuntimeInfoTest.java +++ b/src/test/java/com/google/devtools/build/lib/rules/python/PyRuntimeInfoTest.java @@ -59,7 +59,7 @@ public void factoryMethod_InBuildRuntime() throws Exception { NestedSet files = NestedSetBuilder.create(Order.STABLE_ORDER, dummyFile); PyRuntimeInfo inBuildRuntime = PyRuntimeInfo.createForInBuildRuntime( - dummyInterpreter, files, null, null, PythonVersion.PY2, null); + dummyInterpreter, files, null, null, PythonVersion.PY2, null, dummyFile); assertThat(inBuildRuntime.getCreationLocation()).isEqualTo(Location.BUILTIN); assertThat(inBuildRuntime.getInterpreterPath()).isNull(); @@ -76,7 +76,8 @@ public void factoryMethod_InBuildRuntime() throws Exception { public void factoryMethod_PlatformRuntime() { PathFragment path = PathFragment.create("/system/interpreter"); PyRuntimeInfo platformRuntime = - PyRuntimeInfo.createForPlatformRuntime(path, null, null, PythonVersion.PY2, null); + PyRuntimeInfo.createForPlatformRuntime( + path, null, null, PythonVersion.PY2, null, dummyFile); assertThat(platformRuntime.getCreationLocation()).isEqualTo(Location.BUILTIN); assertThat(platformRuntime.getInterpreterPath()).isEqualTo(path); @@ -96,6 +97,7 @@ public void starlarkConstructor_InBuildRuntime() throws Exception { " interpreter = dummy_interpreter,", " files = depset([dummy_file]),", " python_version = 'PY2',", + " bootstrap_template = dummy_file,", ")"); PyRuntimeInfo info = (PyRuntimeInfo) ev.lookup("info"); assertThat(info.getCreationLocation().toString()).isEqualTo(":1:21"); @@ -104,6 +106,7 @@ public void starlarkConstructor_InBuildRuntime() throws Exception { assertHasOrderAndContainsExactly(info.getFiles(), Order.STABLE_ORDER, dummyFile); assertThat(info.getPythonVersion()).isEqualTo(PythonVersion.PY2); assertThat(info.getStubShebang()).isEqualTo(PyRuntimeInfo.DEFAULT_STUB_SHEBANG); + assertThat(info.getBootstrapTemplate()).isEqualTo(dummyFile); } @Test @@ -112,6 +115,7 @@ public void starlarkConstructor_PlatformRuntime() throws Exception { "info = PyRuntimeInfo(", // " interpreter_path = '/system/interpreter',", " python_version = 'PY2',", + " bootstrap_template = dummy_file,", ")"); PyRuntimeInfo info = (PyRuntimeInfo) ev.lookup("info"); assertThat(info.getCreationLocation().toString()).isEqualTo(":1:21"); @@ -129,6 +133,7 @@ public void starlarkConstructor_CustomShebang() throws Exception { " interpreter_path = '/system/interpreter',", " python_version = 'PY2',", " stub_shebang = '#!/usr/bin/custom',", + " bootstrap_template = dummy_file,", ")"); PyRuntimeInfo info = (PyRuntimeInfo) ev.lookup("info"); assertThat(info.getStubShebang()).isEqualTo("#!/usr/bin/custom"); @@ -140,6 +145,7 @@ public void starlarkConstructor_FilesDefaultsToEmpty() throws Exception { "info = PyRuntimeInfo(", // " interpreter = dummy_interpreter,", " python_version = 'PY2',", + " bootstrap_template = dummy_file,", ")"); PyRuntimeInfo info = (PyRuntimeInfo) ev.lookup("info"); assertHasOrderAndContainsExactly(info.getFiles(), Order.STABLE_ORDER); diff --git a/tools/python/BUILD.tools b/tools/python/BUILD.tools index 5031ee9fe14390..076c182914c34f 100644 --- a/tools/python/BUILD.tools +++ b/tools/python/BUILD.tools @@ -24,6 +24,7 @@ package(default_visibility = ["//visibility:public"]) # don't have access to Skylib here. See # https://github.com/bazelbuild/skydoc/issues/166. exports_files([ + "python_bootstrap_template.txt", "python_version.bzl", "srcs_version.bzl", "toolchain.bzl", diff --git a/src/main/java/com/google/devtools/build/lib/bazel/rules/python/python_stub_template.txt b/tools/python/python_bootstrap_template.txt similarity index 100% rename from src/main/java/com/google/devtools/build/lib/bazel/rules/python/python_stub_template.txt rename to tools/python/python_bootstrap_template.txt