Skip to content

Commit

Permalink
Restore (limited) dynamic feature capability + introduce sample app
Browse files Browse the repository at this point in the history
  • Loading branch information
oliviernotteghem committed Oct 31, 2024
1 parent 4d719e1 commit 8452f18
Show file tree
Hide file tree
Showing 32 changed files with 359 additions and 14 deletions.
Binary file added examples/bundle/.DS_Store
Binary file not shown.
10 changes: 10 additions & 0 deletions examples/bundle/.bazelrc
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
# Flags needed while the Android rules are being migrated to Starlark.
common --experimental_google_legacy_api
common --experimental_enable_android_migration_apis
common --android_sdk=@androidsdk//:sdk
common:core_library_desugaring --desugar_java8_libs

# Flags to enable mobile-install v3
mobile-install --mode=skylark --mobile_install_aspect=@rules_android//mobile_install:mi.bzl --mobile_install_supported_rules=android_binary
# Required to invoke the Studio deployer jar
mobile-install --tool_java_runtime_version=17
1 change: 1 addition & 0 deletions examples/bundle/.gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
*.lock
1 change: 1 addition & 0 deletions examples/bundle/BUILD
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
# Empty build file to satisfy gazelle for rules_go.
55 changes: 55 additions & 0 deletions examples/bundle/MODULE.bazel
Original file line number Diff line number Diff line change
@@ -0,0 +1,55 @@
module(
name = "bundle",
)

bazel_dep(name = "rules_java", version = "7.4.0")
bazel_dep(name = "bazel_skylib", version = "1.3.0")

bazel_dep(
name = "rules_android",
version = "0.5.1",
)

local_path_override(
module_name = "rules_android",
path = "../../",
)

remote_android_extensions = use_extension(
"@rules_android//bzlmod_extensions:android_extensions.bzl",
"remote_android_tools_extensions")
use_repo(remote_android_extensions, "android_gmaven_r8", "android_tools")

register_toolchains(
"@rules_android//toolchains/android:android_default_toolchain",
"@rules_android//toolchains/android_sdk:android_sdk_tools",
)

android_sdk_repository_extension = use_extension("@rules_android//rules/android_sdk_repository:rule.bzl", "android_sdk_repository_extension")
use_repo(android_sdk_repository_extension, "androidsdk")

register_toolchains("@androidsdk//:sdk-toolchain", "@androidsdk//:all")

bazel_dep(name = "rules_jvm_external", version = "5.3")

# Load the maven extension from rules_jvm_external
maven = use_extension("@rules_jvm_external//:extensions.bzl", "maven")

maven.install(
name = "maven",
aar_import_bzl_label = "@rules_android//rules:rules.bzl",
artifacts = [
"com.google.guava:guava:32.1.2-android",
"com.arthenica:ffmpeg-kit-https:4.4.LTS",
],
repositories = [
"https://maven.google.com",
"https://repo1.maven.org/maven2",
],
use_starlark_android_rules = True,
)
use_repo(maven, "maven")




12 changes: 12 additions & 0 deletions examples/bundle/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
To build, ensure that the `ANDROID_HOME` environment variable is set to the path
to an Android SDK, and run:

```
bazel build app:assets
```

This will build application bundle containing a dynamic feature containing assets (named assets.txt). Verify with :

```
jar -tf bazel-bin/app/assets_unsigned.aab | grep assets.txt
```
58 changes: 58 additions & 0 deletions examples/bundle/WORKSPACE
Original file line number Diff line number Diff line change
@@ -0,0 +1,58 @@
load("@bazel_tools//tools/build_defs/repo:http.bzl", "http_archive")
load("@bazel_tools//tools/build_defs/repo:utils.bzl", "maybe")

maybe(
http_archive,
name = "rules_jvm_external",
strip_prefix = "rules_jvm_external-fa73b1a8e4846cee88240d0019b8f80d39feb1c3",
sha256 = "7e13e48b50f9505e8a99cc5a16c557cbe826e9b68d733050cd1e318d69f94bb5",
url = "https://github.com/bazelbuild/rules_jvm_external/archive/fa73b1a8e4846cee88240d0019b8f80d39feb1c3.zip",
)

maybe(
http_archive,
name = "bazel_skylib",
urls = [
"https://github.com/bazelbuild/bazel-skylib/releases/download/1.0.3/bazel-skylib-1.0.3.tar.gz",
"https://mirror.bazel.build/github.com/bazelbuild/bazel-skylib/releases/download/1.0.3/bazel-skylib-1.0.3.tar.gz",
],
sha256 = "1c531376ac7e5a180e0237938a2536de0c54d93f5c278634818e0efc952dd56c",
)
load("@bazel_skylib//:workspace.bzl", "bazel_skylib_workspace")
bazel_skylib_workspace()

