mirror of
https://github.com/meshtastic/Meshtastic-Android.git
synced 2026-06-02 06:24:16 +02:00
fix(flatpak): generate complete offline-buildable manifest for desktopApp (#5610)
Co-authored-by: Claude Opus 4.7 <noreply@anthropic.com>
This commit is contained in:
@@ -5,7 +5,7 @@ can validate `flatpak-sources.json` end-to-end without round-tripping through hi
|
||||
|
||||
## What it tests that our CI doesn't
|
||||
|
||||
Our CI (`:desktopApp:assemble :captureFlatpakSources`) only proves the manifest can be *generated*.
|
||||
Our CI (`:desktopApp:packageUberJarForCurrentOS :captureFlatpakSources`) only proves the manifest can be *generated*.
|
||||
Vid's CI is where the manifest actually gets *consumed* by `flatpak-builder`. This script
|
||||
runs that step locally:
|
||||
|
||||
@@ -28,15 +28,28 @@ instead of waiting on cross-repo CI.
|
||||
- Docker (Docker Desktop on macOS works — the container needs `--privileged` to use
|
||||
bubblewrap; that's enabled by default).
|
||||
- ~10 GB free disk for the SDK + Gradle cache.
|
||||
- A populated Gradle cache (`./gradlew :desktopApp:assemble` must have run; the script
|
||||
does this implicitly via `:captureFlatpakSources`).
|
||||
- A populated Gradle cache (`./gradlew :desktopApp:packageUberJarForCurrentOS` must have
|
||||
run; the script does this implicitly via `:captureFlatpakSources`).
|
||||
|
||||
## Usage
|
||||
|
||||
```bash
|
||||
# Full offline build (~10–20 min the first time, faster after — Docker image is cached)
|
||||
# Full offline build — Linux host required (~15–30 min first time)
|
||||
scripts/verify-flatpak/verify.sh
|
||||
|
||||
# URLs + sha256 verification only; skips the Gradle build phase.
|
||||
# Works on macOS where nested bwrap fails under Docker Desktop's seccomp.
|
||||
scripts/verify-flatpak/verify.sh --download-only
|
||||
|
||||
# Reuse an already-generated flatpak-sources.json (don't re-run Gradle)
|
||||
scripts/verify-flatpak/verify.sh --skip-regen
|
||||
|
||||
# Tight iteration loop after a failed run: refresh overlay yaml + manifest
|
||||
# only, then re-run flatpak-builder. Skips Gradle regen, vid-repo fetch,
|
||||
# and Meshtastic-Android rsync. Use when you've just patched the YAML
|
||||
# overlay or regenerated flatpak-sources.json by hand.
|
||||
scripts/verify-flatpak/verify.sh --rebuild-only
|
||||
|
||||
# Cross-arch test via QEMU emulation (slower)
|
||||
scripts/verify-flatpak/verify.sh --arch aarch64
|
||||
|
||||
@@ -44,12 +57,20 @@ scripts/verify-flatpak/verify.sh --arch aarch64
|
||||
scripts/verify-flatpak/verify.sh --shell
|
||||
```
|
||||
|
||||
### macOS limitation
|
||||
|
||||
`flatpak-builder` runs the build phase inside `bwrap` (bubblewrap). Nested
|
||||
bwrap fails inside Docker Desktop on macOS with
|
||||
`prctl(PR_SET_SECCOMP) EINVAL`. The script refuses to run a full build on
|
||||
macOS by default — pass `--download-only` to validate URLs + sha256s without
|
||||
executing the Gradle build, or run the full script on a Linux host.
|
||||
|
||||
## Interpreting failures
|
||||
|
||||
| Symptom | Likely cause |
|
||||
| ---------------------------------------------------------------- | --------------------------------------------------------------------------------------------------------- |
|
||||
| `Error downloading mirror: ... 404` on `repo.maven.apache.org` | URL captured by the listener was wrong or artifact moved. Check the source repo hosting it. |
|
||||
| `sha256 mismatch` | Stale `flatpak-sources.json`; re-run `:desktopApp:assemble :captureFlatpakSources`. |
|
||||
| `sha256 mismatch` | Stale `flatpak-sources.json`; re-run `:desktopApp:packageUberJarForCurrentOS :captureFlatpakSources`. |
|
||||
| Gradle: `Could not resolve all artifacts ... offline mode` | Missing dep in the manifest — usually a compiler plugin or BOM that wasn't downloaded during capture. |
|
||||
|
||||
## Files
|
||||
|
||||
@@ -68,17 +68,19 @@ modules:
|
||||
- install -Dm644 -t /app/share/icons/hicolor/scalable/apps org.meshtastic.desktop.svg
|
||||
- install -Dm644 -t /app/share/applications org.meshtastic.desktop.desktop
|
||||
- install -Dm644 -t /app/share/metainfo org.meshtastic.desktop.metainfo.xml
|
||||
# Redirect the Gradle wrapper to the bundled distribution (no network).
|
||||
- sed -i 's|distributionUrl=.*|distributionUrl=gradle-all.zip|' gradle/wrapper/gradle-wrapper.properties
|
||||
- echo "org.gradle.java.installations.auto-detect=false" >> gradle.properties
|
||||
- echo "org.gradle.java.installations.auto-download=false" >> gradle.properties
|
||||
- echo "org.gradle.java.installations.paths=/usr/lib/sdk/openjdk21/jvm/openjdk-21,/usr/lib/sdk/openjdk17/jvm/openjdk-17" >> gradle.properties
|
||||
- >
|
||||
sed -i
|
||||
's/^\(\s*\)vendor\.set(JvmVendorSpec\.JETBRAINS)/\1\/\/ vendor.set(JvmVendorSpec.JETBRAINS)/'
|
||||
desktop/build.gradle.kts
|
||||
desktopApp/build.gradle.kts
|
||||
# Force Gradle to resolve from the bundled offline-repository ONLY (true offline test).
|
||||
- ./gradlew --offline :desktop:packageUberJarForCurrentOS
|
||||
- ./gradlew --offline :desktopApp:packageUberJarForCurrentOS
|
||||
- >
|
||||
JAR_FILE=$(find desktop/build/compose/jars/ -name "*.jar" -type f | head -1)
|
||||
JAR_FILE=$(find desktopApp/build/compose/jars/ -name "*.jar" -type f | head -1)
|
||||
&& install -Dm755 "$JAR_FILE" /app/lib/meshtastic-desktop.jar
|
||||
sources:
|
||||
- type: file
|
||||
@@ -90,8 +92,10 @@ modules:
|
||||
- type: dir
|
||||
path: meshtastic-android
|
||||
- type: file
|
||||
url: https://services.gradle.org/distributions/gradle-9.4.1-bin.zip
|
||||
sha256: 2ab2958f2a1e51120c326cad6f385153bb11ee93b3c216c5fccebfdfbb7ec6cb
|
||||
# Must match the version in gradle/wrapper/gradle-wrapper.properties.
|
||||
# Bumping the wrapper? Update both the URL and sha256 here.
|
||||
url: https://services.gradle.org/distributions/gradle-9.5.1-all.zip
|
||||
sha256: c72fb9991f6025cbe337d52ba77e531b3faf62bdd3e348fe1ccee9f51c71adb0
|
||||
dest: "gradle/wrapper"
|
||||
dest-filename: "gradle-bin.zip"
|
||||
dest-filename: "gradle-all.zip"
|
||||
- flatpak-sources.json
|
||||
|
||||
@@ -1,33 +1,50 @@
|
||||
#!/usr/bin/env bash
|
||||
# Local replica of vid's flatpak CI (vidplace7/org.meshtastic.desktop, .github/workflows/build-flatpak.yml)
|
||||
# but flipped to true-offline mode: our flatpak-sources.json is included and --share=network is removed.
|
||||
# Local replica of vid's flatpak CI (vidplace7/org.meshtastic.desktop,
|
||||
# .github/workflows/build-flatpak.yml) but flipped to true-offline mode: our
|
||||
# flatpak-sources.json is included and --share=network is removed from the
|
||||
# build phase.
|
||||
#
|
||||
# Goal: validate flatpak-sources.json without bugging vid to push & re-run his workflow.
|
||||
# Goal: validate flatpak-sources.json end-to-end (download + verify sha256s +
|
||||
# offline Gradle build) without bugging vid to push & re-run his workflow.
|
||||
#
|
||||
# Requirements:
|
||||
# - Docker (Docker Desktop on macOS is fine; needs ~10GB free + ability to run --privileged)
|
||||
# - This Meshtastic-Android checkout has produced flatpak-sources.json
|
||||
# (run `./gradlew :desktopApp:assemble :captureFlatpakSources` first, or this script will do it)
|
||||
# - Docker (Docker Desktop on macOS works for --download-only mode; full builds
|
||||
# need a Linux host because flatpak-builder uses nested bwrap which fails
|
||||
# under Docker Desktop's seccomp sandbox).
|
||||
# - ~15GB free disk for the SDK + Gradle cache + builddir.
|
||||
#
|
||||
# Usage:
|
||||
# scripts/verify-flatpak/verify.sh # full build, x86_64
|
||||
# scripts/verify-flatpak/verify.sh --arch aarch64 # cross-arch via QEMU emulation
|
||||
# scripts/verify-flatpak/verify.sh --shell # drop into the container shell instead of building
|
||||
# scripts/verify-flatpak/verify.sh # full offline build (Linux only)
|
||||
# scripts/verify-flatpak/verify.sh --download-only # URLs+sha256 only (works on macOS)
|
||||
# scripts/verify-flatpak/verify.sh --arch aarch64 # cross-arch via QEMU emulation
|
||||
# scripts/verify-flatpak/verify.sh --shell # drop into builder container shell
|
||||
# scripts/verify-flatpak/verify.sh --skip-regen # reuse flatpak-sources.json; still re-clone vid + re-rsync source
|
||||
# scripts/verify-flatpak/verify.sh --rebuild-only # tight loop: refresh overlay+manifest only, then re-run flatpak-builder
|
||||
#
|
||||
# Iteration tip: after a build fails partway, fix the overlay YAML (or the
|
||||
# Meshtastic-Android source) and re-run with --rebuild-only — Gradle regen,
|
||||
# vid-repo fetch, and full source rsync are all skipped, so you get straight
|
||||
# back to flatpak-builder in seconds.
|
||||
|
||||
set -euo pipefail
|
||||
|
||||
ARCH="x86_64"
|
||||
DROP_TO_SHELL=0
|
||||
DOWNLOAD_ONLY=0
|
||||
SKIP_REGEN=0
|
||||
REBUILD_ONLY=0
|
||||
while [[ $# -gt 0 ]]; do
|
||||
case "$1" in
|
||||
--arch) ARCH="$2"; shift 2 ;;
|
||||
--shell) DROP_TO_SHELL=1; shift ;;
|
||||
-h|--help) sed -n '2,17p' "$0"; exit 0 ;;
|
||||
--download-only) DOWNLOAD_ONLY=1; shift ;;
|
||||
--skip-regen) SKIP_REGEN=1; shift ;;
|
||||
--rebuild-only) REBUILD_ONLY=1; SKIP_REGEN=1; shift ;;
|
||||
-h|--help) sed -n '2,28p' "$0"; exit 0 ;;
|
||||
*) echo "Unknown arg: $1" >&2; exit 2 ;;
|
||||
esac
|
||||
done
|
||||
|
||||
# Map flatpak arch names to docker platform names
|
||||
case "$ARCH" in
|
||||
x86_64) DOCKER_PLATFORM="linux/amd64" ;;
|
||||
aarch64) DOCKER_PLATFORM="linux/arm64" ;;
|
||||
@@ -38,11 +55,12 @@ REPO_ROOT="$(git -C "$(dirname "$0")" rev-parse --show-toplevel)"
|
||||
WORK="$REPO_ROOT/build/flatpak-verify"
|
||||
OVERLAY="$REPO_ROOT/scripts/verify-flatpak/desktop-offline.yaml"
|
||||
SOURCES_JSON="$REPO_ROOT/flatpak-sources.json"
|
||||
GRADLE_HOME_ISOLATED="$REPO_ROOT/build/flatpak-gradle-home"
|
||||
VID_REPO="https://github.com/vidplace7/org.meshtastic.desktop.git"
|
||||
|
||||
# Image provides flatpak + flatpak-builder. The freedesktop 25.08 runtime declared in
|
||||
# the manifest is pulled from flathub at build time (no 25.08 image exists yet; 24.08 is
|
||||
# fine as the builder host because the SDK used at compile time comes from flathub).
|
||||
# bilelmoussaoui's image is what vid's CI uses; freedesktop-24.08 is the latest
|
||||
# tag available. The 25.08 runtime declared in the manifest is pulled from
|
||||
# flathub at build time inside the container.
|
||||
BUILDER_IMAGE="bilelmoussaoui/flatpak-github-actions:freedesktop-24.08"
|
||||
|
||||
step() { printf '\n\033[1;34m==> %s\033[0m\n' "$*"; }
|
||||
@@ -50,67 +68,94 @@ fail() { printf '\033[1;31m!! %s\033[0m\n' "$*" >&2; exit 1; }
|
||||
|
||||
command -v docker >/dev/null 2>&1 || fail "docker is required; install Docker Desktop or equivalent."
|
||||
|
||||
step "Ensuring flatpak-sources.json is fresh"
|
||||
if [[ ! -f "$SOURCES_JSON" ]]; then
|
||||
(cd "$REPO_ROOT" && ./gradlew --no-build-cache --no-configuration-cache :desktopApp:assemble :captureFlatpakSources)
|
||||
# Refuse full-build mode on macOS — nested bwrap fails under Docker Desktop's
|
||||
# seccomp and the user will spend 20 minutes finding out. They can override
|
||||
# with --download-only.
|
||||
if [[ "$(uname -s)" == "Darwin" && $DOWNLOAD_ONLY -eq 0 && $DROP_TO_SHELL -eq 0 ]]; then
|
||||
fail "Full flatpak-builder runs require a Linux host (nested bwrap fails under Docker Desktop on macOS). Re-run with --download-only, or use --shell to poke around manually."
|
||||
fi
|
||||
|
||||
if [[ $SKIP_REGEN -eq 0 ]]; then
|
||||
step "Regenerating flatpak-sources.json via isolated Gradle home"
|
||||
rm -rf "$GRADLE_HOME_ISOLATED"
|
||||
# Drive the same task the in-flatpak build runs so runtime-classpath deps (skiko, ktor-cio,
|
||||
# datastore-proto, etc.) are resolved and captured — :assemble only triggers compileClasspath.
|
||||
(cd "$REPO_ROOT" && ./gradlew --no-build-cache --no-configuration-cache \
|
||||
-Dgradle.user.home="$GRADLE_HOME_ISOLATED" \
|
||||
-I gradle/init-scripts/flatpak-ops.init.gradle.kts \
|
||||
:desktopApp:packageUberJarForCurrentOS :captureFlatpakSources)
|
||||
cp "$REPO_ROOT/build/flatpak-ops-sources.json" "$SOURCES_JSON"
|
||||
elif [[ ! -f "$SOURCES_JSON" ]]; then
|
||||
fail "--skip-regen specified but $SOURCES_JSON does not exist."
|
||||
fi
|
||||
|
||||
step "Preparing workspace at $WORK"
|
||||
mkdir -p "$WORK"
|
||||
if [[ ! -d "$WORK/org.meshtastic.desktop/.git" ]]; then
|
||||
git clone --depth 1 --recurse-submodules "$VID_REPO" "$WORK/org.meshtastic.desktop"
|
||||
if [[ $REBUILD_ONLY -eq 1 ]]; then
|
||||
[[ -d "$WORK/org.meshtastic.desktop/.git" ]] || \
|
||||
fail "--rebuild-only needs an existing workspace at $WORK; run without it once first."
|
||||
else
|
||||
git -C "$WORK/org.meshtastic.desktop" fetch --depth 1 origin main
|
||||
git -C "$WORK/org.meshtastic.desktop" reset --hard origin/main
|
||||
git -C "$WORK/org.meshtastic.desktop" submodule update --init --recursive --depth 1
|
||||
step "Preparing workspace at $WORK"
|
||||
mkdir -p "$WORK"
|
||||
if [[ ! -d "$WORK/org.meshtastic.desktop/.git" ]]; then
|
||||
git clone --depth 1 --recurse-submodules "$VID_REPO" "$WORK/org.meshtastic.desktop"
|
||||
else
|
||||
git -C "$WORK/org.meshtastic.desktop" fetch --depth 1 origin main
|
||||
git -C "$WORK/org.meshtastic.desktop" reset --hard origin/main
|
||||
git -C "$WORK/org.meshtastic.desktop" submodule update --init --recursive --depth 1
|
||||
fi
|
||||
fi
|
||||
|
||||
# Always refreshed — these are the iteration knobs:
|
||||
# overlay yaml = the manifest we're testing
|
||||
# flatpak-sources.json = the artifact we're validating
|
||||
step "Wiring overlay manifest + our flatpak-sources.json"
|
||||
cp "$OVERLAY" "$WORK/org.meshtastic.desktop/org.meshtastic.desktop.yaml"
|
||||
cp "$SOURCES_JSON" "$WORK/org.meshtastic.desktop/flatpak-sources.json"
|
||||
|
||||
# Materialize a clean copy of our checkout (excluding build outputs) for `type: dir`.
|
||||
# flatpak-builder copies the whole tree — skip heavy/irrelevant paths.
|
||||
step "Snapshotting Meshtastic-Android checkout (excluding build/, .gradle/)"
|
||||
rsync -a --delete \
|
||||
--exclude='/build/' \
|
||||
--exclude='/.gradle/' \
|
||||
--exclude='*/build/' \
|
||||
--exclude='*/.gradle/' \
|
||||
--exclude='/.idea/' \
|
||||
--exclude='/local.properties' \
|
||||
"$REPO_ROOT/" "$WORK/org.meshtastic.desktop/meshtastic-android/"
|
||||
if [[ $REBUILD_ONLY -eq 0 ]]; then
|
||||
step "Snapshotting Meshtastic-Android checkout (excluding build/, .gradle/)"
|
||||
rsync -a --delete \
|
||||
--exclude='/build/' \
|
||||
--exclude='/.gradle/' \
|
||||
--exclude='*/build/' \
|
||||
--exclude='*/.gradle/' \
|
||||
--exclude='/.idea/' \
|
||||
--exclude='/local.properties' \
|
||||
"$REPO_ROOT/" "$WORK/org.meshtastic.desktop/meshtastic-android/"
|
||||
fi
|
||||
|
||||
step "Pulling builder image: $BUILDER_IMAGE ($DOCKER_PLATFORM)"
|
||||
docker pull --platform "$DOCKER_PLATFORM" "$BUILDER_IMAGE" >/dev/null
|
||||
|
||||
DOCKER_RUN_ARGS=(
|
||||
--rm
|
||||
--privileged
|
||||
-v "$WORK/org.meshtastic.desktop:/work"
|
||||
-w /work
|
||||
--platform "$DOCKER_PLATFORM"
|
||||
--security-opt seccomp=unconfined
|
||||
)
|
||||
|
||||
if [[ $DROP_TO_SHELL -eq 1 ]]; then
|
||||
step "Dropping into builder shell — flatpak-builder is on PATH"
|
||||
exec docker run --rm -it --privileged \
|
||||
-v "$WORK/org.meshtastic.desktop:/work" \
|
||||
-w /work \
|
||||
--platform "$DOCKER_PLATFORM" \
|
||||
--security-opt seccomp=unconfined \
|
||||
"$BUILDER_IMAGE" bash
|
||||
exec docker run -it "${DOCKER_RUN_ARGS[@]}" "$BUILDER_IMAGE" bash
|
||||
fi
|
||||
|
||||
step "Running flatpak-builder (arch=$ARCH)"
|
||||
docker run --rm --privileged \
|
||||
-v "$WORK/org.meshtastic.desktop:/work" \
|
||||
-w /work \
|
||||
--platform "$DOCKER_PLATFORM" \
|
||||
--security-opt seccomp=unconfined \
|
||||
"$BUILDER_IMAGE" \
|
||||
bash -c "set -e
|
||||
flatpak remote-add --user --if-not-exists flathub https://dl.flathub.org/repo/flathub.flatpakrepo
|
||||
# --download-only verifies every source URL + sha256 and exits before the bwrap
|
||||
# sandbox phase. We do this because nested bwrap fails inside Docker Desktop on
|
||||
# macOS (prctl(PR_SET_SECCOMP) EINVAL). For full sandbox build, run on Linux directly
|
||||
# — or rely on vid's GHA CI which uses bare ubuntu-24.04 runners.
|
||||
flatpak-builder --user --repo=repo --install-deps-from=flathub --force-clean \
|
||||
--disable-rofiles-fuse --download-only \
|
||||
builddir org.meshtastic.desktop.yaml
|
||||
echo
|
||||
echo '=== All sources downloaded and sha256-verified successfully ==='
|
||||
"
|
||||
# Build flatpak-builder invocation. --download-only mode skips the bwrap-based
|
||||
# build phase, which is the part that fails under Docker Desktop on macOS.
|
||||
if [[ $DOWNLOAD_ONLY -eq 1 ]]; then
|
||||
BUILDER_EXTRA_FLAGS="--download-only"
|
||||
SUCCESS_MSG="All sources downloaded and sha256-verified successfully (URLs + hashes OK; Gradle build NOT exercised)"
|
||||
else
|
||||
BUILDER_EXTRA_FLAGS=""
|
||||
SUCCESS_MSG="Full offline build succeeded — flatpak-sources.json is complete and self-sufficient"
|
||||
fi
|
||||
|
||||
step "Running flatpak-builder (arch=$ARCH, mode=$([[ $DOWNLOAD_ONLY -eq 1 ]] && echo download-only || echo full-build))"
|
||||
docker run "${DOCKER_RUN_ARGS[@]}" "$BUILDER_IMAGE" bash -c "set -e
|
||||
flatpak remote-add --user --if-not-exists flathub https://dl.flathub.org/repo/flathub.flatpakrepo
|
||||
flatpak-builder --user --repo=repo --install-deps-from=flathub --force-clean \
|
||||
--disable-rofiles-fuse $BUILDER_EXTRA_FLAGS \
|
||||
builddir org.meshtastic.desktop.yaml
|
||||
echo
|
||||
echo '=== $SUCCESS_MSG ==='
|
||||
"
|
||||
|
||||
Reference in New Issue
Block a user