2020-08-20 17:08:02 +00:00
|
|
|
{ composeAndroidPackages, stdenv, lib, runtimeShell }:
|
2024-01-02 11:29:13 +00:00
|
|
|
{ name
|
|
|
|
, app ? null
|
2024-10-11 05:15:48 +00:00
|
|
|
, platformVersion ? "35"
|
|
|
|
, abiVersion ? "x86"
|
2023-03-04 12:14:45 +00:00
|
|
|
, systemImageType ? "default"
|
2024-01-02 11:29:13 +00:00
|
|
|
, enableGPU ? false # Enable GPU acceleration. It's deprecated, instead use `configOptions` below.
|
|
|
|
, configOptions ? (
|
|
|
|
# List of options to add in config.ini
|
|
|
|
lib.optionalAttrs enableGPU
|
|
|
|
(lib.warn
|
|
|
|
"enableGPU argument is deprecated and will be removed; use configOptions instead"
|
|
|
|
{ "hw.gpu.enabled" = "yes"; }
|
|
|
|
)
|
|
|
|
)
|
|
|
|
, extraAVDFiles ? [ ]
|
2023-03-04 12:14:45 +00:00
|
|
|
, package ? null
|
|
|
|
, activity ? null
|
|
|
|
, androidUserHome ? null
|
|
|
|
, avdHomeDir ? null # Support old variable with non-standard naming!
|
|
|
|
, androidAvdHome ? avdHomeDir
|
2024-01-02 11:29:13 +00:00
|
|
|
, deviceName ? "device"
|
|
|
|
, sdkExtraArgs ? { }
|
2023-03-04 12:14:45 +00:00
|
|
|
, androidAvdFlags ? null
|
|
|
|
, androidEmulatorFlags ? null
|
2020-04-24 23:36:52 +00:00
|
|
|
}:
|
|
|
|
|
|
|
|
let
|
|
|
|
sdkArgs = {
|
|
|
|
includeEmulator = true;
|
|
|
|
includeSystemImages = true;
|
2023-03-04 12:14:45 +00:00
|
|
|
} // sdkExtraArgs // {
|
|
|
|
cmdLineToolsVersion = "8.0";
|
|
|
|
platformVersions = [ platformVersion ];
|
2020-04-24 23:36:52 +00:00
|
|
|
systemImageTypes = [ systemImageType ];
|
|
|
|
abiVersions = [ abiVersion ];
|
2023-03-04 12:14:45 +00:00
|
|
|
};
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
sdk = (composeAndroidPackages sdkArgs).androidsdk;
|
|
|
|
in
|
|
|
|
stdenv.mkDerivation {
|
|
|
|
inherit name;
|
|
|
|
|
|
|
|
buildCommand = ''
|
|
|
|
mkdir -p $out/bin
|
|
|
|
|
|
|
|
cat > $out/bin/run-test-emulator << "EOF"
|
2020-08-20 17:08:02 +00:00
|
|
|
#!${runtimeShell} -e
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
# We need a TMPDIR
|
|
|
|
if [ "$TMPDIR" = "" ]
|
|
|
|
then
|
|
|
|
export TMPDIR=/tmp
|
|
|
|
fi
|
|
|
|
|
2023-03-04 12:14:45 +00:00
|
|
|
${if androidUserHome == null then ''
|
2020-04-24 23:36:52 +00:00
|
|
|
# Store the virtual devices somewhere else, instead of polluting a user's HOME directory
|
2023-03-04 12:14:45 +00:00
|
|
|
export ANDROID_USER_HOME=$(mktemp -d $TMPDIR/nix-android-user-home-XXXX)
|
|
|
|
'' else ''
|
|
|
|
mkdir -p "${androidUserHome}"
|
|
|
|
export ANDROID_USER_HOME="${androidUserHome}"
|
|
|
|
''}
|
|
|
|
|
|
|
|
${if androidAvdHome == null then ''
|
|
|
|
export ANDROID_AVD_HOME=$ANDROID_USER_HOME/avd
|
2020-04-24 23:36:52 +00:00
|
|
|
'' else ''
|
2023-03-04 12:14:45 +00:00
|
|
|
mkdir -p "${androidAvdHome}"
|
|
|
|
export ANDROID_AVD_HOME="${androidAvdHome}"
|
2020-04-24 23:36:52 +00:00
|
|
|
''}
|
|
|
|
|
|
|
|
# We need to specify the location of the Android SDK root folder
|
|
|
|
export ANDROID_SDK_ROOT=${sdk}/libexec/android-sdk
|
|
|
|
|
2023-03-04 12:14:45 +00:00
|
|
|
${lib.optionalString (androidAvdFlags != null) ''
|
|
|
|
# If NIX_ANDROID_AVD_FLAGS is empty
|
|
|
|
if [[ -z "$NIX_ANDROID_AVD_FLAGS" ]]; then
|
|
|
|
NIX_ANDROID_AVD_FLAGS="${androidAvdFlags}"
|
|
|
|
fi
|
|
|
|
''}
|
|
|
|
|
|
|
|
${lib.optionalString (androidEmulatorFlags != null) ''
|
|
|
|
# If NIX_ANDROID_EMULATOR_FLAGS is empty
|
|
|
|
if [[ -z "$NIX_ANDROID_EMULATOR_FLAGS" ]]; then
|
|
|
|
NIX_ANDROID_EMULATOR_FLAGS="${androidEmulatorFlags}"
|
|
|
|
fi
|
|
|
|
''}
|
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
# We have to look for a free TCP port
|
|
|
|
|
|
|
|
echo "Looking for a free TCP port in range 5554-5584" >&2
|
|
|
|
|
|
|
|
for i in $(seq 5554 2 5584)
|
|
|
|
do
|
2023-03-04 12:14:45 +00:00
|
|
|
if [ -z "$(${sdk}/bin/adb devices | grep emulator-$i)" ]
|
2020-04-24 23:36:52 +00:00
|
|
|
then
|
|
|
|
port=$i
|
|
|
|
break
|
|
|
|
fi
|
|
|
|
done
|
|
|
|
|
|
|
|
if [ -z "$port" ]
|
|
|
|
then
|
|
|
|
echo "Unfortunately, the emulator port space is exhausted!" >&2
|
|
|
|
exit 1
|
|
|
|
else
|
|
|
|
echo "We have a free TCP port: $port" >&2
|
|
|
|
fi
|
|
|
|
|
|
|
|
export ANDROID_SERIAL="emulator-$port"
|
|
|
|
|
2021-09-18 10:52:07 +00:00
|
|
|
# Create a virtual android device for testing if it does not exist
|
2024-01-02 11:29:13 +00:00
|
|
|
if [ "$(${sdk}/bin/avdmanager list avd | grep 'Name: ${deviceName}')" = "" ]
|
2020-04-24 23:36:52 +00:00
|
|
|
then
|
|
|
|
# Create a virtual android device
|
2024-01-02 11:29:13 +00:00
|
|
|
yes "" | ${sdk}/bin/avdmanager create avd --force -n ${deviceName} -k "system-images;android-${platformVersion};${systemImageType};${abiVersion}" -p $ANDROID_AVD_HOME/${deviceName}.avd $NIX_ANDROID_AVD_FLAGS
|
2020-04-24 23:36:52 +00:00
|
|
|
|
2024-01-02 11:29:13 +00:00
|
|
|
${builtins.concatStringsSep "\n" (
|
|
|
|
lib.mapAttrsToList (configKey: configValue: ''
|
|
|
|
echo "${configKey} = ${configValue}" >> $ANDROID_AVD_HOME/${deviceName}.avd/config.ini
|
|
|
|
'') configOptions
|
|
|
|
)}
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
${lib.concatMapStrings (extraAVDFile: ''
|
2024-01-02 11:29:13 +00:00
|
|
|
ln -sf ${extraAVDFile} $ANDROID_AVD_HOME/${deviceName}.avd
|
2020-04-24 23:36:52 +00:00
|
|
|
'') extraAVDFiles}
|
|
|
|
fi
|
|
|
|
|
|
|
|
# Launch the emulator
|
2023-03-04 12:14:45 +00:00
|
|
|
echo "\nLaunch the emulator"
|
2024-01-02 11:29:13 +00:00
|
|
|
$ANDROID_SDK_ROOT/emulator/emulator -avd ${deviceName} -no-boot-anim -port $port $NIX_ANDROID_EMULATOR_FLAGS &
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
# Wait until the device has completely booted
|
2024-01-02 11:29:13 +00:00
|
|
|
echo "Waiting until the emulator has booted the ${deviceName} and the package manager is ready..." >&2
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
${sdk}/libexec/android-sdk/platform-tools/adb -s emulator-$port wait-for-device
|
|
|
|
|
|
|
|
echo "Device state has been reached" >&2
|
|
|
|
|
|
|
|
while [ -z "$(${sdk}/libexec/android-sdk/platform-tools/adb -s emulator-$port shell getprop dev.bootcomplete | grep 1)" ]
|
|
|
|
do
|
|
|
|
sleep 5
|
|
|
|
done
|
|
|
|
|
|
|
|
echo "dev.bootcomplete property is 1" >&2
|
|
|
|
|
|
|
|
#while [ -z "$(${sdk}/libexec/android-sdk/platform-tools/adb -s emulator-$port shell getprop sys.boot_completed | grep 1)" ]
|
|
|
|
#do
|
|
|
|
#sleep 5
|
|
|
|
#done
|
|
|
|
|
|
|
|
#echo "sys.boot_completed property is 1" >&2
|
|
|
|
|
|
|
|
echo "ready" >&2
|
|
|
|
|
|
|
|
${lib.optionalString (app != null) ''
|
|
|
|
# Install the App through the debugger, if it has not been installed yet
|
|
|
|
|
|
|
|
if [ -z "${package}" ] || [ "$(${sdk}/libexec/android-sdk/platform-tools/adb -s emulator-$port shell pm list packages | grep package:${package})" = "" ]
|
|
|
|
then
|
|
|
|
if [ -d "${app}" ]
|
|
|
|
then
|
|
|
|
appPath="$(echo ${app}/*.apk)"
|
|
|
|
else
|
|
|
|
appPath="${app}"
|
|
|
|
fi
|
|
|
|
|
|
|
|
${sdk}/libexec/android-sdk/platform-tools/adb -s emulator-$port install "$appPath"
|
|
|
|
fi
|
|
|
|
|
|
|
|
# Start the application
|
|
|
|
${lib.optionalString (package != null && activity != null) ''
|
|
|
|
${sdk}/libexec/android-sdk/platform-tools/adb -s emulator-$port shell am start -a android.intent.action.MAIN -n ${package}/${activity}
|
|
|
|
''}
|
|
|
|
''}
|
|
|
|
EOF
|
|
|
|
chmod +x $out/bin/run-test-emulator
|
|
|
|
'';
|
|
|
|
}
|