local_repository(
name = "rules_android",
path = "../..", # rules_android's WORKSPACE relative to this inner workspace
)

load("@rules_android//:prereqs.bzl", "rules_android_prereqs")
rules_android_prereqs()
load("@rules_android//:defs.bzl", "rules_android_workspace")
rules_android_workspace()

load("@rules_android//rules:rules.bzl", "android_sdk_repository")
android_sdk_repository(
name = "androidsdk",
)

register_toolchains(
"@rules_android//toolchains/android:android_default_toolchain",
"@rules_android//toolchains/android_sdk:android_sdk_tools",
)

load("@rules_jvm_external//:defs.bzl", "maven_install")

maven_install(
name = "maven",
aar_import_bzl_label = "@rules_android//rules:rules.bzl",
artifacts = [
"com.google.guava:guava:32.1.2-android",
"com.arthenica:ffmpeg-kit-https:4.4.LTS",
],
repositories = [
"https://maven.google.com",
"https://repo1.maven.org/maven2",
],
use_starlark_android_rules = True,
)
1 change: 1 addition & 0 deletions examples/bundle/WORKSPACE.bzlmod
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
workspace(name = "bundle")
Binary file added examples/bundle/app/.DS_Store
Binary file not shown.
22 changes: 22 additions & 0 deletions examples/bundle/app/AndroidManifest.xml
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.examples.bundle.app" >

<uses-sdk
android:minSdkVersion="21"
android:targetSdkVersion="30" />

<application
android:allowBackup="true"
android:icon="@drawable/ic_launcher"
android:label="@string/app_name" >
<activity
android:name="com.examples.bundle.app.BasicActivity"
android:exported="true">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
</manifest>
31 changes: 31 additions & 0 deletions examples/bundle/app/BUILD
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
load("@rules_android//android:rules.bzl", "android_application", "android_library")

android_application(
name = "assets",
manifest_values = {
"applicationId" : "com.examples.bundle.app",
"versionCode": "0",
},
feature_modules = ["//features/assets:feature_module"],
manifest = "AndroidManifest.xml",
deps = [":lib"],
)

android_application(
name = "native",
manifest_values = {
"applicationId" : "com.examples.bundle.app",
"versionCode": "0",
},
feature_modules = ["//features/native:feature_module"],
manifest = "AndroidManifest.xml",
deps = [":lib"],
)

android_library(
name = "lib",
srcs = ["BasicActivity.java"],
manifest = "AndroidManifest.xml",
resource_files = glob(["res/**"]),
deps = ["@maven//:com_google_guava_guava",]
)
59 changes: 59 additions & 0 deletions examples/bundle/app/BasicActivity.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,59 @@
// Copyright 2022 The Bazel Authors. 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 com.examples.bundle.app;

import android.app.Activity;
import android.os.Bundle;
import android.view.Menu;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;

/**
* The main activity of the Basic Sample App.
*/
public class BasicActivity extends Activity {

@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.basic_activity);

final Button buttons[] = {
findViewById(R.id.button_id_fizz), findViewById(R.id.button_id_buzz),
};

for (Button b : buttons) {
b.setOnClickListener(
new View.OnClickListener() {
public void onClick(View v) {
TextView tv = findViewById(R.id.text_hello);
if (v.getId() == R.id.button_id_fizz) {
tv.setText("fizz");
} else if (v.getId() == R.id.button_id_buzz) {
tv.setText("buzz ");
}
}
});
}
}

@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.menu, menu);
return true;
}
}
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
23 changes: 23 additions & 0 deletions examples/bundle/app/res/layout/basic_activity.xml
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical" >

<TextView
android:id="@+id/text_hello"
android:text="@string/hello_world"
android:layout_width="wrap_content"
android:layout_height="wrap_content" />

<Button
android:id="@+id/button_id_fizz"
android:layout_height="wrap_content"
android:layout_width="wrap_content"
android:text="fizz" />
<Button
android:id="@+id/button_id_buzz"
android:layout_height="wrap_content"
android:layout_width="wrap_content"
android:text="buzz" />

