| // Copyright 2023 Google Inc. All rights reserved. |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| package python |
| |
| import ( |
| "path/filepath" |
| "strings" |
| |
| "github.com/google/blueprint/proptools" |
| |
| "android/soong/android" |
| "android/soong/bazel" |
| ) |
| |
| type bazelPythonLibraryAttributes struct { |
| Srcs bazel.LabelListAttribute |
| Deps bazel.LabelListAttribute |
| Imports bazel.StringListAttribute |
| Srcs_version *string |
| } |
| |
| type bazelPythonProtoLibraryAttributes struct { |
| Deps bazel.LabelListAttribute |
| |
| // A list of proto_library targets that the proto_library in `deps` depends on |
| // This list is overestimation. |
| // Overestimation is necessary since Soong includes other protos via proto.include_dirs and not |
| // a specific .proto file module explicitly. |
| Transitive_deps bazel.LabelListAttribute |
| } |
| |
| type baseAttributes struct { |
| // TODO(b/200311466): Probably not translate b/c Bazel has no good equiv |
| //Pkg_path bazel.StringAttribute |
| // TODO: Related to Pkg_bath and similarLy gated |
| //Is_internal bazel.BoolAttribute |
| // Combines Srcs and Exclude_srcs |
| Srcs bazel.LabelListAttribute |
| Deps bazel.LabelListAttribute |
| // Combines Data and Java_data (invariant) |
| Data bazel.LabelListAttribute |
| Imports bazel.StringListAttribute |
| } |
| |
| func (m *PythonLibraryModule) makeArchVariantBaseAttributes(ctx android.TopDownMutatorContext) baseAttributes { |
| var attrs baseAttributes |
| archVariantBaseProps := m.GetArchVariantProperties(ctx, &BaseProperties{}) |
| for axis, configToProps := range archVariantBaseProps { |
| for config, props := range configToProps { |
| if baseProps, ok := props.(*BaseProperties); ok { |
| attrs.Srcs.SetSelectValue(axis, config, |
| android.BazelLabelForModuleSrcExcludes(ctx, baseProps.Srcs, baseProps.Exclude_srcs)) |
| attrs.Deps.SetSelectValue(axis, config, |
| android.BazelLabelForModuleDeps(ctx, baseProps.Libs)) |
| data := android.BazelLabelForModuleSrc(ctx, baseProps.Data) |
| data.Append(android.BazelLabelForModuleSrc(ctx, baseProps.Java_data)) |
| attrs.Data.SetSelectValue(axis, config, data) |
| } |
| } |
| } |
| |
| partitionedSrcs := bazel.PartitionLabelListAttribute(ctx, &attrs.Srcs, bazel.LabelPartitions{ |
| "proto": android.ProtoSrcLabelPartition, |
| "py": bazel.LabelPartition{Keep_remainder: true}, |
| }) |
| attrs.Srcs = partitionedSrcs["py"] |
| |
| if !partitionedSrcs["proto"].IsEmpty() { |
| protoInfo, _ := android.Bp2buildProtoProperties(ctx, &m.ModuleBase, partitionedSrcs["proto"]) |
| |
| pyProtoLibraryName := m.Name() + "_py_proto" |
| ctx.CreateBazelTargetModule(bazel.BazelTargetModuleProperties{ |
| Rule_class: "py_proto_library", |
| Bzl_load_location: "//build/bazel/rules/python:py_proto.bzl", |
| }, android.CommonAttributes{ |
| Name: pyProtoLibraryName, |
| }, &bazelPythonProtoLibraryAttributes{ |
| Deps: bazel.MakeLabelListAttribute(protoInfo.Proto_libs), |
| Transitive_deps: bazel.MakeLabelListAttribute(protoInfo.Transitive_proto_libs), |
| }) |
| |
| attrs.Deps.Add(bazel.MakeLabelAttribute(":" + pyProtoLibraryName)) |
| } |
| |
| // Bazel normally requires `import path.from.top.of.tree` statements in |
| // python code, but with soong you can directly import modules from libraries. |
| // Add "imports" attributes to the bazel library so it matches soong's behavior. |
| imports := "." |
| if m.properties.Pkg_path != nil { |
| // TODO(b/215119317) This is a hack to handle the fact that we don't convert |
| // pkg_path properly right now. If the folder structure that contains this |
| // Android.bp file matches pkg_path, we can set imports to an appropriate |
| // number of ../..s to emulate moving the files under a pkg_path folder. |
| pkg_path := filepath.Clean(*m.properties.Pkg_path) |
| if strings.HasPrefix(pkg_path, "/") { |
| ctx.ModuleErrorf("pkg_path cannot start with a /: %s", pkg_path) |
| } |
| |
| if !strings.HasSuffix(ctx.ModuleDir(), "/"+pkg_path) && ctx.ModuleDir() != pkg_path { |
| ctx.ModuleErrorf("Currently, bp2build only supports pkg_paths that are the same as the folders the Android.bp file is in. pkg_path: %s, module directory: %s", pkg_path, ctx.ModuleDir()) |
| } |
| numFolders := strings.Count(pkg_path, "/") + 1 |
| dots := make([]string, numFolders) |
| for i := 0; i < numFolders; i++ { |
| dots[i] = ".." |
| } |
| imports = strings.Join(dots, "/") |
| } |
| attrs.Imports = bazel.MakeStringListAttribute([]string{imports}) |
| |
| return attrs |
| } |
| |
| func (m *PythonLibraryModule) bp2buildPythonVersion(ctx android.TopDownMutatorContext) *string { |
| py3Enabled := proptools.BoolDefault(m.properties.Version.Py3.Enabled, true) |
| py2Enabled := proptools.BoolDefault(m.properties.Version.Py2.Enabled, false) |
| if py2Enabled && !py3Enabled { |
| return &pyVersion2 |
| } else if !py2Enabled && py3Enabled { |
| return &pyVersion3 |
| } else if !py2Enabled && !py3Enabled { |
| ctx.ModuleErrorf("bp2build converter doesn't understand having neither py2 nor py3 enabled") |
| return &pyVersion3 |
| } else { |
| return &pyVersion2And3 |
| } |
| } |
| |
| type bazelPythonBinaryAttributes struct { |
| Main *bazel.Label |
| Srcs bazel.LabelListAttribute |
| Deps bazel.LabelListAttribute |
| Python_version *string |
| Imports bazel.StringListAttribute |
| } |
| |
| func (p *PythonLibraryModule) ConvertWithBp2build(ctx android.TopDownMutatorContext) { |
| // TODO(b/182306917): this doesn't fully handle all nested props versioned |
| // by the python version, which would have been handled by the version split |
| // mutator. This is sufficient for very simple python_library modules under |
| // Bionic. |
| baseAttrs := p.makeArchVariantBaseAttributes(ctx) |
| pyVersion := p.bp2buildPythonVersion(ctx) |
| if *pyVersion == pyVersion2And3 { |
| // Libraries default to python 2 and 3 |
| pyVersion = nil |
| } |
| |
| attrs := &bazelPythonLibraryAttributes{ |
| Srcs: baseAttrs.Srcs, |
| Deps: baseAttrs.Deps, |
| Srcs_version: pyVersion, |
| Imports: baseAttrs.Imports, |
| } |
| |
| props := bazel.BazelTargetModuleProperties{ |
| // Use the native py_library rule. |
| Rule_class: "py_library", |
| } |
| |
| ctx.CreateBazelTargetModule(props, android.CommonAttributes{ |
| Name: p.Name(), |
| Data: baseAttrs.Data, |
| }, attrs) |
| } |
| |
| func (p *PythonBinaryModule) bp2buildBinaryProperties(ctx android.TopDownMutatorContext) (*bazelPythonBinaryAttributes, bazel.LabelListAttribute) { |
| // TODO(b/182306917): this doesn't fully handle all nested props versioned |
| // by the python version, which would have been handled by the version split |
| // mutator. This is sufficient for very simple python_binary_host modules |
| // under Bionic. |
| |
| baseAttrs := p.makeArchVariantBaseAttributes(ctx) |
| pyVersion := p.bp2buildPythonVersion(ctx) |
| if *pyVersion == pyVersion3 { |
| // Binaries default to python 3 |
| pyVersion = nil |
| } else if *pyVersion == pyVersion2And3 { |
| ctx.ModuleErrorf("error for '%s' module: bp2build's python_binary_host converter "+ |
| "does not support converting a module that is enabled for both Python 2 and 3 at the "+ |
| "same time.", p.Name()) |
| } |
| |
| attrs := &bazelPythonBinaryAttributes{ |
| Main: nil, |
| Srcs: baseAttrs.Srcs, |
| Deps: baseAttrs.Deps, |
| Python_version: pyVersion, |
| Imports: baseAttrs.Imports, |
| } |
| |
| // main is optional. |
| if p.binaryProperties.Main != nil { |
| main := android.BazelLabelForModuleSrcSingle(ctx, *p.binaryProperties.Main) |
| attrs.Main = &main |
| } |
| return attrs, baseAttrs.Data |
| } |
| |
| func (p *PythonBinaryModule) ConvertWithBp2build(ctx android.TopDownMutatorContext) { |
| attrs, data := p.bp2buildBinaryProperties(ctx) |
| |
| props := bazel.BazelTargetModuleProperties{ |
| // Use the native py_binary rule. |
| Rule_class: "py_binary", |
| } |
| |
| ctx.CreateBazelTargetModule(props, android.CommonAttributes{ |
| Name: p.Name(), |
| Data: data, |
| }, attrs) |
| } |
| |
| func (p *PythonTestModule) ConvertWithBp2build(ctx android.TopDownMutatorContext) { |
| // Python tests are currently exactly the same as binaries, but with a different module type |
| attrs, data := p.bp2buildBinaryProperties(ctx) |
| |
| props := bazel.BazelTargetModuleProperties{ |
| // Use the native py_binary rule. |
| Rule_class: "py_test", |
| Bzl_load_location: "//build/bazel/rules/python:py_test.bzl", |
| } |
| |
| ctx.CreateBazelTargetModule(props, android.CommonAttributes{ |
| Name: p.Name(), |
| Data: data, |
| }, attrs) |
| } |