bits-and-bytes/src/flutter/build.sh

249 lines
7.4 KiB
Bash

#!/bin/bash
set -e
# Parse CLI arguments
# (Thanks StackOverflow https://stackoverflow.com/questions/192249/how-do-i-parse-command-line-arguments-in-bash)
while [[ $# -gt 0 ]]; do
case $1 in
--name)
# The name of the application. Only useful for displaying in the script.
# If not specified, will use the "name: " attribute from the pubspec.yaml
NAME=$2
shift
shift
;;
--not-signed)
# Tells the script that the APKs are *NOT* signed and have to be manually
# signed.
ALREADY_SIGNED=n
shift
;;
--zipalign)
# The path to the zipalign binary. If not specified, lets the shell resolve the path
# itself.
# Only useful with --not-signed.
ZIPALIGN=$2
shift
shift
;;
--apksigner)
# The path to the apksigner binary. If not specified, lets the shell resolve the path
# itself.
# Only useful with --not-signed.
APKSIGNER=$2
shift
shift
;;
--notify-send)
# The path to the notify-send binary. If not specified, lets the shell resolve the path
# itself.
# Only useful with --not-signed.
NOTIFY_SEND=$2
shift
shift
;;
--provider-config)
# The path to the provider config passed to apksigner via --provider-arg.
# Only useful with --not-signed.
PROVIDER_CONFIG=$2
shift
shift
;;
--no-clean)
# Does not run "flutter clean" before running the build.
CLEAN_BUILD=n
shift
;;
--min-sdk-version)
# The minimum SDK version that should be able to validate the APK signature.
# Only useful with --not-signed.
MIN_SDK_VERSION=$2
shift
shift
;;
--check)
# Prints all parsed options and exits.
JUST_CHECK=y
shift
;;
--skip-build)
# Skips the entire build but not the signing, if --not-signed is specified. If --not-signed
# is not specified, then the build is skipped, but the APKs are copied into the correct folder.
# This flag expects that the build has run at least once and the correct release directory
# already exists.
SKIP_BUILD=y
shift
;;
--no-notification)
# Does not create a notification when asking for the PIN of the Yubikey.
# Only useful with --not-signed.
SEND_NOTIFICATION=n
shift
;;
--pigeon)
# Specifies to run `dart run pigeon --input $2`
PIGEON_FILES="$PIGEON_FILES $2"
shift
shift
;;
--flutter)
# Specifies the path to the flutter binary
FLUTTER=$2
shift
shift
;;
*)
echo "Unknown argument: $1"
shift
;;
esac
done
# Extract the name, if not passed
if [[ -z "$NAME" ]]; then
NAME=$(grep -E "^name: " pubspec.yaml | cut -b 6-)
fi
# Default values
APKSIGNER=${APKSIGNER:-apksigner}
ZIPALIGN=${ZIPALIGN:-zipalign}
ALREADY_SIGNED=${ALREADY_SIGNED:-y}
MIN_SDK_VERSION=${MIN_SDK_VERSION:-24}
CLEAN_BUILD=${CLEAN_BUILD:-y}
SKIP_BUILD=${SKIP_BUILD:-n}
NOTIFY_SEND=${NOTIFY_SEND:-notify-send}
SEND_NOTIFICATION=${SEND_NOTIFICATION:-y}
FLUTTER=${FLUTTER:-flutter}
# Parse version info
version=$(grep -E "^version: " pubspec.yaml | cut -b 10-)
IFS="+" read -ra version_parts <<< "$version"
version_code="${version_parts[1]}"
release_dir="./release-${version}"
# Print a header
echo "===== ${NAME} ====="
echo "Building version ${version}"
echo "Moving APKs into ${release_dir} after build"
echo "Flutter: ${FLUTTER}"
echo "Clean build: ${CLEAN_BUILD}"
echo "Pigeons to build: ${PIGEON_FILES}"
echo "Skipping build: ${SKIP_BUILD}"
echo "Sending notification: ${SEND_NOTIFICATION}"
echo "APKs already signed: ${ALREADY_SIGNED}"
if [[ "$ALREADY_SIGNED" = "n" ]]; then
echo "Used zipalign: ${ZIPALIGN}"
echo "Used apksigner: ${APKSIGNER}"
echo "Provider config: ${PROVIDER_CONFIG}"
echo "Minimum SDK verifiability: ${MIN_SDK_VERSION}"
fi
if [[ "${JUST_CHECK}" = "y" ]]; then
echo
echo "Terminating here because --check was used"
exit
fi
# For better readability, print an empty line
echo
# Check if we have a changelog file for that version
if [[ ! -f "./fastlane/metadata/android/en-US/changelogs/$version_code.txt" ]]; then
echo "Warning: No changelog item for $version_code"
fi
if [[ "${SKIP_BUILD}" = "y" ]]; then
echo "Skipping build because of --skip-build"
else
# Create the directory
[[ -d "${release_dir}" ]] && echo "Warning: Release directory ${release_dir} already exists"
mkdir "${release_dir}" || true
if [[ "${CLEAN_BUILD}" = "y" ]]; then
# Clean flutter build
$FLUTTER clean
fi
# Get dependencies
$FLUTTER pub get
# Build everything again
$FLUTTER pub run build_runner build --delete-conflicting-outputs
# Build pigeons
for pigeon in $PIGEON_FILES; do
$FLUTTER run pigeon --input $pigeon
done
# Build the release apk
$FLUTTER build apk \
--release \
--split-per-abi \
--split-debug-info="${release_dir}/debug-info"
fi
# If we have to sign, ask for the PIN upfront
if [[ "$ALREADY_SIGNED" = "n" ]]; then
[[ "${SEND_NOTIFICATION}" = "y" ]] && ${NOTIFY_SEND} \
--urgency normal \
--app-name "Flutter Build" \
--icon dialog-password \
"Signing PIN required" \
"The build is done and the PIN for manually signing the APKs must be entered"
# Thanks https://geoffreymetais.github.io/code/key-signing/#scripting
echo "Please enter Yubikey PIN code "
stty -echo
trap 'stty echo' EXIT
read -p 'PIN: ' YUBI_PIN
stty echo
trap - EXIT
fi
# Move everything
for platform in arm64-v8a armeabi-v7a x86_64; do
echo "Processing the $platform release..."
raw_apk="build/app/outputs/flutter-apk/app-${platform}-release.apk"
if [[ "$ALREADY_SIGNED" = "y" ]]; then
# Simply copy artifacts
cp -f "$raw_apk" "${release_dir}/${NAME}-${platform}-release.apk"
else
# https://developer.android.com/build/building-cmdline#sign_cmdline
aligned_apk="${release_dir}/app-${platform}-release-aligned.apk"
signed_apk="${release_dir}/app-${platform}-release.apk"
${ZIPALIGN} -p 4 "$raw_apk" "${aligned_apk}"
# NOTE: "-J-add-exports jdk.crypto.cryptoki/sun.security.pkcs11=ALL-UNNAMED" tells the wrapper that is
# `apksigner` to append "--add-exports jdk.[...]" as a parameter to the Java CLI. It is in
# quotation marks such that it is passed with only one (1) dash (it would otherwise be "--add-exports -jdk.[...]",
# which is wrong).
# At least on NixOS, apksigner refuses to use the sun.security.pkcs11.SunPKCS11 provider because
# it is not exported for its unnamed class. So, we have to do a little trickery to "export" the
# class to apksigner.
${APKSIGNER} "-J-add-exports jdk.crypto.cryptoki/sun.security.pkcs11=ALL-UNNAMED" sign \
--ks NONE \
--ks-pass "pass:$YUBI_PIN" \
--provider-class sun.security.pkcs11.SunPKCS11 \
--provider-arg "${PROVIDER_CONFIG}" \
--ks-type PKCS11 \
--min-sdk-version "${MIN_SDK_VERSION}" \
--max-sdk-version 34 \
--in "${aligned_apk}" \
--out "${signed_apk}" \
--v1-signing-enabled \
--v2-signing-enabled \
--v3-signing-enabled \
--v4-signing-enabled
# Safety check
${APKSIGNER} verify --min-sdk-version "${MIN_SDK_VERSION}" "${signed_apk}"
# Remove temporary artifact
rm "${aligned_apk}"
fi
done
# Prevent leaking the PIN
unset YUBI_PIN