- 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
495 lines
17 KiB
Bash
Executable File
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é !" |