mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-27 17:26:03 +00:00
197 lines
6.5 KiB
Bash
Executable File
197 lines
6.5 KiB
Bash
Executable File
#!/usr/bin/env bash
|
|
set -euo pipefail
|
|
|
|
# Build the mac app bundle, then create a zip (Sparkle) + styled DMG (humans).
|
|
#
|
|
# Output:
|
|
# - dist/OpenClaw.app
|
|
# - dist/OpenClaw-<version>.zip
|
|
# - dist/OpenClaw-<version>.dmg
|
|
|
|
ROOT_DIR="$(cd "$(dirname "$0")/.." && pwd)"
|
|
source "$ROOT_DIR/scripts/lib/plistbuddy.sh"
|
|
|
|
BUILD_ROOT="$ROOT_DIR/apps/macos/.build"
|
|
PRODUCT="OpenClaw"
|
|
BUILD_CONFIG="${BUILD_CONFIG:-release}"
|
|
APP_VERSION_INPUT="${APP_VERSION:-$(cd "$ROOT_DIR" && node -p "require('./package.json').version" 2>/dev/null || echo "0.0.0")}"
|
|
|
|
# Default to universal binary for distribution builds (supports both Apple Silicon and Intel Macs)
|
|
export BUILD_ARCHS="${BUILD_ARCHS:-all}"
|
|
export BUILD_CONFIG
|
|
DSYM_ARCHS_VALUE="$BUILD_ARCHS"
|
|
if [[ "$DSYM_ARCHS_VALUE" == "all" ]]; then
|
|
DSYM_ARCHS_VALUE="arm64 x86_64"
|
|
fi
|
|
IFS=' ' read -r -a DSYM_ARCHS <<< "$DSYM_ARCHS_VALUE"
|
|
|
|
# Use release bundle ID (not .debug) so Sparkle auto-update works.
|
|
# The .debug suffix in package-mac-app.sh blanks SUFeedURL intentionally for dev builds.
|
|
export BUNDLE_ID="${BUNDLE_ID:-ai.openclaw.mac}"
|
|
|
|
canonical_sparkle_build() {
|
|
node --import tsx "$ROOT_DIR/scripts/sparkle-build.ts" canonical-build "$1"
|
|
}
|
|
|
|
correction_build_from_exact_tag() {
|
|
local version="$1"
|
|
local canonical="$2"
|
|
local tag correction highest
|
|
|
|
highest=""
|
|
while IFS= read -r tag; do
|
|
if [[ "$tag" =~ ^v${version//./\\.}-([1-9][0-9]*)$ ]]; then
|
|
correction="${BASH_REMATCH[1]}"
|
|
if [[ -z "$highest" || "$correction" -gt "$highest" ]]; then
|
|
highest="$correction"
|
|
fi
|
|
fi
|
|
done < <(git -C "$ROOT_DIR" tag --points-at HEAD 2>/dev/null || true)
|
|
|
|
if [[ -n "$highest" ]]; then
|
|
printf '%s\n' "$((canonical + highest))"
|
|
fi
|
|
}
|
|
|
|
# Local fallback releases must not silently fall back to a git-rev-count build number.
|
|
# For correction tags, pass a higher explicit APP_BUILD than the canonical floor.
|
|
if [[ -z "${APP_BUILD:-}" && "$BUILD_CONFIG" == "release" ]]; then
|
|
CANONICAL_APP_BUILD="$(canonical_sparkle_build "$APP_VERSION_INPUT" 2>/dev/null || true)"
|
|
if [[ "$CANONICAL_APP_BUILD" =~ ^[0-9]+$ ]]; then
|
|
APP_BUILD="$(correction_build_from_exact_tag "$APP_VERSION_INPUT" "$CANONICAL_APP_BUILD")"
|
|
export APP_BUILD="${APP_BUILD:-$CANONICAL_APP_BUILD}"
|
|
fi
|
|
fi
|
|
|
|
"$ROOT_DIR/scripts/package-mac-app.sh"
|
|
|
|
APP="$ROOT_DIR/dist/OpenClaw.app"
|
|
if [[ ! -d "$APP" ]]; then
|
|
echo "Error: missing app bundle at $APP" >&2
|
|
exit 1
|
|
fi
|
|
|
|
VERSION="$(plist_print_required "$APP/Contents/Info.plist" CFBundleShortVersionString)"
|
|
BUNDLE_VERSION="$(plist_print_required "$APP/Contents/Info.plist" CFBundleVersion)"
|
|
ACTUAL_BUNDLE_ID="$(plist_print_required "$APP/Contents/Info.plist" CFBundleIdentifier)"
|
|
ACTUAL_FEED_URL="$(plist_print_required "$APP/Contents/Info.plist" SUFeedURL)"
|
|
ZIP="$ROOT_DIR/dist/OpenClaw-$VERSION.zip"
|
|
DMG="$ROOT_DIR/dist/OpenClaw-$VERSION.dmg"
|
|
NOTARY_ZIP="$ROOT_DIR/dist/OpenClaw-$VERSION.notary.zip"
|
|
DSYM_ZIP="$ROOT_DIR/dist/OpenClaw-$VERSION.dSYM.zip"
|
|
SKIP_NOTARIZE="${SKIP_NOTARIZE:-0}"
|
|
NOTARIZE=1
|
|
SKIP_DSYM="${SKIP_DSYM:-0}"
|
|
SKIP_DMG="${SKIP_DMG:-0}"
|
|
|
|
if [[ "$SKIP_NOTARIZE" == "1" ]]; then
|
|
NOTARIZE=0
|
|
fi
|
|
|
|
if [[ "$BUILD_CONFIG" == "release" ]]; then
|
|
if [[ "$ACTUAL_BUNDLE_ID" != "$BUNDLE_ID" ]]; then
|
|
echo "Error: release packaging produced bundle id '$ACTUAL_BUNDLE_ID', expected '$BUNDLE_ID'." >&2
|
|
exit 1
|
|
fi
|
|
|
|
if [[ -z "$ACTUAL_FEED_URL" ]]; then
|
|
echo "Error: release packaging produced an empty SUFeedURL." >&2
|
|
exit 1
|
|
fi
|
|
|
|
CANONICAL_APP_BUILD="$(canonical_sparkle_build "$VERSION" 2>/dev/null || true)"
|
|
if [[ "$CANONICAL_APP_BUILD" =~ ^[0-9]+$ ]]; then
|
|
if [[ ! "$BUNDLE_VERSION" =~ ^[0-9]+$ ]]; then
|
|
echo "Error: release packaging produced non-numeric CFBundleVersion '$BUNDLE_VERSION'." >&2
|
|
exit 1
|
|
fi
|
|
if (( BUNDLE_VERSION < CANONICAL_APP_BUILD )); then
|
|
echo "Error: CFBundleVersion '$BUNDLE_VERSION' is below the canonical Sparkle floor '$CANONICAL_APP_BUILD' for '$VERSION'." >&2
|
|
echo "Set APP_BUILD explicitly only when you need a higher correction build." >&2
|
|
exit 1
|
|
fi
|
|
fi
|
|
fi
|
|
|
|
if [[ "$NOTARIZE" == "1" ]]; then
|
|
echo "📦 Notary zip: $NOTARY_ZIP"
|
|
rm -f "$NOTARY_ZIP"
|
|
ditto -c -k --sequesterRsrc --keepParent "$APP" "$NOTARY_ZIP"
|
|
STAPLE_APP_PATH="$APP" "$ROOT_DIR/scripts/notarize-mac-artifact.sh" "$NOTARY_ZIP"
|
|
rm -f "$NOTARY_ZIP"
|
|
fi
|
|
|
|
echo "📦 Zip: $ZIP"
|
|
rm -f "$ZIP"
|
|
ditto -c -k --sequesterRsrc --keepParent "$APP" "$ZIP"
|
|
|
|
if [[ "$SKIP_DMG" != "1" ]]; then
|
|
echo "💿 DMG: $DMG"
|
|
"$ROOT_DIR/scripts/create-dmg.sh" "$APP" "$DMG"
|
|
|
|
if [[ "$NOTARIZE" == "1" ]]; then
|
|
if [[ -n "${SIGN_IDENTITY:-}" ]]; then
|
|
echo "🔏 Signing DMG: $DMG"
|
|
/usr/bin/codesign --force --sign "$SIGN_IDENTITY" --timestamp "$DMG"
|
|
fi
|
|
"$ROOT_DIR/scripts/notarize-mac-artifact.sh" "$DMG"
|
|
fi
|
|
else
|
|
echo "💿 Skipping DMG (SKIP_DMG=1)"
|
|
fi
|
|
|
|
if [[ "$SKIP_DSYM" != "1" ]]; then
|
|
DSYM_PATHS=()
|
|
MISSING_DSYM_ARCHS=()
|
|
for arch in "${DSYM_ARCHS[@]}"; do
|
|
if [[ ! -d "$BUILD_ROOT/$arch" ]]; then
|
|
MISSING_DSYM_ARCHS+=("$arch")
|
|
continue
|
|
fi
|
|
DSYM_FOR_ARCH="$(find "$BUILD_ROOT/$arch" -type d -path "*/$BUILD_CONFIG/$PRODUCT.dSYM" -print -quit)"
|
|
if [[ -n "$DSYM_FOR_ARCH" ]]; then
|
|
DSYM_PATHS+=("$DSYM_FOR_ARCH")
|
|
else
|
|
MISSING_DSYM_ARCHS+=("$arch")
|
|
fi
|
|
done
|
|
|
|
if [[ "${#MISSING_DSYM_ARCHS[@]}" -gt 0 ]]; then
|
|
echo "Error: dSYM not found for architecture(s): ${MISSING_DSYM_ARCHS[*]} (set SKIP_DSYM=1 to skip symbols)" >&2
|
|
exit 1
|
|
fi
|
|
|
|
if [[ "${#DSYM_PATHS[@]}" -gt 0 ]]; then
|
|
TMP_DSYM="$ROOT_DIR/dist/$PRODUCT.dSYM"
|
|
rm -rf "$TMP_DSYM"
|
|
if [[ "${#DSYM_PATHS[@]}" -gt 1 ]]; then
|
|
cp -R "${DSYM_PATHS[0]}" "$TMP_DSYM"
|
|
DWARF_OUT="$TMP_DSYM/Contents/Resources/DWARF/$PRODUCT"
|
|
DWARF_INPUTS=()
|
|
for dsym in "${DSYM_PATHS[@]}"; do
|
|
DWARF_INPUT="$dsym/Contents/Resources/DWARF/$PRODUCT"
|
|
if [[ ! -f "$DWARF_INPUT" ]]; then
|
|
echo "Error: missing DWARF binaries for dSYM merge (set SKIP_DSYM=1 to skip symbols)" >&2
|
|
exit 1
|
|
fi
|
|
DWARF_INPUTS+=("$DWARF_INPUT")
|
|
done
|
|
if [[ "${#DWARF_INPUTS[@]}" -gt 1 ]]; then
|
|
/usr/bin/lipo -create "${DWARF_INPUTS[@]}" -output "$DWARF_OUT"
|
|
else
|
|
echo "Error: missing DWARF binaries for dSYM merge (set SKIP_DSYM=1 to skip symbols)" >&2
|
|
exit 1
|
|
fi
|
|
else
|
|
cp -R "${DSYM_PATHS[0]}" "$TMP_DSYM"
|
|
fi
|
|
echo "🧩 dSYM: $DSYM_ZIP"
|
|
rm -f "$DSYM_ZIP"
|
|
ditto -c -k --keepParent "$TMP_DSYM" "$DSYM_ZIP"
|
|
rm -rf "$TMP_DSYM"
|
|
else
|
|
echo "Error: dSYM not found (set SKIP_DSYM=1 to skip symbols)" >&2
|
|
exit 1
|
|
fi
|
|
fi
|