Files
geo/app/android.sh
Pierre 0687900564 fix: Récupérer l'opération active depuis la table operations
- Corrige l'erreur SQL 'Unknown column fk_operation in users'
- L'opération active est récupérée depuis operations.chk_active = 1
- Jointure avec users pour filtrer par entité de l'admin créateur
- Query: SELECT o.id FROM operations o INNER JOIN users u ON u.fk_entite = o.fk_entite WHERE u.id = ? AND o.chk_active = 1
2026-01-26 16:57:08 +01:00

495 lines
17 KiB
Bash
Executable File

#!/bin/bash
# Script de génération du bundle Android pour GEOSECTOR
# Usage: ./android.sh
set -e # Arrêter le script en cas d'erreur
# Couleurs pour les messages
RED='\033[0;31m'
GREEN='\033[0;32m'
YELLOW='\033[1;33m'
BLUE='\033[0;34m'
NC='\033[0m' # No Color
# Fonction pour afficher les messages
print_message() {
echo -e "${BLUE}[INFO]${NC} $1"
}
print_success() {
echo -e "${GREEN}[SUCCESS]${NC} $1"
}
print_error() {
echo -e "${RED}[ERROR]${NC} $1"
}
print_warning() {
echo -e "${YELLOW}[WARNING]${NC} $1"
}
# Fonction pour gérer les erreurs
handle_error() {
print_error "Une erreur est survenue lors de l'exécution de la commande"
print_error "Ligne $1"
exit 1
}
# Trap pour capturer les erreurs
trap 'handle_error $LINENO' ERR
# Variables pour IN1 (compilation distante)
IN1_HOST="IN1"
IN1_PROJECT_PATH="/home/pierre/dev/geosector/app"
print_message "🖥️ Compilation distante sur IN1:${IN1_PROJECT_PATH}"
echo
print_message "========================================="
print_message " GEOSECTOR - Génération Bundle Android"
print_message "========================================="
echo
# Étape 0 : Synchroniser la version depuis ../VERSION (lecture locale via SSHFS)
print_message "Étape 0/5 : Synchronisation de la version..."
echo
VERSION_FILE="../VERSION"
if [ ! -f "$VERSION_FILE" ]; then
print_error "Fichier VERSION introuvable : $VERSION_FILE"
exit 1
fi
# Lire la version depuis le fichier (enlever espaces/retours à la ligne)
VERSION_NUMBER=$(cat "$VERSION_FILE" | tr -d '\n\r ' | tr -d '[:space:]')
if [ -z "$VERSION_NUMBER" ]; then
print_error "Le fichier VERSION est vide"
exit 1
fi
print_message "Version lue depuis $VERSION_FILE : $VERSION_NUMBER"
# Calculer le versionCode (supprimer les points)
VERSION_CODE=$(echo $VERSION_NUMBER | tr -d '.')
if [ -z "$VERSION_CODE" ]; then
print_error "Impossible de calculer le versionCode"
exit 1
fi
print_message "Version code calculé : $VERSION_CODE"
# Mettre à jour pubspec.yaml
print_message "Mise à jour de pubspec.yaml..."
sed -i.bak "s/^version:.*/version: $VERSION_NUMBER+$VERSION_CODE/" pubspec.yaml
# Vérifier que la mise à jour a réussi
UPDATED_VERSION=$(grep "^version:" pubspec.yaml | sed 's/version: //')
if [ "$UPDATED_VERSION" != "$VERSION_NUMBER+$VERSION_CODE" ]; then
print_error "Échec de la mise à jour de pubspec.yaml"
print_error "Attendu : $VERSION_NUMBER+$VERSION_CODE"
print_error "Obtenu : $UPDATED_VERSION"
exit 1
fi
print_success "pubspec.yaml mis à jour : version: $VERSION_NUMBER+$VERSION_CODE"
print_message "build.gradle.kts se synchronisera automatiquement via Flutter Gradle Plugin"
echo
# Récupérer la version finale pour l'affichage
VERSION="$VERSION_NUMBER-$VERSION_CODE"
print_message "Version finale : $VERSION"
print_message "Version code : $VERSION_CODE"
echo
# Demander le mode Debug ou Release
print_message "========================================="
print_message " MODE DE BUILD"
print_message "========================================="
echo
print_message "Choisissez le mode de build :"
echo
print_message " ${YELLOW}[D]${NC} Debug"
print_message " ✓ Installation rapide via ADB"
print_message " ✓ Hot reload possible"
print_message " ✓ Logs complets"
print_message " ⚠ Tap to Pay simulé uniquement"
print_message " ⚠ Performance non optimisée"
echo
print_message " ${GREEN}[R]${NC} Release (recommandé)"
print_message " ✓ APK/AAB optimisé"
print_message " ✓ Tap to Pay réel en production"
print_message " ✓ Performance maximale"
echo
read -p "Votre choix (D/R) [défaut: R] : " -n 1 -r BUILD_TYPE
echo
echo
# Définir le flag de build et le suffixe pour les noms de fichiers
BUILD_MODE_FLAG="--release"
MODE_SUFFIX="release"
SKIP_R8_CHOICE=false
if [[ $BUILD_TYPE =~ ^[Dd]$ ]]; then
BUILD_MODE_FLAG="--debug"
MODE_SUFFIX="debug"
SKIP_R8_CHOICE=true
print_success "Mode Debug sélectionné"
echo
print_warning "Attention : Tap to Pay ne fonctionnera qu'en mode simulé"
echo
# En mode debug, pas de choix R8 ni de vérification keystore
USE_R8=false
COPY_DEBUG_FILES=false
else
print_success "Mode Release sélectionné"
echo
fi
# Demander le mode R8 SEULEMENT si Release
if [ "$SKIP_R8_CHOICE" = false ]; then
print_message "========================================="
print_message " OPTIMISATION RELEASE"
print_message "========================================="
echo
print_message "Choisissez le niveau d'optimisation :"
echo
print_message " ${GREEN}[A]${NC} Production - R8/ProGuard activé"
print_message " ✓ Taille réduite (~30-40%)"
print_message " ✓ Code obscurci (sécurité)"
print_message " ✓ Génère mapping.txt pour débogage"
print_message " ✓ Génère symboles natifs"
echo
print_message " ${YELLOW}[B]${NC} Test interne - Sans R8/ProGuard (défaut)"
print_message " ✓ Build plus rapide"
print_message " ✓ Pas d'obscurcissement (débogage facile)"
print_message " ⚠ Taille plus importante"
print_message " ⚠ Avertissements Google Play Console"
echo
read -p "Votre choix (A/B) [défaut: B] : " -n 1 -r BUILD_MODE
echo
echo
# Définir les variables selon le choix
USE_R8=false
COPY_DEBUG_FILES=false
if [[ $BUILD_MODE =~ ^[Aa]$ ]]; then
USE_R8=true
COPY_DEBUG_FILES=true
print_success "Mode Production sélectionné - R8/ProGuard activé"
else
print_success "Mode Test interne sélectionné - R8/ProGuard désactivé"
fi
echo
fi
# Vérifier la présence du keystore SEULEMENT si Release
if [ "$SKIP_R8_CHOICE" = false ]; then
if [ ! -f "android/app/geosector2025.jks" ]; then
print_error "Fichier keystore introuvable : android/app/geosector2025.jks"
exit 1
fi
# Vérifier la présence du fichier key.properties
if [ ! -f "android/key.properties" ]; then
print_error "Fichier key.properties introuvable"
print_error "Ce fichier est nécessaire pour signer l'application"
exit 1
fi
print_success "Configuration de signature vérifiée"
echo
fi
# Activer R8 si demandé (modification temporaire du build.gradle.kts)
GRADLE_FILE="android/app/build.gradle.kts"
GRADLE_BACKUP="android/app/build.gradle.kts.backup"
if [ "$USE_R8" = true ]; then
print_message "Activation de R8/ProGuard dans build.gradle.kts..."
# Créer une sauvegarde
cp "$GRADLE_FILE" "$GRADLE_BACKUP"
# Activer minifyEnabled et shrinkResources
sed -i.tmp 's/isMinifyEnabled = false/isMinifyEnabled = true/' "$GRADLE_FILE"
sed -i.tmp 's/isShrinkResources = false/isShrinkResources = true/' "$GRADLE_FILE"
# Nettoyer les fichiers temporaires de sed
rm -f "${GRADLE_FILE}.tmp"
print_success "R8/ProGuard activé temporairement"
echo
fi
# Lancer le build sur IN1 via SSH
print_message "🚀 Lancement du build Android sur IN1..."
print_message "Cette opération peut prendre plusieurs minutes..."
echo
ssh ${IN1_HOST} "bash -l -s" <<REMOTE_SCRIPT || { print_error "Remote build failed on IN1"; exit 1; }
set -euo pipefail
# Charger le profil utilisateur pour avoir Flutter dans le PATH
if [ -f ~/.bashrc ]; then
source ~/.bashrc
fi
if [ -f ~/.profile ]; then
source ~/.profile
fi
cd ${IN1_PROJECT_PATH}
echo "[IN1] Starting Android build process..."
# Variables transmises
VERSION_NUMBER="${VERSION_NUMBER}"
VERSION_CODE="${VERSION_CODE}"
BUILD_MODE_FLAG="${BUILD_MODE_FLAG}"
MODE_SUFFIX="${MODE_SUFFIX}"
USE_R8="${USE_R8}"
GRADLE_FILE="android/app/build.gradle.kts"
GRADLE_BACKUP="android/app/build.gradle.kts.backup"
# Mise à jour de pubspec.yaml
echo "[IN1] Updating pubspec.yaml to version \${VERSION_NUMBER}+\${VERSION_CODE}..."
sed -i.bak "s/^version:.*/version: \${VERSION_NUMBER}+\${VERSION_CODE}/" pubspec.yaml
# Activer R8 si demandé
if [ "\$USE_R8" = "true" ]; then
echo "[IN1] Enabling R8/ProGuard..."
cp "\$GRADLE_FILE" "\$GRADLE_BACKUP"
sed -i.tmp 's/isMinifyEnabled = false/isMinifyEnabled = true/' "\$GRADLE_FILE"
sed -i.tmp 's/isShrinkResources = false/isShrinkResources = true/' "\$GRADLE_FILE"
rm -f "\${GRADLE_FILE}.tmp"
fi
# Étape 1 : Nettoyer
echo "[IN1] Step 1/5: Cleaning project..."
flutter clean
# Étape 2 : Dépendances
echo "[IN1] Step 2/5: Getting dependencies..."
flutter pub get
# Étape 2.5 : Patch nfc_manager
echo "[IN1] Step 2.5/5: Patching nfc_manager..."
./fastlane/scripts/commun/fix-nfc-manager.sh || echo "[IN1] Warning: nfc_manager patch failed"
# Étape 3 : Analyse (optionnel)
echo "[IN1] Step 3/5: Analyzing code..."
flutter analyze --no-fatal-infos --no-fatal-warnings || echo "[IN1] Warning: Code analysis had warnings"
# Étape 4 : Build bundle
echo "[IN1] Step 4/5: Building Android App Bundle..."
flutter build appbundle \$BUILD_MODE_FLAG
# Restaurer gradle si modifié
if [ "\$USE_R8" = "true" ] && [ -f "\$GRADLE_BACKUP" ]; then
echo "[IN1] Restoring original build.gradle.kts..."
mv "\$GRADLE_BACKUP" "\$GRADLE_FILE"
fi
echo "[IN1] Build completed successfully!"
REMOTE_SCRIPT
print_success "Build Android terminé sur IN1"
echo
# Télécharger le bundle depuis IN1
BUNDLE_PATH="build/app/outputs/bundle/$MODE_SUFFIX/app-$MODE_SUFFIX.aab"
FINAL_NAME="geosector-$VERSION_CODE-$MODE_SUFFIX.aab"
print_message "📥 Téléchargement du bundle depuis IN1..."
scp ${IN1_HOST}:${IN1_PROJECT_PATH}/${BUNDLE_PATH} "$FINAL_NAME" || {
print_error "Failed to download bundle from IN1"
exit 1
}
if [ -f "$FINAL_NAME" ]; then
print_success "Bundle copié avec succès"
# Afficher la taille du fichier
FILE_SIZE=$(du -h "$FINAL_NAME" | cut -f1)
print_message "Taille du bundle : $FILE_SIZE"
else
print_error "Échec de la copie du bundle"
exit 1
fi
# Télécharger les fichiers de débogage si Option A sélectionnée
if [ "$COPY_DEBUG_FILES" = true ]; then
echo
print_message "📥 Téléchargement des fichiers de débogage depuis IN1..."
# Créer un dossier de release
RELEASE_DIR="release-$VERSION_CODE"
mkdir -p "$RELEASE_DIR"
# Copier le bundle
cp "$FINAL_NAME" "$RELEASE_DIR/"
# Télécharger le fichier mapping.txt (R8/ProGuard)
MAPPING_FILE="build/app/outputs/mapping/release/mapping.txt"
print_message "Downloading mapping.txt..."
scp ${IN1_HOST}:${IN1_PROJECT_PATH}/${MAPPING_FILE} "$RELEASE_DIR/mapping.txt" 2>/dev/null && {
print_success "Fichier mapping.txt téléchargé"
} || {
print_warning "Fichier mapping.txt introuvable sur IN1 (peut être normal)"
}
# Créer l'archive des symboles natifs sur IN1 et la télécharger
print_message "Creating and downloading native symbols..."
ssh ${IN1_HOST} "bash -l -c '
cd ${IN1_PROJECT_PATH}/build/app/intermediates/merged_native_libs/release/out 2>/dev/null &&
zip -r /tmp/native-symbols-$VERSION_CODE.zip lib/ &&
echo \"Symbols archived\"
'" 2>/dev/null && {
scp ${IN1_HOST}:/tmp/native-symbols-$VERSION_CODE.zip "$RELEASE_DIR/native-symbols.zip" && {
print_success "Symboles natifs téléchargés"
ssh ${IN1_HOST} "rm -f /tmp/native-symbols-$VERSION_CODE.zip"
}
} || {
print_warning "Symboles natifs introuvables (peut être normal)"
}
print_success "Fichiers de débogage dans : $RELEASE_DIR/"
echo
print_message "Pour uploader sur Google Play Console :"
print_message "1. Bundle : $RELEASE_DIR/$FINAL_NAME"
print_message "2. Mapping : $RELEASE_DIR/mapping.txt"
print_message "3. Symboles : $RELEASE_DIR/native-symbols.zip"
fi
echo
print_message "========================================="
print_success " GÉNÉRATION TERMINÉE AVEC SUCCÈS !"
print_message "========================================="
echo
print_message "Bundle généré : ${GREEN}$FINAL_NAME${NC}"
print_message "Version : $VERSION"
print_message "Chemin : $(pwd)/$FINAL_NAME"
if [ "$BUILD_MODE_FLAG" = "--debug" ]; then
echo
print_message "Mode : ${YELLOW}Debug${NC}"
print_message "⚠ Tap to Pay simulé uniquement"
print_message "✓ Logs complets disponibles"
echo
print_message "Prochaines étapes :"
print_message "1. Installer l'APK sur l'appareil (proposé ci-dessous)"
print_message "2. Tester l'application avec adb logcat"
print_message "3. Pour Tap to Pay réel, relancer en mode Release"
elif [ "$USE_R8" = true ]; then
echo
print_message "Mode : ${GREEN}Release - Production (R8/ProGuard activé)${NC}"
print_message "Dossier release : ${GREEN}$RELEASE_DIR/${NC}"
echo
print_message "Prochaines étapes :"
print_message "1. Tester le bundle sur un appareil Android"
print_message "2. Uploader le bundle sur Google Play Console"
print_message "3. Uploader mapping.txt et native-symbols.zip"
print_message "4. Soumettre pour review"
else
echo
print_message "Mode : ${GREEN}Release${NC} - ${YELLOW}Test interne (R8/ProGuard désactivé)${NC}"
print_warning "Avertissements attendus sur Google Play Console"
echo
print_message "Prochaines étapes :"
print_message "1. Tester le bundle sur un appareil Android"
print_message "2. Uploader sur Google Play Console (test interne)"
print_message "3. Pour production, relancer avec Option A"
fi
echo
# Optionnel : Générer aussi l'APK
read -p "Voulez-vous aussi générer l'APK pour des tests ? (y/n) " -n 1 -r
echo
if [[ $REPLY =~ ^[Yy]$ ]]; then
print_message "Génération de l'APK sur IN1..."
ssh ${IN1_HOST} "bash -l -s" <<REMOTE_APK_SCRIPT || { print_error "Remote APK build failed"; }
set -euo pipefail
if [ -f ~/.bashrc ]; then source ~/.bashrc; fi
if [ -f ~/.profile ]; then source ~/.profile; fi
cd ${IN1_PROJECT_PATH}
echo "[IN1] Building APK..."
flutter build apk ${BUILD_MODE_FLAG}
REMOTE_APK_SCRIPT
if [ $? -eq 0 ]; then
APK_PATH="build/app/outputs/flutter-apk/app-$MODE_SUFFIX.apk"
APK_NAME="geosector-$VERSION_CODE-$MODE_SUFFIX.apk"
print_message "📥 Téléchargement de l'APK depuis IN1..."
scp ${IN1_HOST}:${IN1_PROJECT_PATH}/${APK_PATH} "$APK_NAME" || {
print_error "Failed to download APK from IN1"
}
if [ -f "$APK_NAME" ]; then
print_success "APK téléchargé : $APK_NAME"
# Afficher la taille de l'APK
APK_SIZE=$(du -h "$APK_NAME" | cut -f1)
print_message "Taille de l'APK : $APK_SIZE"
# Si mode Debug, proposer installation automatique (sur machine locale)
if [ "$BUILD_MODE_FLAG" = "--debug" ]; then
echo
print_message "💡 L'APK est maintenant disponible localement : $APK_NAME"
read -p "Installer l'APK debug sur l'appareil connecté à CE poste ? (y/n) " -n 1 -r
echo
if [[ $REPLY =~ ^[Yy]$ ]]; then
# Vérifier qu'ADB est disponible localement
if ! command -v adb &> /dev/null; then
print_error "ADB n'est pas installé sur ce poste"
print_message "Installez Android SDK Platform Tools"
else
# Vérifier qu'un appareil est connecté
print_message "Vérification des appareils connectés..."
DEVICES=$(adb devices | grep -v "List" | grep "device$" | wc -l)
if [ "$DEVICES" -eq 0 ]; then
print_error "Aucun appareil Android détecté en USB sur ce poste"
print_message "Vérifiez la connexion USB et activez le débogage USB"
print_message "Commande : adb devices"
else
print_success "$DEVICES appareil(s) détecté(s)"
print_message "Désinstallation de l'ancienne version..."
adb uninstall fr.geosector.app3 2>/dev/null || print_warning "Aucune version précédente trouvée"
print_message "Installation sur l'appareil..."
adb install "$APK_NAME"
if [ $? -eq 0 ]; then
print_success "APK installé avec succès sur l'appareil connecté"
# Proposer de lancer l'app
read -p "Lancer l'application ? (y/n) " -n 1 -r
echo
if [[ $REPLY =~ ^[Yy]$ ]]; then
adb shell am start -n fr.geosector.app3/.MainActivity
if [ $? -eq 0 ]; then
print_success "Application lancée"
else
print_warning "Impossible de lancer l'application"
fi
fi
else
print_error "Échec de l'installation"
print_message "Commande : adb install $APK_NAME"
fi
fi
fi
fi
fi
fi
else
print_warning "Échec de la génération de l'APK (le bundle a été créé avec succès)"
fi
fi
echo
print_success "Script terminé !"