</LinearLayout>
8 changes: 8 additions & 0 deletions examples/bundle/app/res/menu/menu.xml
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
<menu xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
tools:context="com.examples.bundle.app.BasicActivity" >
<item android:id="@+id/action_settings"
android:title="@string/action_settings"
android:orderInCategory="100" />
</menu>
5 changes: 5 additions & 0 deletions examples/bundle/app/res/values/dimens.xml
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
<resources>
<!-- Default screen margins, per the Android Design guidelines. -->
<dimen name="activity_horizontal_margin">16dp</dimen>
<dimen name="activity_vertical_margin">16dp</dimen>
</resources>
8 changes: 8 additions & 0 deletions examples/bundle/app/res/values/strings.xml
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
<?xml version="1.0" encoding="utf-8"?>
<resources>

<string name="app_name" translatable="false">basicbundle</string>
<string name="hello_world" translatable="false">Hello world!</string>
<string name="action_settings" translatable="false">Settings</string>

</resources>
18 changes: 18 additions & 0 deletions examples/bundle/features/assets/BUILD
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
load("@rules_android//android:rules.bzl", "android_library")
load("@rules_android//rules:rules.bzl", "android_feature_module")

android_library(
name = "lib",
manifest = "src/AndroidManifest.xml",
assets = ["src/assets.txt"],
)

android_feature_module(
name = "feature_module",
custom_package = "com.example.bundle.features.assets",
manifest = "src/AndroidManifest.xml",
title = "asset_feature",
library = ":lib",
feature_name = "asset_feature",
visibility = ["//visibility:public"],
)
Binary file added examples/bundle/features/assets/src/.DS_Store
Binary file not shown.
13 changes: 13 additions & 0 deletions examples/bundle/features/assets/src/AndroidManifest.xml
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:dist="http://schemas.android.com/apk/distribution"
package="com.examples.bundle.features.assets">

<dist:module
dist:onDemand="true"
dist:instant="false"
dist:title="${MODULE_TITLE}">
<dist:fusing dist:include="true" />
</dist:module>

</manifest>
1 change: 1 addition & 0 deletions examples/bundle/features/assets/src/assets.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
This text originates from a dynamically loaded feature.
7 changes: 0 additions & 7 deletions rules/android_application/BUILD
Original file line number Diff line number Diff line change
Expand Up @@ -41,10 +41,3 @@ py_binary(
"@py_absl//absl/flags",
],
)

filegroup(
name = "merge_feature_manifests.par",
srcs = [":merge_feature_manifests"],
output_group = "python_zip_file",
visibility = ["//visibility:public"],
)
8 changes: 4 additions & 4 deletions rules/android_application/android_feature_module_rule.bzl
Original file line number Diff line number Diff line change
Expand Up @@ -94,10 +94,10 @@ def get_feature_module_paths(fqn):
# Given a fqn to an android_feature_module, returns the absolute paths to
# all implicitly generated targets
return struct(
binary = Label("%s_bin" % fqn),
manifest_lib = Label("%s_AndroidManifest" % fqn),
title_strings_xml = Label("%s_title_strings_xml" % fqn),
title_lib = Label("%s_title_lib" % fqn),
binary = native.package_relative_label("%s_bin" % fqn),
manifest_lib = native.package_relative_label("%s_AndroidManifest" % fqn),
title_strings_xml = native.package_relative_label("%s_title_strings_xml" % fqn),
title_lib = native.package_relative_label("%s_title_lib" % fqn),
)

def android_feature_module_macro(_android_binary, _android_library, **attrs):
Expand Down
6 changes: 3 additions & 3 deletions rules/android_application/attrs.bzl
Original file line number Diff line number Diff line change
Expand Up @@ -67,7 +67,7 @@ ANDROID_APPLICATION_ATTRS = _attrs.add(
),
_bundle_keystore_properties = attr.label(
allow_single_file = True,
default = "//rules:bundle_keystore_properties.tmpl",
default = None,
),
_feature_manifest_script = attr.label(
allow_single_file = True,
Expand All @@ -79,8 +79,7 @@ ANDROID_APPLICATION_ATTRS = _attrs.add(
default = Label("//tools/jdk:toolchain_android_only"),
),
_merge_manifests = attr.label(
default = ":merge_feature_manifests.par",
allow_single_file = True,
default = ":merge_feature_manifests",
cfg = "exec",
executable = True,
),
Expand Down Expand Up @@ -114,6 +113,7 @@ ANDROID_FEATURE_MODULE_ATTRS = dict(
manifest = attr.label(allow_single_file = True),
title_id = attr.string(),
title_lib = attr.string(),
fused = attr.bool(),
_feature_module_validation_script = attr.label(
allow_single_file = True,
cfg = "exec",
Expand Down
Empty file modified rules/android_application/feature_module_validation.sh
100644 → 100755
Empty file.
Empty file.
Loading

0 comments on commit 8452f18

Please sign in to comment.