Android Kotlin Projects
This page provides an example of using Mill as a build tool for Android applications. This workflow is still pretty rough and nowhere near production ready, but can serve as a starting point for further experimentation and development.
Relevant Modules
These are the main Mill Modules that are relevant for building Android apps:
-
mill.javalib.android.AndroidSdkModule
: Handles Android SDK management and tools. -
mill.kotlinlib.android.AndroidAppKotlinModule
: Provides a framework for building Android applications. -
mill.kotlinlib.KotlinModule
: General Kotlin build tasks like compiling Kotlin code and creating JAR files.
Simple Android Hello World Application
This section sets up a basic Android project using Mill.
We utilize AndroidAppKotlinModule
and AndroidSdkModule
to streamline the process of
building an Android application with minimal configuration.
By extending AndroidAppKotlinModule
, we inherit all Android-related tasks such as
resource generation, APK building, DEX conversion, and APK signing.
Additionally, AndroidSdkModule
is embedded, making SDK management seamless.
package build
import mill._
import kotlinlib._
import mill.kotlinlib.android.AndroidAppKotlinModule
import mill.javalib.android.AndroidTestModule
import mill.javalib.android.AndroidSdkModule
import coursier.maven.MavenRepository
Create and configure an Android SDK module to manage Android SDK paths and tools.
object androidSdkModule0 extends AndroidSdkModule {
def buildToolsVersion = "35.0.0"
def bundleToolVersion = "1.17.2"
}
Actual android application
object app extends AndroidAppKotlinModule {
def kotlinVersion = "2.0.20"
def androidSdkModule = mill.define.ModuleRef(androidSdkModule0)
// Configuration for ReleaseKey
override def releaseKeyPath = millSourcePath
def androidReleaseKeyName: T[String] = Task { "releaseKey.jks" }
def androidReleaseKeyAlias: T[String] = Task { "releaseKey" }
def androidReleaseKeyPass: T[String] = Task { "MillBuildTool" }
def androidReleaseKeyStorePass: T[String] = Task { "MillBuildTool" }
override def androidVirtualDeviceIdentifier: String = "kotlin-test"
override def androidEmulatorPort: String = "5556"
/* TODO this won't be needed once the debug keystore is implemented */
private def mainRoot = millSourcePath
object test extends AndroidAppKotlinTests with TestModule.Junit4 {
def ivyDeps = super.ivyDeps() ++ Agg(
ivy"junit:junit:4.13.2"
)
}
object it extends AndroidAppKotlinIntegrationTests with AndroidTestModule.AndroidJUnit {
override def instrumentationPackage = "com.helloworld.app"
/* TODO this needs to change to something better once integration tests work with debug keystore */
override def releaseKeyPath = mainRoot
def androidReleaseKeyName: T[String] = Task { "releaseKey.jks" }
def androidReleaseKeyAlias: T[String] = Task { "releaseKey" }
def androidReleaseKeyPass: T[String] = Task { "MillBuildTool" }
def androidReleaseKeyStorePass: T[String] = Task { "MillBuildTool" }
/* TODO currently the dependency resolution ignores the platform type and kotlinx-coroutines-core has
* conflicting classes with kotlinx-coroutines-core-jvm . Remove the exclusions once the dependency
* resolution resolves conflicts between androidJvm and jvm platform types
*/
def ivyDeps = super.ivyDeps() ++ Agg(
ivy"androidx.test.ext:junit:1.2.1".exclude((
"org.jetbrains.kotlinx",
"kotlinx-coroutines-core-jvm"
)),
ivy"androidx.test:runner:1.6.2",
ivy"androidx.test.espresso:espresso-core:3.5.1".exclude((
"org.jetbrains.kotlinx",
"kotlinx-coroutines-core-jvm"
)),
ivy"junit:junit:4.13.2"
)
}
}
> ./mill show app.androidApk
".../out/app/androidApk.dest/app.apk"
This command triggers the build process, which installs the Android Setup, compiles the kotlin
code, generates Android resources, converts kotlin bytecode to DEX format, packages everything
into an APK, optimizes the APK using zipalign
, and finally signs it.
This Mill build configuration is designed to build a simple "Hello World" Android application.
By extending AndroidAppKotlinModule
, we leverage its predefined Android build tasks, ensuring that
all necessary steps (resource generation, APK creation, and signing) are executed automatically.
Project Structure:
The project follows the standard Android app layout. Below is a typical project folder structure:
. ├── app │ └── src │ ├── androidTest/java/com/helloworld/app/ExampleInstrumentedTest.kt │ ├── main │ │ ├── AndroidManifest.xml │ │ ├── java/com/helloworld/app/MainActivity.kt │ │ └── res │ │ └── values │ │ ├── colors.xml │ │ └── strings.xml │ └── test/java/com/helloworld/app/ExampleUnitTest.kt └── build.mill
> ./mill show app.test
...Compiling 5 Kotlin sources...
> cat out/app/test/test.dest/out.json
["",[{"fullyQualifiedName":"com.helloworld.ExampleUnitTest.text_size_is_correct","selector":"com.helloworld.ExampleUnitTest.text_size_is_correct","duration":...,"status":"Success"},{"fullyQualifiedName":"com.helloworld.ExampleUnitTestInKotlinDir.kotlin_dir_text_size_is_correct","selector":"com.helloworld.ExampleUnitTestInKotlinDir.kotlin_dir_text_size_is_correct","duration":...,"status":"Success"}]]
This command runs unit tests on your local environment.
> ./mill app.createAndroidVirtualDevice
> sleep 60 && ./mill show app.startAndroidEmulator
> ./mill show app.waitForDevice
...emulator-5556...
> ./mill show app.it | grep '"OK (1 test)"'
..."OK (1 test)",
> cat out/app/it/test.json | grep '"OK (1 test)"'
..."OK (1 test)"...
> ./mill show app.stopAndroidEmulator
> ./mill show app.deleteAndroidVirtualDevice
The android tests (existing typically in androidTest directory, aka instrumented tests) typically run on an android device. The createAndroidVirtualDevice command creates an AVD (Android Virtual Device) and the startAndroidEmulator command starts the AVD. The it task runs the integration tests against the available AVD. The stopAndroidEmulator command stops the AVD and the destroyAndroidVirtualDevice command destroys the AVD. The provided commands can be used in a CI/CD pipeline assuming the right setup is in place.
This example demonstrates how to create a basic "Hello World" Android application using the Mill build tool. It outlines the minimum setup required to compile Kotlin code, package it into an APK, and run the app on an Android device.
Understanding AndroidSdkModule
and AndroidAppKotlinModule
The two main modules you need to understand when building Android apps with Mill
are AndroidSdkModule
and AndroidAppKotlinModule
.
AndroidSdkModule
:
-
This module manages the installation and configuration of the Android SDK, which includes tools like
aapt
,d8
,zipalign
, andapksigner
. These tools are used for compiling, packaging, and signing Android applications.
AndroidAppKotlinModule
:
This module provides the step-by-step workflow for building an Android app. It handles
everything from compiling the code to generating a signed APK for distribution.
-
Compiling Kotlin code: The module compiles your Kotlin code into
.class
files, which is the first step in creating an Android app. -
Packaging into JAR: It then packages the compiled
.class
files into a JAR file, which is necessary before converting to Android’s format. -
Converting to DEX format: The JAR file is converted into DEX format, which is the executable format for Android applications.
-
Creating an APK: The DEX files and Android resources (like layouts and strings) are packaged together into an APK file, which is the installable file for Android devices.
-
Optimizing with zipalign: The APK is optimized using
zipalign
to ensure better performance on Android devices. -
Signing the APK: Finally, the APK is signed with a digital signature, allowing it to be distributed and installed on Android devices.