An Android application for Kolibri. Chaquopy is used to access Python from the Android runtime.
The application consists of 2 components:
The Kolibri application is available in Google Play Store
as Endless Key using the app ID org.endlessos.Key
. Endless users
should disable analytics as explained
below.
-
Install Java 17 and python3:
apt install openjdk-17-jdk-headless python3 python3-pip
-
Install the Android SDK:
make sdk
This will install the SDK in
$ANDROID_HOME
(~/.android/sdk
by default). -
Install the Python dependencies:
pip install -r requirements.txt
-
Build the debug and release APKs:
./gradlew build
-
Run the instrumented tests tests:
./gradlew allDevicesCheck
See the documentation for more details on running tests from the command line.
Toolbox allows a mixture of the above build processes by providing a development environment inside a container.
-
Install toolbox
-
Run
make build_toolbox
. -
Run
toolbox enter android_kolibri
to enter the container. -
Continue at step 3 in Building for development.
By default, kolibri-installer-android includes the newest released version of
kolibri-explore-plugin.
To test a different version, set the exploreUrl
and appsBundleUrl
in
~/.gradle/gradle.properties
:
exploreUrl=file:///path/to/kolibri_explore_plugin.whl
appsBundleUrl=file:///path/to/apps-bundle.zip
In order to upload APKs or AABs to Google Play, they need to be signed by a
trusted key. A signing configuration for released builds will be created if the
upload.properties
file is present. This is a Java properties file with the
following required keys:
storeFile
- Path to the Java keystore file.storePassword
- Password for the Java keystore.keyAlias
- Alias of the signing key with the keystore.keyPassword
(optional) - Password for the signing key. If this is not set, the keystore password will be used.
-
Connect your Android device over USB, with USB Debugging enabled.
-
Ensure that
adb devices
brings up your device. -
Run
./gradlew installDebug
to install onto the device.
- Run
adb shell am start -n org.endlessos.Key/org.endlessos.key.KolibriActivity
To get all debug logs from the application, run:
adb logcat '*:F' EndlessKey EKWebConsole AndroidRuntime python.stdout python.stderr
- Start the Kolibri server via Android app
- Open a browser and see debug logs
- If your device doesn't aggressively kill the server, you can open Chrome and use remote debugging tools to see the logs on your desktop.
- You can also leave the app open and port forward the Android device's Kolibri port using adb:
adb forward tcp:8080 tcp:8081
then going into your desktop's browser and accessing localhost:8081
. Note that you can map to any port on the host machine, the second argument.
Alternatively, you can debug the webview directly. Modern Android versions should let you do so from the developer settings.
You could also do so using Weinre. Visit the site to learn how to install and setup. You will have to build a custom Kolibri .whl file that contains the weinre script tag in the base.html file.
Metrics and crashes are collected from the application using Firebase Analytics and Crashlytics. In order to see details of the information being collected, increase the log levels for the relevant tags:
adb shell setprop log.tag.FA VERBOSE
adb shell setprop log.tag.FA-SVC VERBOSE
adb shell setprop log.tag.FirebaseCrashlytics DEBUG
Normally the analytics events are cached locally on disk and sent to the Firebase server periodically. In order to send the events immediately, enable Analytics debug mode for this application:
adb shell setprop debug.firebase.analytics.app org.endlessos.Key
Finally, events can be seen in the Firebase console
DebugView in realtime by setting the application
as the Android debug app. This requires enabling
Developer options
on the device, choosing Select debug app
, and
setting it to org.endlessos.Key
.
By default, Analytics and Crashlytics are enabled on release builds and
disabled on debug builds. This prevents development work from polluting
our production metrics. Event collection can be explicitly enabled or
disabled at runtime using the debug.org.endlessos.key.analytics
system
property. For example:
adb shell setprop debug.org.endlessos.key.analytics true
The primary use case is for testing analytics development on debug
builds. However, it can also be used to opt out of analytics by setting
the property value to false
. Endless testing users should explicitly
disable analytics on release builds so that production metrics are not
affected by test usage.
- adb is pretty helpful. Here are some useful uses:
adb logcat -b all -c
will clear out the device's log. (Docs)- Logcat also has a large variety of filtering options. Check out
the docs for those. Particularly, the filter
*:S
silences all tags without a filterspec. Combined with a desired tag, this will show only logs from that tag. Theadb
CLI option-s
is a shorthand for*:S
, soadb logcat -s SomeTag
will only show logs fromSomeTag
. Alternatively,*:F
will show only fatal logs from other tags.
- Logcat also has a large variety of filtering options. Check out
the docs for those. Particularly, the filter
- Uninstall from terminal using
adb shell pm uninstall org.endlessos.Key
. (Docs)
- Docker shouldn't be rebuilding very often, so it shouldn't be using that much storage. But if it does, you can run
docker system prune
to clear out all "dangling" images, containers, and layers. If you've been constantly rebuilding, it will likely get you several gigabytes of storage.
Installing the APK to a real device during development is slow. Instead, the Android Emulator can be used for faster installs in a clean environment. The recommended way to use the emulator is through Android Studio, but it can also be invoked directly.
First, ensure the emulator is installed in the Android SDK. The make setup
target will install the necessary pieces. Assuming the SDK is installed in
/opt/android/sdk
:
/opt/android/sdk/cmdline-tools/latest/bin/sdkmanager emulator
Next, install a platform and system image:
/opt/android/sdk/cmdline-tools/latest/bin/sdkmanager "platforms;android-34"
/opt/android/sdk/cmdline-tools/latest/bin/sdkmanager "system-images;android-34;google_apis_playstore;x86_64"
Now an Android Virtual Device (AVD) needs to be created. This can be done from the command line with avdmanager.
There are many device definitions available. To see the list, run:
/opt/android/sdk/cmdline-tools/latest/bin/avdmanager list device
To create an AVD, a name, system image and device must be provided. Assuming the system image provided above and a Pixel 5 device:
/opt/android/sdk/cmdline-tools/latest/bin/avdmanager create avd --name test \
--package "system-images;android-34;google_apis_playstore;x86_64" --device pixel_5
The AVD should be ready now and avdmanager list avd
will show the details.
Start it from the
emulator:
/opt/android/sdk/emulator/emulator -avd test
A window should show up showing Android booting. Once it's running, it can be
connected to with adb
in all the ways shown above. When you're done with the
emulator, you can stop it with Ctrl-C
from the terminal where you started
it. By default, the emulator stores snapshots and the next time you start that
AVD it will be in the same state.
The Android Emulator does not support removable storage such as USB drives.
However, you can approximate that experience using an SD card. By default,
avdmanager
will not create an SD card connected to the device. To use one,
add --sdcard 1G
to the avdmanager create avd
command above. That will
create a 1 GB card image, but other sizes can be used with typical suffixes
like M
for MB.
The SD card can be populated within Android from the file manager, but this
can be cumbersome. Instead, you can work with the SD card image from the host.
Run avdmanager list avd
to get the path to the AVD that was created. The SD
card disk image will be in that directory named sdcard.img
. This is a raw
disk image with no partitions containing a FAT filesystem. Once the emulator
is started, a second file named sdcard.img.qcow2
will be created. This is a
QEMU disk
image
that's setup to use the original sdcard.img
file as a read only base image.
Once the qcow2 image has been created, any changes to the sdcard.img
file
will not be reflected in the SD card seen in the emulator.
If the emulator hasn't been run yet, the raw disk image can be updated to include any desired files by mounting the FAT filesystem locally. First, create a loop block device pointing to the raw disk image:
sudo losetup --show -f ~/.android/avd/test.avd/sdcard.img
This will show the connected loop device. Assuming the first device,
/dev/loop0
, it can now be mounted:
sudo mount /dev/loop0 /mnt
Now the filesystem will be mounted at /mnt
and files can be added or
removed. When done, unmount the filesystem and disconnect the loop device:
sudo umount /mnt
sudo losetup -d /dev/loop0
If the qcow2 image has already been created, it can be accessed with some help
from QEMU's network block device (NBD) server,
qemu-nbd. This tool is
included in the qemu-utils
package on Debian systems.
First, ensure that the kernel's nbd
module is loaded:
sudo modprobe nbd
Now a network block device can be connected and mounted similar to the above raw image usage:
sudo qemu-nbd -c /dev/nbd0 ~/.android/avd/test.avd/sdcard.img.qcow2
sudo mount /dev/nbd0 /mnt
When done, unmount the filesystem and disconnect the block device:
sudo umount /mnt
sudo qemu-nbd -d /dev/nbd0
In order to share an SD card among multiple AVDs, one can be created ahead of
time using the emulator's mksdcard
tool:
/opt/android/sdk/emulator/mksdcard 512M ~/.android/avd/test-sdcard.img
This would create a 512 MB SD card image at ~/.android/avd/test-sdcard.img
.
When creating an AVD, provide this path to the --sdcard
option instead of
providing a size. Once this image is used is an emulator, a qcow2 image will
be created wrapping it at ~/.android/avd/test-sdcard.img.qcow2
. In this way,
it can be accessed from the host in the same ways as an SD card image created
by avdmanager
.