diff --git a/.gitignore b/.gitignore
index cd0ae6af..4963614a 100644
--- a/.gitignore
+++ b/.gitignore
@@ -20,6 +20,12 @@ web/storage/*.key
web/public/storage
web/public/hot
+# API
+api/sessions/
+api/logs/
+api/uploads/
+api/vendor/
+
# Général
*.DS_Store
*.log
diff --git a/MONITORING.md b/MONITORING.md
new file mode 100644
index 00000000..9ca99654
--- /dev/null
+++ b/MONITORING.md
@@ -0,0 +1,608 @@
+# Guide de Monitoring - Container Incus (Application Web PHP + MariaDB)
+
+## Vue d'ensemble
+
+Ce guide décrit les métriques essentielles à surveiller pour un container Incus hébergeant une application web PHP avec API mobile et base de données MariaDB.
+
+---
+
+## 1. Ressources Système
+
+### CPU
+**Pourquoi ?** Identifier les pics de charge et les processus gourmands
+
+```bash
+# Utilisation CPU globale du container
+incus info nx4 | grep "CPU utilisé"
+
+# Détail par processus (dans le container)
+top -bn1 | head -20
+htop
+```
+
+**Métriques à surveiller :**
+- Load average (devrait être < nombre de CPU)
+- % CPU par processus (MariaDB, PHP-FPM, nginx)
+
+**Seuils d'alerte :**
+- ⚠️ Warning : Load average > 70% des CPU
+- 🚨 Critical : Load average > 150% des CPU pendant >5min
+
+---
+
+### Mémoire RAM
+**Pourquoi ?** Éviter les OOM (Out of Memory) qui tuent les processus
+
+```bash
+# Vue globale depuis le host
+incus info nx4 | grep "Mémoire"
+
+# Détail dans le container
+free -h
+ps aux --sort=-%mem | head -10
+```
+
+**Métriques à surveiller :**
+- RAM utilisée / totale
+- Swap utilisé (devrait rester minimal)
+- Top processus consommateurs
+
+**Seuils d'alerte :**
+- ⚠️ Warning : RAM > 85%
+- 🚨 Critical : RAM > 95% ou Swap > 1GB
+
+---
+
+### Disque I/O
+**Pourquoi ?** MariaDB est très sensible aux lenteurs disque
+
+```bash
+# Sur le host
+iostat -x 2 5
+
+# Dans le container
+iostat -x 1 3
+iotop -oa
+```
+
+**Métriques à surveiller :**
+- Latence disque (await)
+- IOPS (r/s, w/s)
+- % utilisation disque
+
+**Seuils d'alerte :**
+- ⚠️ Warning : await > 50ms
+- 🚨 Critical : await > 200ms ou %util > 90%
+
+---
+
+### Espace Disque
+**Pourquoi ?** MariaDB ne peut plus écrire si disque plein
+
+```bash
+df -h
+du -sh /var/lib/mysql
+```
+
+**Seuils d'alerte :**
+- ⚠️ Warning : > 80% utilisé
+- 🚨 Critical : > 90% utilisé
+
+---
+
+## 2. PHP-FPM (Application Web)
+
+### Pool de Workers
+**Pourquoi ?** Cause #1 des timeouts et coupures de service (votre cas !)
+
+```bash
+# Nombre de workers actifs
+ps aux | grep "php-fpm: pool www" | wc -l
+
+# Config du pool
+grep "^pm" /etc/php/8.3/fpm/pool.d/www.conf
+
+# Logs d'alertes
+tail -f /var/log/php8.3-fpm.log | grep "max_children"
+```
+
+**Métriques critiques :**
+- Nombre de workers actifs vs `pm.max_children`
+- Warnings "max_children reached"
+- Slow requests (>2s)
+
+**Seuils d'alerte :**
+- ⚠️ Warning : Workers actifs > 80% de max_children
+- 🚨 Critical : Warning "max_children reached" apparaît
+
+**Configuration recommandée pour votre cas :**
+```ini
+pm = dynamic
+pm.max_children = 50-100 (selon RAM disponible)
+pm.start_servers = 10-20
+pm.min_spare_servers = 5-10
+pm.max_spare_servers = 20-35
+pm.max_requests = 500
+```
+
+---
+
+### Temps de Réponse PHP
+**Pourquoi ?** Scripts lents = workers bloqués
+
+```bash
+# Activer le slow log PHP-FPM
+# Dans /etc/php/8.3/fpm/pool.d/www.conf
+slowlog = /var/log/php8.3-fpm-slow.log
+request_slowlog_timeout = 3s
+
+# Voir les requêtes lentes
+tail -f /var/log/php8.3-fpm-slow.log
+```
+
+**Seuils d'alerte :**
+- ⚠️ Warning : Requêtes > 3s
+- 🚨 Critical : Requêtes > 10s ou timeouts fréquents
+
+---
+
+## 3. MariaDB / MySQL
+
+### Connexions
+**Pourquoi ?** Trop de connexions = refus de nouvelles connexions
+
+```bash
+mysql -e "SHOW STATUS LIKE 'Threads_connected';"
+mysql -e "SHOW STATUS LIKE 'Max_used_connections';"
+mysql -e "SHOW VARIABLES LIKE 'max_connections';"
+mysql -e "SHOW FULL PROCESSLIST;"
+```
+
+**Métriques critiques :**
+- Connexions actives vs max_connections
+- Connexions en attente / bloquées
+- Requêtes longues (>5s)
+
+**Seuils d'alerte :**
+- ⚠️ Warning : Connexions > 80% de max_connections
+- 🚨 Critical : Connexions = max_connections
+
+**Config recommandée :**
+```ini
+max_connections = 200-500 (selon votre trafic)
+```
+
+---
+
+### Slow Queries
+**Pourquoi ?** Requêtes lentes = workers PHP bloqués
+
+```bash
+# Activer le slow query log
+mysql -e "SET GLOBAL slow_query_log = 'ON';"
+mysql -e "SET GLOBAL long_query_time = 2;"
+mysql -e "SET GLOBAL log_queries_not_using_indexes = 'ON';"
+
+# Voir les slow queries
+tail -f /var/lib/mysql/slow-query.log
+# ou
+mysqldumpslow /var/lib/mysql/slow-query.log
+```
+
+**Seuils d'alerte :**
+- ⚠️ Warning : Requêtes > 2s
+- 🚨 Critical : Requêtes > 10s ou >50 slow queries/min
+
+---
+
+### InnoDB Buffer Pool
+**Pourquoi ?** Si trop petit, beaucoup de lectures disque (lent)
+
+```bash
+mysql -e "SHOW VARIABLES LIKE 'innodb_buffer_pool_size';"
+mysql -e "SHOW STATUS LIKE 'Innodb_buffer_pool_read%';"
+```
+
+**Métriques critiques :**
+- Buffer pool hit ratio (devrait être >99%)
+- Read requests vs reads from disk
+
+**Config recommandée :**
+- `innodb_buffer_pool_size` = 70-80% de la RAM dédiée à MySQL
+- Pour 20GB de données : `innodb_buffer_pool_size = 16G`
+
+---
+
+### Deadlocks et Locks
+**Pourquoi ?** Peuvent causer des timeouts
+
+```bash
+mysql -e "SHOW ENGINE INNODB STATUS\G" | grep -A 50 "LATEST DETECTED DEADLOCK"
+mysql -e "SHOW OPEN TABLES WHERE In_use > 0;"
+```
+
+**Seuils d'alerte :**
+- ⚠️ Warning : >1 deadlock/heure
+- 🚨 Critical : Tables lockées >30s
+
+---
+
+## 4. Nginx / Serveur Web
+
+### Connexions et Erreurs
+**Pourquoi ?** Identifier les 502/504 (backend timeout)
+
+```bash
+# Connexions actives
+netstat -an | grep :80 | wc -l
+netstat -an | grep :443 | wc -l
+
+# Erreurs récentes
+tail -100 /var/log/nginx/error.log | grep -E "502|504|timeout"
+tail -100 /var/log/nginx/access.log | grep -E " 502 | 504 "
+```
+
+**Métriques critiques :**
+- Erreurs 502 Bad Gateway (PHP-FPM down)
+- Erreurs 504 Gateway Timeout (PHP-FPM trop lent)
+- Connexions actives
+
+**Seuils d'alerte :**
+- ⚠️ Warning : >5 erreurs 502/504 en 5min
+- 🚨 Critical : >20 erreurs 502/504 en 5min
+
+---
+
+### Worker Connections
+**Pourquoi ?** Limite le nombre de connexions simultanées
+
+```bash
+# Config nginx
+grep worker_connections /etc/nginx/nginx.conf
+ps aux | grep nginx | wc -l
+```
+
+**Config recommandée :**
+```nginx
+worker_processes auto;
+worker_connections 2048;
+```
+
+---
+
+## 5. Réseau
+
+### Bande Passante
+**Pourquoi ?** Identifier les pics de trafic
+
+```bash
+# Depuis le host
+incus info nx4 | grep -A 10 "eth0:"
+
+# Dans le container
+iftop -i eth0
+vnstat -i eth0
+```
+
+**Métriques :**
+- Octets reçus/émis
+- Paquets reçus/émis
+- Erreurs réseau
+
+---
+
+### Connexions TCP
+**Pourquoi ?** Trop de connexions TIME_WAIT = problème
+
+```bash
+netstat -an | grep -E "ESTABLISHED|TIME_WAIT|CLOSE_WAIT" | wc -l
+ss -s
+```
+
+**Seuils d'alerte :**
+- ⚠️ Warning : >1000 connexions TIME_WAIT
+- 🚨 Critical : >5000 connexions TIME_WAIT
+
+---
+
+## 6. Logs et Événements
+
+### Logs Système
+```bash
+# Container Debian
+journalctl -u nginx -n 100
+journalctl -u php8.3-fpm -n 100
+journalctl -u mariadb -n 100
+
+# Container Alpine (si pas de systemd)
+tail -100 /var/log/messages
+```
+
+### Logs Applicatifs
+```bash
+# PHP errors
+tail -100 /var/log/php8.3-fpm.log
+tail -100 /var/www/html/logs/*.log
+
+# Nginx
+tail -100 /var/log/nginx/error.log
+tail -100 /var/log/nginx/access.log
+```
+
+---
+
+## 7. Scripts de Monitoring Automatisés
+
+### Script de Monitoring Global
+
+Créer `/root/monitor.sh` :
+
+```bash
+#!/bin/bash
+
+LOG_FILE="/var/log/system-monitor.log"
+ALERT_FILE="/var/log/alerts.log"
+
+echo "=== Monitoring $(date) ===" >> $LOG_FILE
+
+# 1. CPU
+LOAD=$(uptime | awk -F'load average:' '{print $2}' | awk '{print $1}')
+echo "Load: $LOAD" >> $LOG_FILE
+
+# 2. RAM
+RAM_PERCENT=$(free | awk '/Mem:/ {printf("%.1f"), ($3/$2)*100}')
+echo "RAM: ${RAM_PERCENT}%" >> $LOG_FILE
+if (( $(echo "$RAM_PERCENT > 85" | bc -l) )); then
+ echo "[WARNING] RAM > 85%: ${RAM_PERCENT}%" >> $ALERT_FILE
+fi
+
+# 3. PHP-FPM Workers
+PHP_WORKERS=$(ps aux | grep "php-fpm: pool www" | wc -l)
+PHP_MAX=$(grep "^pm.max_children" /etc/php/8.3/fpm/pool.d/www.conf | awk '{print $3}')
+echo "PHP Workers: $PHP_WORKERS / $PHP_MAX" >> $LOG_FILE
+if [ $PHP_WORKERS -gt $(echo "$PHP_MAX * 0.8" | bc) ]; then
+ echo "[WARNING] PHP Workers > 80%: $PHP_WORKERS / $PHP_MAX" >> $ALERT_FILE
+fi
+
+# 4. MySQL Connexions
+MYSQL_CONN=$(mysql -e "SHOW STATUS LIKE 'Threads_connected';" | awk 'NR==2 {print $2}')
+MYSQL_MAX=$(mysql -e "SHOW VARIABLES LIKE 'max_connections';" | awk 'NR==2 {print $2}')
+echo "MySQL Connections: $MYSQL_CONN / $MYSQL_MAX" >> $LOG_FILE
+if [ $MYSQL_CONN -gt $(echo "$MYSQL_MAX * 0.8" | bc) ]; then
+ echo "[WARNING] MySQL Connections > 80%: $MYSQL_CONN / $MYSQL_MAX" >> $ALERT_FILE
+fi
+
+# 5. Disque
+DISK_PERCENT=$(df -h / | awk 'NR==2 {print $5}' | sed 's/%//')
+echo "Disk: ${DISK_PERCENT}%" >> $LOG_FILE
+if [ $DISK_PERCENT -gt 80 ]; then
+ echo "[WARNING] Disk > 80%: ${DISK_PERCENT}%" >> $ALERT_FILE
+fi
+
+# 6. Erreurs nginx
+NGINX_ERRORS=$(grep -c "error" /var/log/nginx/error.log | tail -100)
+if [ $NGINX_ERRORS -gt 10 ]; then
+ echo "[WARNING] Nginx errors: $NGINX_ERRORS in last 100 lines" >> $ALERT_FILE
+fi
+
+echo "" >> $LOG_FILE
+```
+
+**Installation :**
+```bash
+chmod +x /root/monitor.sh
+
+# Exécuter toutes les 5 minutes
+(crontab -l 2>/dev/null; echo "*/5 * * * * /root/monitor.sh") | crontab -
+```
+
+---
+
+### Script de Monitoring PHP-FPM Spécifique
+
+Créer `/root/monitor-php-fpm.sh` :
+
+```bash
+#!/bin/bash
+
+LOG="/var/log/php-fpm-monitor.log"
+TIMESTAMP=$(date '+%Y-%m-%d %H:%M:%S')
+
+# Compter les workers (sans grep lui-même)
+WORKERS=$(ps aux | grep 'php-fpm: pool www' | grep -v grep | wc -l)
+MAX_CHILDREN=$(grep '^pm.max_children' /etc/php/8.3/fpm/pool.d/www.conf | awk '{print $3}')
+PERCENT=$(echo "scale=1; ($WORKERS / $MAX_CHILDREN) * 100" | bc)
+
+# Log au format: timestamp,workers,max,percentage
+echo "$TIMESTAMP,$WORKERS,$MAX_CHILDREN,$PERCENT%" >> $LOG
+
+# Alerte si >80%
+if (( $(echo "$PERCENT > 80" | bc -l) )); then
+ echo "[$TIMESTAMP] [ALERT] PHP Workers > 80%: $WORKERS / $MAX_CHILDREN ($PERCENT%)" >> /var/log/alerts.log
+fi
+
+# Vérifier si max_children atteint dans les logs récents
+if tail -10 /var/log/php8.3-fpm.log | grep -q "max_children"; then
+ echo "[$TIMESTAMP] [CRITICAL] MAX_CHILDREN REACHED!" >> /var/log/alerts.log
+ tail -3 /var/log/php8.3-fpm.log >> /var/log/alerts.log
+fi
+```
+
+**Installation :**
+```bash
+chmod +x /root/monitor-php-fpm.sh
+
+# Exécuter toutes les minutes
+(crontab -l 2>/dev/null; echo "* * * * * /root/monitor-php-fpm.sh") | crontab -
+```
+
+**Visualiser les données :**
+```bash
+# Afficher les dernières 60 minutes
+tail -60 /var/log/php-fpm-monitor.log
+
+# Voir l'évolution graphique (nécessite gnuplot)
+echo 'set datafile separator ","; plot "/var/log/php-fpm-monitor.log" using 2 with lines title "Workers"' | gnuplot -p
+
+# Statistiques rapides
+echo "Max workers last hour: $(tail -60 /var/log/php-fpm-monitor.log | cut -d',' -f2 | sort -n | tail -1)"
+echo "Min workers last hour: $(tail -60 /var/log/php-fpm-monitor.log | cut -d',' -f2 | sort -n | head -1)"
+echo "Avg workers last hour: $(tail -60 /var/log/php-fpm-monitor.log | cut -d',' -f2 | awk '{sum+=$1} END {print sum/NR}')"
+
+# Alertes récentes
+tail -20 /var/log/alerts.log | grep "PHP Workers"
+```
+
+**Rotation automatique des logs :**
+Créer `/etc/logrotate.d/php-fpm-monitor` :
+```
+/var/log/php-fpm-monitor.log {
+ daily
+ rotate 30
+ compress
+ missingok
+ notifempty
+}
+```
+
+---
+
+## 8. Solutions de Monitoring Automatisées
+
+### Option 1 : Netdata (Recommandé)
+**Avantages :** Installation simple, interface web, détection auto des services
+
+```bash
+# Installation
+bash <(curl -Ss https://get.netdata.cloud/kickstart.sh) --dont-wait
+
+# Accessible sur http://IP:19999
+# Détecte automatiquement : PHP-FPM, MariaDB, Nginx, ressources système
+```
+
+**Métriques auto-détectées :**
+- ✅ CPU, RAM, I/O, réseau
+- ✅ PHP-FPM (workers, slow requests)
+- ✅ MariaDB (connexions, queries, locks)
+- ✅ Nginx (connexions, erreurs)
+
+---
+
+### Option 2 : Prometheus + Grafana
+**Avantages :** Professionnel, historique long terme, alerting avancé
+
+```bash
+# Exposer les métriques Incus
+incus config set core.metrics_address :8443
+
+# Installer Prometheus + Grafana
+# (plus complexe, voir doc officielle)
+```
+
+---
+
+### Option 3 : Scripts + Monitoring Simple
+
+Si vous préférez rester léger, surveillez manuellement avec :
+
+```bash
+# Dashboard temps réel
+watch -n 2 'echo "=== RESSOURCES ===";
+ free -h | head -2;
+ echo "";
+ echo "=== PHP-FPM ===";
+ ps aux | grep "php-fpm: pool" | wc -l;
+ echo "";
+ echo "=== MySQL ===";
+ mysql -e "SHOW STATUS LIKE \"Threads_connected\";" 2>/dev/null;
+ echo "";
+ echo "=== NGINX ===";
+ netstat -an | grep :80 | wc -l'
+```
+
+---
+
+## 9. Checklist de Diagnostic en Cas de Problème
+
+### Coupures / Timeouts
+1. ✅ **Vérifier PHP-FPM** : `tail /var/log/php8.3-fpm.log | grep max_children`
+2. ✅ **Vérifier MySQL** : `mysql -e "SHOW PROCESSLIST;"`
+3. ✅ **Vérifier nginx** : `tail /var/log/nginx/error.log | grep -E "502|504"`
+4. ✅ **Vérifier RAM** : `free -h`
+5. ✅ **Vérifier I/O** : `iostat -x 1 5`
+
+### Lenteurs
+1. ✅ **Slow queries MySQL** : `tail /var/lib/mysql/slow-query.log`
+2. ✅ **Slow PHP** : `tail /var/log/php8.3-fpm-slow.log`
+3. ✅ **CPU** : `top -bn1 | head -20`
+4. ✅ **I/O disque** : `iotop -oa`
+
+### Crashes / Erreurs 502
+1. ✅ **PHP-FPM status** : `systemctl status php8.3-fpm`
+2. ✅ **OOM Killer** : `dmesg | grep -i "out of memory"`
+3. ✅ **Logs PHP** : `tail -100 /var/log/php8.3-fpm.log`
+
+---
+
+## 10. Optimisations Recommandées
+
+### PHP-FPM
+```ini
+# /etc/php/8.3/fpm/pool.d/www.conf
+pm = dynamic
+pm.max_children = 50-100
+pm.start_servers = 10-20
+pm.min_spare_servers = 5-10
+pm.max_spare_servers = 20-35
+pm.max_requests = 500
+request_slowlog_timeout = 3s
+```
+
+### MariaDB
+```ini
+# /etc/mysql/mariadb.conf.d/50-server.cnf
+max_connections = 200-500
+innodb_buffer_pool_size = 16G # 70-80% de la RAM MySQL
+slow_query_log = 1
+long_query_time = 2
+log_queries_not_using_indexes = 1
+```
+
+### Nginx
+```nginx
+# /etc/nginx/nginx.conf
+worker_processes auto;
+worker_connections 2048;
+keepalive_timeout 30;
+```
+
+---
+
+## Résumé des Seuils Critiques
+
+| Métrique | Warning | Critical |
+|----------|---------|----------|
+| **RAM** | >85% | >95% |
+| **CPU Load** | >70% CPU count | >150% CPU count |
+| **Disk I/O await** | >50ms | >200ms |
+| **Disk Space** | >80% | >90% |
+| **PHP Workers** | >80% max | max_children reached |
+| **MySQL Connections** | >80% max | = max_connections |
+| **Slow Queries** | >2s | >10s ou >50/min |
+| **Nginx 502/504** | >5 en 5min | >20 en 5min |
+
+---
+
+## Contacts et Escalade
+
+En cas d'incident critique :
+1. Vérifier les logs (`/var/log/alerts.log`)
+2. Identifier le goulot (CPU/RAM/PHP/MySQL)
+3. Appliquer le correctif approprié
+4. Documenter l'incident
+
+---
+
+**Date de création :** 2025-10-18
+**Dernière mise à jour :** 2025-10-18
+**Version :** 1.0
diff --git a/VERSION b/VERSION
index fa7adc7a..87ce4929 100644
--- a/VERSION
+++ b/VERSION
@@ -1 +1 @@
-3.3.5
+3.5.2
diff --git a/__MACOSX/geo-app-20251014/._pubspec.yaml b/__MACOSX/geo-app-20251014/._pubspec.yaml
new file mode 100755
index 00000000..f6878123
Binary files /dev/null and b/__MACOSX/geo-app-20251014/._pubspec.yaml differ
diff --git a/__MACOSX/geo-app-20251014/build/._ios b/__MACOSX/geo-app-20251014/build/._ios
new file mode 100755
index 00000000..69731abb
Binary files /dev/null and b/__MACOSX/geo-app-20251014/build/._ios differ
diff --git a/__MACOSX/geo-app-20251014/build/ios/._Release-iphoneos b/__MACOSX/geo-app-20251014/build/ios/._Release-iphoneos
new file mode 100755
index 00000000..69731abb
Binary files /dev/null and b/__MACOSX/geo-app-20251014/build/ios/._Release-iphoneos differ
diff --git a/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._Flutter b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._Flutter
new file mode 100755
index 00000000..69731abb
Binary files /dev/null and b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._Flutter differ
diff --git a/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._Stripe b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._Stripe
new file mode 100755
index 00000000..69731abb
Binary files /dev/null and b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._Stripe differ
diff --git a/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._StripeApplePay b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._StripeApplePay
new file mode 100755
index 00000000..69731abb
Binary files /dev/null and b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._StripeApplePay differ
diff --git a/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._StripeCore b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._StripeCore
new file mode 100755
index 00000000..69731abb
Binary files /dev/null and b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._StripeCore differ
diff --git a/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._StripeFinancialConnections b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._StripeFinancialConnections
new file mode 100755
index 00000000..69731abb
Binary files /dev/null and b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._StripeFinancialConnections differ
diff --git a/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._StripePaymentSheet b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._StripePaymentSheet
new file mode 100755
index 00000000..69731abb
Binary files /dev/null and b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._StripePaymentSheet differ
diff --git a/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._StripePayments b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._StripePayments
new file mode 100755
index 00000000..69731abb
Binary files /dev/null and b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._StripePayments differ
diff --git a/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._StripePaymentsUI b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._StripePaymentsUI
new file mode 100755
index 00000000..69731abb
Binary files /dev/null and b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._StripePaymentsUI differ
diff --git a/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._StripeTerminal b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._StripeTerminal
new file mode 100755
index 00000000..69731abb
Binary files /dev/null and b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._StripeTerminal differ
diff --git a/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._StripeUICore b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._StripeUICore
new file mode 100755
index 00000000..69731abb
Binary files /dev/null and b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._StripeUICore differ
diff --git a/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._battery_plus b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._battery_plus
new file mode 100755
index 00000000..69731abb
Binary files /dev/null and b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._battery_plus differ
diff --git a/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._connectivity_plus b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._connectivity_plus
new file mode 100755
index 00000000..69731abb
Binary files /dev/null and b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._connectivity_plus differ
diff --git a/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._device_info_plus b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._device_info_plus
new file mode 100755
index 00000000..69731abb
Binary files /dev/null and b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._device_info_plus differ
diff --git a/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._flutter_local_notifications b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._flutter_local_notifications
new file mode 100755
index 00000000..69731abb
Binary files /dev/null and b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._flutter_local_notifications differ
diff --git a/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._geolocator_apple b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._geolocator_apple
new file mode 100755
index 00000000..69731abb
Binary files /dev/null and b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._geolocator_apple differ
diff --git a/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._image_picker_ios b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._image_picker_ios
new file mode 100755
index 00000000..69731abb
Binary files /dev/null and b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._image_picker_ios differ
diff --git a/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._mek_stripe_terminal b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._mek_stripe_terminal
new file mode 100755
index 00000000..69731abb
Binary files /dev/null and b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._mek_stripe_terminal differ
diff --git a/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._nfc_manager b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._nfc_manager
new file mode 100755
index 00000000..69731abb
Binary files /dev/null and b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._nfc_manager differ
diff --git a/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._path_provider_foundation b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._path_provider_foundation
new file mode 100755
index 00000000..69731abb
Binary files /dev/null and b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._path_provider_foundation differ
diff --git a/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._permission_handler_apple b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._permission_handler_apple
new file mode 100755
index 00000000..69731abb
Binary files /dev/null and b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._permission_handler_apple differ
diff --git a/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._stripe_ios b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._stripe_ios
new file mode 100755
index 00000000..69731abb
Binary files /dev/null and b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._stripe_ios differ
diff --git a/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._url_launcher_ios b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._url_launcher_ios
new file mode 100755
index 00000000..69731abb
Binary files /dev/null and b/__MACOSX/geo-app-20251014/build/ios/Release-iphoneos/._url_launcher_ios differ
diff --git a/__MACOSX/geo-app-20251014/ios/._Runner.xcodeproj b/__MACOSX/geo-app-20251014/ios/._Runner.xcodeproj
new file mode 100755
index 00000000..d17f20a3
Binary files /dev/null and b/__MACOSX/geo-app-20251014/ios/._Runner.xcodeproj differ
diff --git a/__MACOSX/geo-app-20251014/ios/._Runner.xcworkspace b/__MACOSX/geo-app-20251014/ios/._Runner.xcworkspace
new file mode 100755
index 00000000..75a1adbb
Binary files /dev/null and b/__MACOSX/geo-app-20251014/ios/._Runner.xcworkspace differ
diff --git a/__MACOSX/geo-app-20251014/ios/Pods/._Pods.xcodeproj b/__MACOSX/geo-app-20251014/ios/Pods/._Pods.xcodeproj
new file mode 100755
index 00000000..72fb2412
Binary files /dev/null and b/__MACOSX/geo-app-20251014/ios/Pods/._Pods.xcodeproj differ
diff --git a/api/.vscode/settings.json b/api/.vscode/settings.json
old mode 100644
new mode 100755
index 024a62cd..d6cfba74
--- a/api/.vscode/settings.json
+++ b/api/.vscode/settings.json
@@ -1,4 +1,116 @@
{
+ "window.zoomLevel": 1, // Permet de zoomer, pratique si vous faites une présentation
+
+ // Apparence
+ // -- Editeur
+ "workbench.startupEditor": "none", // On ne veut pas une page d'accueil chargée
+ "editor.minimap.enabled": true, // On veut voir la minimap
+ "editor.minimap.showSlider": "always", // On veut voir la minimap
+ "editor.minimap.size": "fill", // On veut voir la minimap
+ "editor.minimap.scale": 2,
+ "editor.tokenColorCustomizations": {
+ "textMateRules": [
+ {
+ "scope": ["storage.type.function", "storage.type.class"],
+ "settings": {
+ "fontStyle": "bold",
+ "foreground": "#4B9CD3"
+ }
+ }
+ ]
+ },
+ "editor.minimap.renderCharacters": true,
+ "editor.minimap.maxColumn": 120,
+ "breadcrumbs.enabled": false,
+ // -- Tabs
+ "workbench.editor.wrapTabs": true, // On veut voir les tabs
+ "workbench.editor.tabSizing": "shrink", // On veut voir les tabs
+ "workbench.editor.pinnedTabSizing": "compact",
+ "workbench.editor.enablePreview": false, // Un clic sur un fichier l'ouvre
+
+ // -- Sidebar
+ "workbench.tree.indent": 15, // Indente plus pour plus de clarté dans la sidebar
+ "workbench.tree.renderIndentGuides": "always",
+ // -- Code
+ "editor.occurrencesHighlight": "singleFile", // On veut voir les occurences d'une variable
+ "editor.renderWhitespace": "trailing", // On ne veut pas laisser d'espace en fin de ligne
+ "editor.renderControlCharacters": true, // On veut voir les caractères de contrôle
+ // Thème
+ "editor.fontFamily": "'JetBrains Mono', 'Fira Code', 'Operator Mono Lig', monospace",
+ "editor.fontLigatures": false,
+ "editor.fontSize": 13,
+ "editor.lineHeight": 22,
+ "editor.guides.bracketPairs": "active",
+
+ // Ergonomie
+ "editor.wordWrap": "off",
+ "editor.rulers": [],
+ "editor.suggest.insertMode": "replace", // L'autocomplétion remplace le mot en cours
+ "editor.acceptSuggestionOnCommitCharacter": false, // Evite que l'autocomplétion soit accepté lors d'un . par exemple
+ "editor.formatOnSave": true,
+ "editor.formatOnPaste": true,
+ "editor.linkedEditing": true, // Quand on change un élément HTML, change la balise fermante
+ "editor.tabSize": 2,
+ "editor.unicodeHighlight.nonBasicASCII": false,
+
+ "[php]": {
+ "editor.defaultFormatter": "bmewburn.vscode-intelephense-client",
+ "editor.formatOnSave": true,
+ "editor.formatOnPaste": true
+ },
+ "intelephense.format.braces": "k&r",
+ "intelephense.format.enable": true,
+
+ "[javascript]": {
+ "editor.defaultFormatter": "esbenp.prettier-vscode",
+ "editor.formatOnSave": true,
+ "editor.formatOnPaste": true
+ },
+ "prettier.printWidth": 360,
+ "prettier.semi": true,
+ "prettier.singleQuote": true,
+ "prettier.tabWidth": 2,
+ "prettier.trailingComma": "es5",
+
+ "explorer.autoReveal": false,
+ "explorer.confirmDragAndDrop": false,
+ "emmet.triggerExpansionOnTab": true,
+ "emmet.includeLanguages": {
+ "javascript": "javascriptreact"
+ },
+"problems.decorations.enabled": true,
+ "explorer.decorations.colors": true,
+ "explorer.decorations.badges": true,
+ "php.validate.enable": true,
+ "php.suggest.basic": false,
+ "dart.analysisExcludedFolders": [],
+ "dart.enableSdkFormatter": true,
+
+ // Fichiers
+ "files.defaultLanguage": "markdown",
+ "files.autoSaveWorkspaceFilesOnly": true,
+ "files.exclude": {
+ "**/.idea": true
+ },
+ // Languages
+ "javascript.preferences.importModuleSpecifierEnding": "js",
+ "typescript.preferences.importModuleSpecifierEnding": "js",
+
+ // Extensions
+ "tailwindCSS.experimental.configFile": "web/tailwind.config.js",
+ "editor.quickSuggestions": {
+ "strings": true
+ },
+
+ "[svelte]": {
+ "editor.defaultFormatter": "svelte.svelte-vscode",
+ "editor.formatOnSave": true
+ },
+ "prettier.documentSelectors": ["**/*.svelte"],
+ "svelte.plugin.svelte.diagnostics.enable": false,
+
+ "js/ts.implicitProjectConfig.checkJs": false,
+ "svelte.enable-ts-plugin": false,
"workbench.colorCustomizations": {
"activityBar.activeBackground": "#fa1b49",
"activityBar.background": "#fa1b49",
@@ -18,5 +130,6 @@
"titleBar.inactiveBackground": "#dd053199",
"titleBar.inactiveForeground": "#e7e7e799"
},
- "peacock.color": "#dd0531"
+ "peacock.color": "#dd0531",
+
}
diff --git a/api/PM7/d6back.sh b/api/PM7/d6back.sh
new file mode 100644
index 00000000..5379f790
--- /dev/null
+++ b/api/PM7/d6back.sh
@@ -0,0 +1,651 @@
+#!/bin/bash
+
+set -uo pipefail
+# Note: Removed -e to allow script to continue on errors
+# Errors are handled explicitly with ERROR_COUNT
+
+# Parse command line arguments
+ONLY_DB=false
+if [[ "${1:-}" == "-onlydb" ]]; then
+ ONLY_DB=true
+ echo "Mode: Database backup only"
+fi
+
+# Configuration
+SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
+CONFIG_FILE="$SCRIPT_DIR/d6back.yaml"
+LOG_DIR="$SCRIPT_DIR/logs"
+mkdir -p "$LOG_DIR"
+LOG_FILE="$LOG_DIR/d6back-$(date +%Y%m%d).log"
+ERROR_COUNT=0
+RECAP_FILE="/tmp/backup_recap_$$.txt"
+
+# Lock file to prevent concurrent executions
+LOCK_FILE="/var/lock/d6back.lock"
+exec 200>"$LOCK_FILE"
+if ! flock -n 200; then
+ echo "ERROR: Another backup is already running" >&2
+ exit 1
+fi
+trap 'flock -u 200' EXIT
+
+# Clean old log files (keep only last 10)
+find "$LOG_DIR" -maxdepth 1 -name "d6back-*.log" -type f 2>/dev/null | sort -r | tail -n +11 | xargs -r rm -f || true
+
+# Check dependencies - COMMENTED OUT
+# for cmd in yq ssh tar openssl; do
+# if ! command -v "$cmd" &> /dev/null; then
+# echo "ERROR: $cmd is required but not installed" | tee -a "$LOG_FILE"
+# exit 1
+# fi
+# done
+
+# Load config
+DIR_BACKUP=$(yq '.global.dir_backup' "$CONFIG_FILE" | tr -d '"')
+ENC_KEY_PATH=$(yq '.global.enc_key' "$CONFIG_FILE" | tr -d '"')
+BACKUP_SERVER=$(yq '.global.backup_server // "BACKUP"' "$CONFIG_FILE" | tr -d '"')
+EMAIL_TO=$(yq '.global.email_to // "support@unikoffice.com"' "$CONFIG_FILE" | tr -d '"')
+KEEP_DIRS=$(yq '.global.keep_dirs' "$CONFIG_FILE" | tr -d '"')
+KEEP_DB=$(yq '.global.keep_db' "$CONFIG_FILE" | tr -d '"')
+
+# Load encryption key
+if [[ ! -f "$ENC_KEY_PATH" ]]; then
+ echo "ERROR: Encryption key not found: $ENC_KEY_PATH" | tee -a "$LOG_FILE"
+ exit 1
+fi
+ENC_KEY=$(cat "$ENC_KEY_PATH")
+
+echo "=== Backup Started $(date) ===" | tee -a "$LOG_FILE"
+echo "Backup directory: $DIR_BACKUP" | tee -a "$LOG_FILE"
+
+# Check available disk space
+DISK_USAGE=$(df "$DIR_BACKUP" | tail -1 | awk '{print $5}' | sed 's/%//')
+DISK_FREE=$((100 - DISK_USAGE))
+
+if [[ $DISK_FREE -lt 20 ]]; then
+ echo "WARNING: Low disk space! Only ${DISK_FREE}% free on backup partition" | tee -a "$LOG_FILE"
+
+ # Send warning email
+ echo "Sending DISK SPACE WARNING email to $EMAIL_TO (${DISK_FREE}% free)" | tee -a "$LOG_FILE"
+ if command -v msmtp &> /dev/null; then
+ {
+ echo "To: $EMAIL_TO"
+ echo "Subject: Backup${BACKUP_SERVER} WARNING - Low disk space (${DISK_FREE}% free)"
+ echo ""
+ echo "WARNING: Low disk space on $(hostname)"
+ echo ""
+ echo "Backup directory: $DIR_BACKUP"
+ echo "Disk usage: ${DISK_USAGE}%"
+ echo "Free space: ${DISK_FREE}%"
+ echo ""
+ echo "The backup will continue but please free up some space soon."
+ echo ""
+ echo "Date: $(date '+%d.%m.%Y %H:%M')"
+ } | msmtp "$EMAIL_TO"
+ echo "DISK SPACE WARNING email sent successfully to $EMAIL_TO" | tee -a "$LOG_FILE"
+ else
+ echo "WARNING: msmtp not found - DISK WARNING email NOT sent" | tee -a "$LOG_FILE"
+ fi
+else
+ echo "Disk space OK: ${DISK_FREE}% free" | tee -a "$LOG_FILE"
+fi
+
+# Initialize recap file
+echo "BACKUP REPORT - $(hostname) - $(date '+%d.%m.%Y %H')h" > "$RECAP_FILE"
+echo "========================================" >> "$RECAP_FILE"
+echo "" >> "$RECAP_FILE"
+
+# Function to format size in MB with thousand separator
+format_size_mb() {
+ local file="$1"
+ if [[ -f "$file" ]]; then
+ local size_kb=$(du -k "$file" | cut -f1)
+ local size_mb=$((size_kb / 1024))
+ # Add thousand separator with printf and sed
+ printf "%d" "$size_mb" | sed ':a;s/\B[0-9]\{3\}\>/\.&/;ta'
+ else
+ echo "0"
+ fi
+}
+
+# Function to calculate age in days
+get_age_days() {
+ local file="$1"
+ local now=$(date +%s)
+ local file_time=$(stat -c %Y "$file" 2>/dev/null || echo 0)
+ echo $(( (now - file_time) / 86400 ))
+}
+
+# Function to get week number of year for a file
+get_week_year() {
+ local file="$1"
+ local file_time=$(stat -c %Y "$file" 2>/dev/null || echo 0)
+ date -d "@$file_time" +"%Y-%W"
+}
+
+# Function to cleanup old backups according to retention policy
+cleanup_old_backups() {
+ local DELETED_COUNT=0
+ local KEPT_COUNT=0
+
+ echo "" | tee -a "$LOG_FILE"
+ echo "=== Starting Backup Retention Cleanup ===" | tee -a "$LOG_FILE"
+
+ # Parse retention periods
+ local KEEP_DIRS_DAYS=${KEEP_DIRS%d} # Remove 'd' suffix
+
+ # Parse database retention (5d,3w,15m)
+ IFS=',' read -r KEEP_DB_DAILY KEEP_DB_WEEKLY KEEP_DB_MONTHLY <<< "$KEEP_DB"
+ local KEEP_DB_DAILY_DAYS=${KEEP_DB_DAILY%d}
+ local KEEP_DB_WEEKLY_WEEKS=${KEEP_DB_WEEKLY%w}
+ local KEEP_DB_MONTHLY_MONTHS=${KEEP_DB_MONTHLY%m}
+
+ # Convert to days
+ local KEEP_DB_WEEKLY_DAYS=$((KEEP_DB_WEEKLY_WEEKS * 7))
+ local KEEP_DB_MONTHLY_DAYS=$((KEEP_DB_MONTHLY_MONTHS * 30))
+
+ echo "Retention policy: dirs=${KEEP_DIRS_DAYS}d, db=${KEEP_DB_DAILY_DAYS}d/${KEEP_DB_WEEKLY_WEEKS}w/${KEEP_DB_MONTHLY_MONTHS}m" | tee -a "$LOG_FILE"
+
+ # Process each host directory
+ for host_dir in "$DIR_BACKUP"/*; do
+ if [[ ! -d "$host_dir" ]]; then
+ continue
+ fi
+
+ local host_name=$(basename "$host_dir")
+ echo " Cleaning host: $host_name" | tee -a "$LOG_FILE"
+
+ # Clean directory backups (*.tar.gz but not *.sql.gz.enc)
+ while IFS= read -r -d '' file; do
+ if [[ $(basename "$file") == *".sql.gz.enc" ]]; then
+ continue # Skip SQL files
+ fi
+
+ local age_days=$(get_age_days "$file")
+
+ if [[ $age_days -gt $KEEP_DIRS_DAYS ]]; then
+ rm -f "$file"
+ echo " Deleted: $(basename "$file") (${age_days}d > ${KEEP_DIRS_DAYS}d)" | tee -a "$LOG_FILE"
+ ((DELETED_COUNT++))
+ else
+ ((KEPT_COUNT++))
+ fi
+ done < <(find "$host_dir" -name "*.tar.gz" -type f -print0 2>/dev/null)
+
+ # Clean database backups with retention policy
+ declare -A db_files
+
+ while IFS= read -r -d '' file; do
+ local filename=$(basename "$file")
+ local db_name=${filename%%_*}
+
+ if [[ -z "${db_files[$db_name]:-}" ]]; then
+ db_files[$db_name]="$file"
+ else
+ db_files[$db_name]+=$'\n'"$file"
+ fi
+ done < <(find "$host_dir" -name "*.sql.gz.enc" -type f -print0 2>/dev/null)
+
+ # Process each database
+ for db_name in "${!db_files[@]}"; do
+ # Sort files by age (newest first)
+ mapfile -t files < <(echo "${db_files[$db_name]}" | while IFS= read -r f; do
+ echo "$f"
+ done | xargs -I {} stat -c "%Y {}" {} 2>/dev/null | sort -rn | cut -d' ' -f2-)
+
+ # Track which files to keep
+ declare -A keep_daily
+ declare -A keep_weekly
+
+ for file in "${files[@]}"; do
+ local age_days=$(get_age_days "$file")
+
+ if [[ $age_days -le $KEEP_DB_DAILY_DAYS ]]; then
+ # Keep all files within daily retention
+ ((KEPT_COUNT++))
+
+ elif [[ $age_days -le $KEEP_DB_WEEKLY_DAYS ]]; then
+ # Weekly retention: keep one per day
+ local file_date=$(date -d "@$(stat -c %Y "$file")" +"%Y-%m-%d")
+
+ if [[ -z "${keep_daily[$file_date]:-}" ]]; then
+ keep_daily[$file_date]="$file"
+ ((KEPT_COUNT++))
+ else
+ rm -f "$file"
+ ((DELETED_COUNT++))
+ fi
+
+ elif [[ $age_days -le $KEEP_DB_MONTHLY_DAYS ]]; then
+ # Monthly retention: keep one per week
+ local week_year=$(get_week_year "$file")
+
+ if [[ -z "${keep_weekly[$week_year]:-}" ]]; then
+ keep_weekly[$week_year]="$file"
+ ((KEPT_COUNT++))
+ else
+ rm -f "$file"
+ ((DELETED_COUNT++))
+ fi
+
+ else
+ # Beyond retention period
+ rm -f "$file"
+ echo " Deleted: $(basename "$file") (${age_days}d > ${KEEP_DB_MONTHLY_DAYS}d)" | tee -a "$LOG_FILE"
+ ((DELETED_COUNT++))
+ fi
+ done
+
+ unset keep_daily keep_weekly
+ done
+
+ unset db_files
+ done
+
+ echo "Cleanup completed: ${DELETED_COUNT} deleted, ${KEPT_COUNT} kept" | tee -a "$LOG_FILE"
+
+ # Add cleanup summary to recap file
+ echo "" >> "$RECAP_FILE"
+ echo "CLEANUP SUMMARY:" >> "$RECAP_FILE"
+ echo " Files deleted: $DELETED_COUNT" >> "$RECAP_FILE"
+ echo " Files kept: $KEPT_COUNT" >> "$RECAP_FILE"
+}
+
+# Function to backup a single database (must be defined before use)
+backup_database() {
+ local database="$1"
+ local timestamp="$(date +%Y%m%d_%H)"
+ local backup_file="$backup_dir/sql/${database}_${timestamp}.sql.gz.enc"
+
+ echo " Backing up database: $database" | tee -a "$LOG_FILE"
+
+ if [[ "$ssh_user" != "root" ]]; then
+ CMD_PREFIX="sudo"
+ else
+ CMD_PREFIX=""
+ fi
+
+ # Execute backup with encryption
+ # First test MySQL connection to get clear error messages (|| true to continue on error)
+ MYSQL_TEST=$(ssh -i "$ssh_key" -p "$ssh_port" -o ConnectTimeout=20 "$ssh_user@$host_ip" \
+ "$CMD_PREFIX incus exec $container_name -- bash -c 'cat > /tmp/d6back.cnf << EOF
+[client]
+user=$db_user
+password=$db_pass
+host=$db_host
+EOF
+chmod 600 /tmp/d6back.cnf
+mariadb --defaults-extra-file=/tmp/d6back.cnf -e \"SELECT 1\" 2>&1
+rm -f /tmp/d6back.cnf'" 2>/dev/null || true)
+
+ if ssh -i "$ssh_key" -p "$ssh_port" -o ConnectTimeout=20 "$ssh_user@$host_ip" \
+ "$CMD_PREFIX incus exec $container_name -- bash -c 'cat > /tmp/d6back.cnf << EOF
+[client]
+user=$db_user
+password=$db_pass
+host=$db_host
+EOF
+chmod 600 /tmp/d6back.cnf
+mariadb-dump --defaults-extra-file=/tmp/d6back.cnf --single-transaction --lock-tables=false --add-drop-table --create-options --databases $database 2>/dev/null | sed -e \"/^CREATE DATABASE/s/\\\`$database\\\`/\\\`${database}_${timestamp}\\\`/\" -e \"/^USE/s/\\\`$database\\\`/\\\`${database}_${timestamp}\\\`/\" | gzip
+rm -f /tmp/d6back.cnf'" | \
+ openssl enc -aes-256-cbc -salt -pass pass:"$ENC_KEY" -pbkdf2 > "$backup_file" 2>/dev/null; then
+
+ # Validate backup file size (encrypted SQL should be > 100 bytes)
+ if [[ -f "$backup_file" ]]; then
+ file_size=$(stat -c%s "$backup_file" 2>/dev/null || echo 0)
+ if [[ $file_size -lt 100 ]]; then
+ # Analyze MySQL connection test results
+ if [[ "$MYSQL_TEST" == *"Access denied"* ]]; then
+ echo " ERROR: MySQL authentication failed for $database on $host_name/$container_name" | tee -a "$LOG_FILE"
+ echo " User: $db_user@$db_host - Check password in configuration" | tee -a "$LOG_FILE"
+ elif [[ "$MYSQL_TEST" == *"Unknown database"* ]]; then
+ echo " ERROR: Database '$database' does not exist on $host_name/$container_name" | tee -a "$LOG_FILE"
+ elif [[ "$MYSQL_TEST" == *"Can't connect"* ]]; then
+ echo " ERROR: Cannot connect to MySQL server at $db_host in $container_name" | tee -a "$LOG_FILE"
+ else
+ echo " ERROR: Backup file too small (${file_size} bytes): $database on $host_name/$container_name" | tee -a "$LOG_FILE"
+ fi
+
+ ((ERROR_COUNT++))
+ rm -f "$backup_file"
+ else
+ size=$(du -h "$backup_file" | cut -f1)
+ size_mb=$(format_size_mb "$backup_file")
+ echo " ✓ Saved (encrypted): $(basename "$backup_file") ($size)" | tee -a "$LOG_FILE"
+ echo " SQL: $(basename "$backup_file") - ${size_mb} Mo" >> "$RECAP_FILE"
+
+ # Test backup integrity
+ if ! openssl enc -aes-256-cbc -d -pass pass:"$ENC_KEY" -pbkdf2 -in "$backup_file" | gunzip -t 2>/dev/null; then
+ echo " ERROR: Backup integrity check failed for $database" | tee -a "$LOG_FILE"
+ ((ERROR_COUNT++))
+ fi
+ fi
+ else
+ echo " ERROR: Backup file not created: $database" | tee -a "$LOG_FILE"
+ ((ERROR_COUNT++))
+ fi
+ else
+ # Analyze MySQL connection test for failed backup
+ if [[ "$MYSQL_TEST" == *"Access denied"* ]]; then
+ echo " ERROR: MySQL authentication failed for $database on $host_name/$container_name" | tee -a "$LOG_FILE"
+ echo " User: $db_user@$db_host - Check password in configuration" | tee -a "$LOG_FILE"
+ elif [[ "$MYSQL_TEST" == *"Unknown database"* ]]; then
+ echo " ERROR: Database '$database' does not exist on $host_name/$container_name" | tee -a "$LOG_FILE"
+ elif [[ "$MYSQL_TEST" == *"Can't connect"* ]]; then
+ echo " ERROR: Cannot connect to MySQL server at $db_host in $container_name" | tee -a "$LOG_FILE"
+ else
+ echo " ERROR: Failed to backup database $database on $host_name/$container_name" | tee -a "$LOG_FILE"
+ fi
+
+ ((ERROR_COUNT++))
+ rm -f "$backup_file"
+ fi
+}
+
+# Process each host
+host_count=$(yq '.hosts | length' "$CONFIG_FILE")
+
+for ((i=0; i<$host_count; i++)); do
+ host_name=$(yq ".hosts[$i].name" "$CONFIG_FILE" | tr -d '"')
+ host_ip=$(yq ".hosts[$i].ip" "$CONFIG_FILE" | tr -d '"')
+ ssh_user=$(yq ".hosts[$i].user" "$CONFIG_FILE" | tr -d '"')
+ ssh_key=$(yq ".hosts[$i].key" "$CONFIG_FILE" | tr -d '"')
+ ssh_port=$(yq ".hosts[$i].port // 22" "$CONFIG_FILE" | tr -d '"')
+
+ echo "Processing host: $host_name ($host_ip)" | tee -a "$LOG_FILE"
+ echo "" >> "$RECAP_FILE"
+ echo "HOST: $host_name ($host_ip)" >> "$RECAP_FILE"
+ echo "----------------------------" >> "$RECAP_FILE"
+
+ # Test SSH connection
+ if ! ssh -i "$ssh_key" -p "$ssh_port" -o ConnectTimeout=20 -o StrictHostKeyChecking=no "$ssh_user@$host_ip" "true" 2>/dev/null; then
+ echo " ERROR: Cannot connect to $host_name ($host_ip)" | tee -a "$LOG_FILE"
+ ((ERROR_COUNT++))
+ continue
+ fi
+
+ # Process containers
+ container_count=$(yq ".hosts[$i].containers | length" "$CONFIG_FILE" 2>/dev/null || echo "0")
+
+ for ((c=0; c<$container_count; c++)); do
+ container_name=$(yq ".hosts[$i].containers[$c].name" "$CONFIG_FILE" | tr -d '"')
+
+ echo " Processing container: $container_name" | tee -a "$LOG_FILE"
+
+ # Add container to recap
+ echo "" >> "$RECAP_FILE"
+ echo " Container: $container_name" >> "$RECAP_FILE"
+
+ # Create backup directories
+ backup_dir="$DIR_BACKUP/$host_name/$container_name"
+ mkdir -p "$backup_dir"
+ mkdir -p "$backup_dir/sql"
+
+ # Backup directories (skip if -onlydb mode)
+ if [[ "$ONLY_DB" == "false" ]]; then
+ dir_count=$(yq ".hosts[$i].containers[$c].dirs | length" "$CONFIG_FILE" 2>/dev/null || echo "0")
+
+ for ((d=0; d<$dir_count; d++)); do
+ dir_path=$(yq ".hosts[$i].containers[$c].dirs[$d]" "$CONFIG_FILE" | sed 's/^"\|"$//g')
+
+ # Use sudo if not root
+ if [[ "$ssh_user" != "root" ]]; then
+ CMD_PREFIX="sudo"
+ else
+ CMD_PREFIX=""
+ fi
+
+ # Special handling for /var/www - backup each subdirectory separately
+ if [[ "$dir_path" == "/var/www" ]]; then
+ echo " Backing up subdirectories of $dir_path" | tee -a "$LOG_FILE"
+
+ # Get list of subdirectories
+ subdirs=$(ssh -i "$ssh_key" -p "$ssh_port" -o ConnectTimeout=20 "$ssh_user@$host_ip" \
+ "$CMD_PREFIX incus exec $container_name -- find /var/www -maxdepth 1 -type d ! -path /var/www" 2>/dev/null || echo "")
+
+ for subdir in $subdirs; do
+ subdir_name=$(basename "$subdir" | tr '/' '_')
+ backup_file="$backup_dir/www_${subdir_name}_$(date +%Y%m%d_%H).tar.gz"
+
+ echo " Backing up: $subdir" | tee -a "$LOG_FILE"
+
+ if ssh -i "$ssh_key" -p "$ssh_port" -o ConnectTimeout=20 "$ssh_user@$host_ip" \
+ "$CMD_PREFIX incus exec $container_name -- tar czf - $subdir 2>/dev/null" > "$backup_file"; then
+
+ # Validate backup file size (tar.gz should be > 1KB)
+ if [[ -f "$backup_file" ]]; then
+ file_size=$(stat -c%s "$backup_file" 2>/dev/null || echo 0)
+ if [[ $file_size -lt 1024 ]]; then
+ echo " WARNING: Backup file very small (${file_size} bytes): $subdir" | tee -a "$LOG_FILE"
+ # Keep the file but note it's small
+ size=$(du -h "$backup_file" | cut -f1)
+ size_mb=$(format_size_mb "$backup_file")
+ echo " ✓ Saved (small): $(basename "$backup_file") ($size)" | tee -a "$LOG_FILE"
+ echo " DIR: $(basename "$backup_file") - ${size_mb} Mo (WARNING: small)" >> "$RECAP_FILE"
+ else
+ size=$(du -h "$backup_file" | cut -f1)
+ size_mb=$(format_size_mb "$backup_file")
+ echo " ✓ Saved: $(basename "$backup_file") ($size)" | tee -a "$LOG_FILE"
+ echo " DIR: $(basename "$backup_file") - ${size_mb} Mo" >> "$RECAP_FILE"
+ fi
+
+ # Test tar integrity
+ if ! tar tzf "$backup_file" >/dev/null 2>&1; then
+ echo " ERROR: Tar integrity check failed" | tee -a "$LOG_FILE"
+ ((ERROR_COUNT++))
+ fi
+ else
+ echo " ERROR: Backup file not created: $subdir" | tee -a "$LOG_FILE"
+ ((ERROR_COUNT++))
+ fi
+ else
+ echo " ERROR: Failed to backup $subdir" | tee -a "$LOG_FILE"
+ ((ERROR_COUNT++))
+ rm -f "$backup_file"
+ fi
+ done
+ else
+ # Normal backup for other directories
+ dir_name=$(basename "$dir_path" | tr '/' '_')
+ backup_file="$backup_dir/${dir_name}_$(date +%Y%m%d_%H).tar.gz"
+
+ echo " Backing up: $dir_path" | tee -a "$LOG_FILE"
+
+ if ssh -i "$ssh_key" -p "$ssh_port" -o ConnectTimeout=20 "$ssh_user@$host_ip" \
+ "$CMD_PREFIX incus exec $container_name -- tar czf - $dir_path 2>/dev/null" > "$backup_file"; then
+
+ # Validate backup file size (tar.gz should be > 1KB)
+ if [[ -f "$backup_file" ]]; then
+ file_size=$(stat -c%s "$backup_file" 2>/dev/null || echo 0)
+ if [[ $file_size -lt 1024 ]]; then
+ echo " WARNING: Backup file very small (${file_size} bytes): $dir_path" | tee -a "$LOG_FILE"
+ # Keep the file but note it's small
+ size=$(du -h "$backup_file" | cut -f1)
+ size_mb=$(format_size_mb "$backup_file")
+ echo " ✓ Saved (small): $(basename "$backup_file") ($size)" | tee -a "$LOG_FILE"
+ echo " DIR: $(basename "$backup_file") - ${size_mb} Mo (WARNING: small)" >> "$RECAP_FILE"
+ else
+ size=$(du -h "$backup_file" | cut -f1)
+ size_mb=$(format_size_mb "$backup_file")
+ echo " ✓ Saved: $(basename "$backup_file") ($size)" | tee -a "$LOG_FILE"
+ echo " DIR: $(basename "$backup_file") - ${size_mb} Mo" >> "$RECAP_FILE"
+ fi
+
+ # Test tar integrity
+ if ! tar tzf "$backup_file" >/dev/null 2>&1; then
+ echo " ERROR: Tar integrity check failed" | tee -a "$LOG_FILE"
+ ((ERROR_COUNT++))
+ fi
+ else
+ echo " ERROR: Backup file not created: $dir_path" | tee -a "$LOG_FILE"
+ ((ERROR_COUNT++))
+ fi
+ else
+ echo " ERROR: Failed to backup $dir_path" | tee -a "$LOG_FILE"
+ ((ERROR_COUNT++))
+ rm -f "$backup_file"
+ fi
+ fi
+ done
+ fi # End of directory backup section
+
+ # Backup databases
+ db_user=$(yq ".hosts[$i].containers[$c].db_user" "$CONFIG_FILE" 2>/dev/null | tr -d '"')
+ db_pass=$(yq ".hosts[$i].containers[$c].db_pass" "$CONFIG_FILE" 2>/dev/null | tr -d '"')
+ db_host=$(yq ".hosts[$i].containers[$c].db_host // \"localhost\"" "$CONFIG_FILE" 2>/dev/null | tr -d '"')
+
+ # Check if we're in onlydb mode
+ if [[ "$ONLY_DB" == "true" ]]; then
+ # Use onlydb list if it exists
+ onlydb_count=$(yq ".hosts[$i].containers[$c].onlydb | length" "$CONFIG_FILE" 2>/dev/null || echo "0")
+ if [[ "$onlydb_count" != "0" ]] && [[ "$onlydb_count" != "null" ]]; then
+ db_count="$onlydb_count"
+ use_onlydb=true
+ else
+ # No onlydb list, skip this container in onlydb mode
+ continue
+ fi
+ else
+ # Normal mode - use databases list
+ db_count=$(yq ".hosts[$i].containers[$c].databases | length" "$CONFIG_FILE" 2>/dev/null || echo "0")
+ use_onlydb=false
+ fi
+
+ if [[ -n "$db_user" ]] && [[ -n "$db_pass" ]] && [[ "$db_count" != "0" ]]; then
+ for ((db=0; db<$db_count; db++)); do
+ if [[ "$use_onlydb" == "true" ]]; then
+ db_name=$(yq ".hosts[$i].containers[$c].onlydb[$db]" "$CONFIG_FILE" | tr -d '"')
+ else
+ db_name=$(yq ".hosts[$i].containers[$c].databases[$db]" "$CONFIG_FILE" | tr -d '"')
+ fi
+
+ if [[ "$db_name" == "ALL" ]]; then
+ echo " Fetching all databases..." | tee -a "$LOG_FILE"
+
+ # Get database list
+ if [[ "$ssh_user" != "root" ]]; then
+ db_list=$(ssh -i "$ssh_key" -p "$ssh_port" -o ConnectTimeout=20 "$ssh_user@$host_ip" \
+ "sudo incus exec $container_name -- bash -c 'cat > /tmp/d6back.cnf << EOF
+[client]
+user=$db_user
+password=$db_pass
+host=$db_host
+EOF
+chmod 600 /tmp/d6back.cnf
+mariadb --defaults-extra-file=/tmp/d6back.cnf -e \"SHOW DATABASES;\" 2>/dev/null
+rm -f /tmp/d6back.cnf'" | \
+ grep -Ev '^(Database|information_schema|performance_schema|mysql|sys)$' || echo "")
+ else
+ db_list=$(ssh -i "$ssh_key" -p "$ssh_port" -o ConnectTimeout=20 "$ssh_user@$host_ip" \
+ "incus exec $container_name -- bash -c 'cat > /tmp/d6back.cnf << EOF
+[client]
+user=$db_user
+password=$db_pass
+host=$db_host
+EOF
+chmod 600 /tmp/d6back.cnf
+mariadb --defaults-extra-file=/tmp/d6back.cnf -e \"SHOW DATABASES;\" 2>/dev/null
+rm -f /tmp/d6back.cnf'" | \
+ grep -Ev '^(Database|information_schema|performance_schema|mysql|sys)$' || echo "")
+ fi
+
+ # Backup each database
+ for single_db in $db_list; do
+ backup_database "$single_db"
+ done
+ else
+ backup_database "$db_name"
+ fi
+ done
+ fi
+ done
+done
+
+echo "=== Backup Completed $(date) ===" | tee -a "$LOG_FILE"
+
+# Cleanup old backups according to retention policy
+cleanup_old_backups
+
+# Show summary
+total_size=$(du -sh "$DIR_BACKUP" 2>/dev/null | cut -f1)
+echo "Total backup size: $total_size" | tee -a "$LOG_FILE"
+
+# Add summary to recap
+echo "" >> "$RECAP_FILE"
+echo "========================================" >> "$RECAP_FILE"
+
+# Add size details per host/container
+echo "BACKUP SIZES:" >> "$RECAP_FILE"
+for host_dir in "$DIR_BACKUP"/*; do
+ if [[ -d "$host_dir" ]]; then
+ host_name=$(basename "$host_dir")
+ host_size=$(du -sh "$host_dir" 2>/dev/null | cut -f1)
+ echo "" >> "$RECAP_FILE"
+ echo " $host_name: $host_size" >> "$RECAP_FILE"
+
+ # Size per container
+ for container_dir in "$host_dir"/*; do
+ if [[ -d "$container_dir" ]]; then
+ container_name=$(basename "$container_dir")
+ container_size=$(du -sh "$container_dir" 2>/dev/null | cut -f1)
+ echo " - $container_name: $container_size" >> "$RECAP_FILE"
+ fi
+ done
+ fi
+done
+
+echo "" >> "$RECAP_FILE"
+echo "TOTAL SIZE: $total_size" >> "$RECAP_FILE"
+echo "COMPLETED: $(date '+%d.%m.%Y %H:%M')" >> "$RECAP_FILE"
+
+# Prepare email subject with date format
+DATE_SUBJECT=$(date '+%d.%m.%Y %H')
+
+# Send recap email
+if [[ $ERROR_COUNT -gt 0 ]]; then
+ echo "Total errors: $ERROR_COUNT" | tee -a "$LOG_FILE"
+
+ # Add errors to recap
+ echo "" >> "$RECAP_FILE"
+ echo "ERRORS DETECTED: $ERROR_COUNT" >> "$RECAP_FILE"
+ echo "----------------------------" >> "$RECAP_FILE"
+ grep -i "ERROR" "$LOG_FILE" >> "$RECAP_FILE"
+
+ # Send email with ERROR in subject
+ echo "Sending ERROR email to $EMAIL_TO (Errors found: $ERROR_COUNT)" | tee -a "$LOG_FILE"
+ if command -v msmtp &> /dev/null; then
+ {
+ echo "To: $EMAIL_TO"
+ echo "Subject: Backup${BACKUP_SERVER} ERROR $DATE_SUBJECT"
+ echo ""
+ cat "$RECAP_FILE"
+ } | msmtp "$EMAIL_TO"
+ echo "ERROR email sent successfully to $EMAIL_TO" | tee -a "$LOG_FILE"
+ else
+ echo "WARNING: msmtp not found - ERROR email NOT sent" | tee -a "$LOG_FILE"
+ fi
+else
+ echo "Backup completed successfully with no errors" | tee -a "$LOG_FILE"
+
+ # Send success recap email
+ echo "Sending SUCCESS recap email to $EMAIL_TO" | tee -a "$LOG_FILE"
+ if command -v msmtp &> /dev/null; then
+ {
+ echo "To: $EMAIL_TO"
+ echo "Subject: Backup${BACKUP_SERVER} $DATE_SUBJECT"
+ echo ""
+ cat "$RECAP_FILE"
+ } | msmtp "$EMAIL_TO"
+ echo "SUCCESS recap email sent successfully to $EMAIL_TO" | tee -a "$LOG_FILE"
+ else
+ echo "WARNING: msmtp not found - SUCCESS recap email NOT sent" | tee -a "$LOG_FILE"
+ fi
+fi
+
+# Clean up recap file
+rm -f "$RECAP_FILE"
+
+# Exit with error code if there were errors
+if [[ $ERROR_COUNT -gt 0 ]]; then
+ exit 1
+fi
diff --git a/api/PM7/d6back.yaml b/api/PM7/d6back.yaml
new file mode 100644
index 00000000..53c933cf
--- /dev/null
+++ b/api/PM7/d6back.yaml
@@ -0,0 +1,112 @@
+# Configuration for MariaDB and directories backup
+# Backup structure: $dir_backup/$hostname/$containername/ for dirs
+# $dir_backup/$hostname/$containername/sql/ for databases
+
+# Global parameters
+global:
+ backup_server: PM7 # Nom du serveur de backup (PM7, PM1, etc.)
+ email_to: support@unikoffice.com # Email de notification
+ dir_backup: /var/pierre/back # Base backup directory
+ enc_key: /home/pierre/.key_enc # Encryption key for SQL backups
+ keep_dirs: 7d # Garde 7 jours pour les dirs
+ keep_db: 5d,3w,15m # 5 jours complets, 3 semaines (1/jour), 15 mois (1/semaine)
+
+# Hosts configuration
+hosts:
+ - name: IN2
+ ip: 145.239.9.105
+ user: debian
+ key: /home/pierre/.ssh/backup_key
+ port: 22
+ dirs:
+ - /etc/nginx
+ containers:
+ - name: nx4
+ db_user: root
+ db_pass: MyDebServer,90b
+ db_host: localhost
+ dirs:
+ - /etc/nginx
+ - /var/www
+ databases:
+ - ALL # Backup all databases
+ onlydb: # Used only with -onlydb parameter (optional)
+ - turing
+
+ - name: IN3
+ ip: 195.154.80.116
+ user: pierre
+ key: /home/pierre/.ssh/backup_key
+ port: 22
+ dirs:
+ - /etc/nginx
+ containers:
+ - name: nx4
+ db_user: root
+ db_pass: MyAlpLocal,90b
+ db_host: localhost
+ dirs:
+ - /etc/nginx
+ - /var/www
+ databases:
+ - ALL # Backup all databases
+ onlydb: # Used only with -onlydb parameter (optional)
+ - geosector
+
+ - name: rca-geo
+ dirs:
+ - /etc/nginx
+ - /var/www
+
+ - name: dva-res
+ db_user: root
+ db_pass: MyAlpineDb.90b
+ db_host: localhost
+ dirs:
+ - /etc/nginx
+ - /var/www
+ databases:
+ - ALL
+ onlydb:
+ - resalice
+
+ - name: dva-front
+ dirs:
+ - /etc/nginx
+ - /var/www
+
+ - name: maria3
+ db_user: root
+ db_pass: MyAlpLocal,90b
+ db_host: localhost
+ dirs:
+ - /etc/my.cnf.d
+ - /var/osm
+ - /var/log
+ databases:
+ - ALL
+ onlydb:
+ - cleo
+ - rca_geo
+
+ - name: IN4
+ ip: 51.159.7.190
+ user: pierre
+ key: /home/pierre/.ssh/backup_key
+ port: 22
+ dirs:
+ - /etc/nginx
+ containers:
+ - name: maria4
+ db_user: root
+ db_pass: MyAlpLocal,90b
+ db_host: localhost
+ dirs:
+ - /etc/my.cnf.d
+ - /var/osm
+ - /var/log
+ databases:
+ - ALL
+ onlydb:
+ - cleo
+ - pra_geo
diff --git a/api/PM7/decpm7.sh b/api/PM7/decpm7.sh
new file mode 100644
index 00000000..9299ae48
--- /dev/null
+++ b/api/PM7/decpm7.sh
@@ -0,0 +1,118 @@
+#!/bin/bash
+
+set -euo pipefail
+
+# Colors for output
+RED='\033[0;31m'
+GREEN='\033[0;32m'
+YELLOW='\033[1;33m'
+BLUE='\033[0;34m'
+NC='\033[0m' # No Color
+
+# Configuration
+CONFIG_FILE="backpm7.yaml"
+
+# Check if file argument is provided
+if [ $# -eq 0 ]; then
+ echo -e "${RED}Error: No input file specified${NC}"
+ echo "Usage: $0 "
+ echo "Example: $0 wordpress_20250905_14.sql.gz.enc"
+ exit 1
+fi
+
+INPUT_FILE="$1"
+
+# Check if input file exists
+if [ ! -f "$INPUT_FILE" ]; then
+ echo -e "${RED}Error: File not found: $INPUT_FILE${NC}"
+ exit 1
+fi
+
+# Function to load encryption key from config
+load_key_from_config() {
+ if [ ! -f "$CONFIG_FILE" ]; then
+ echo -e "${YELLOW}Warning: $CONFIG_FILE not found${NC}"
+ return 1
+ fi
+
+ # Check for yq
+ if ! command -v yq &> /dev/null; then
+ echo -e "${RED}Error: yq is required to read config file${NC}"
+ echo "Install with: sudo wget -qO /usr/local/bin/yq https://github.com/mikefarah/yq/releases/latest/download/yq_linux_amd64 && sudo chmod +x /usr/local/bin/yq"
+ return 1
+ fi
+
+ local key_path=$(yq '.global.enc_key' "$CONFIG_FILE" | tr -d '"')
+
+ if [ -z "$key_path" ]; then
+ echo -e "${RED}Error: enc_key not found in $CONFIG_FILE${NC}"
+ return 1
+ fi
+
+ if [ ! -f "$key_path" ]; then
+ echo -e "${RED}Error: Encryption key file not found: $key_path${NC}"
+ return 1
+ fi
+
+ ENC_KEY=$(cat "$key_path")
+ echo -e "${GREEN}Encryption key loaded from: $key_path${NC}"
+ return 0
+}
+
+# Check file type early - accept both old and new naming
+if [[ "$INPUT_FILE" != *.sql.gz.enc ]] && [[ "$INPUT_FILE" != *.sql.tar.gz.enc ]]; then
+ echo -e "${RED}Error: File must be a .sql.gz.enc or .sql.tar.gz.enc file${NC}"
+ echo "This tool only decrypts SQL backup files created by backpm7.sh"
+ exit 1
+fi
+
+# Get encryption key from config
+if ! load_key_from_config; then
+ echo -e "${RED}Error: Cannot load encryption key${NC}"
+ echo "Make sure $CONFIG_FILE exists and contains enc_key path"
+ exit 1
+fi
+
+# Process SQL backup file
+echo -e "${BLUE}Decrypting SQL backup: $INPUT_FILE${NC}"
+
+# Determine output file - extract just the filename and put in current directory
+BASENAME=$(basename "$INPUT_FILE")
+if [[ "$BASENAME" == *.sql.tar.gz.enc ]]; then
+ OUTPUT_FILE="${BASENAME%.sql.tar.gz.enc}.sql"
+else
+ OUTPUT_FILE="${BASENAME%.sql.gz.enc}.sql"
+fi
+
+# Decrypt and decompress in one command
+echo "Decrypting to: $OUTPUT_FILE"
+
+# Decrypt and decompress in one pipeline
+if openssl enc -aes-256-cbc -d -salt -pass pass:"$ENC_KEY" -pbkdf2 -in "$INPUT_FILE" | gunzip > "$OUTPUT_FILE" 2>/dev/null; then
+ # Get file size
+ size=$(du -h "$OUTPUT_FILE" | cut -f1)
+ echo -e "${GREEN}✓ Successfully decrypted: $OUTPUT_FILE ($size)${NC}"
+
+ # Show first few lines of SQL
+ echo -e "${BLUE}First 5 lines of SQL:${NC}"
+ head -n 5 "$OUTPUT_FILE"
+else
+ echo -e "${RED}✗ Decryption failed${NC}"
+ echo "Possible causes:"
+ echo " - Wrong encryption key"
+ echo " - Corrupted file"
+ echo " - File was encrypted differently"
+
+ # Try to help debug
+ echo -e "\n${YELLOW}Debug info:${NC}"
+ echo "File size: $(du -h "$INPUT_FILE" | cut -f1)"
+ echo "First bytes (should start with 'Salted__'):"
+ hexdump -C "$INPUT_FILE" | head -n 1
+
+ # Let's also check what key we're using (first 10 chars)
+ echo "Key begins with: ${ENC_KEY:0:10}..."
+
+ exit 1
+fi
+
+echo -e "${GREEN}Operation completed successfully${NC}"
\ No newline at end of file
diff --git a/api/PM7/sync_geosector.sh b/api/PM7/sync_geosector.sh
new file mode 100644
index 00000000..fb7ef718
--- /dev/null
+++ b/api/PM7/sync_geosector.sh
@@ -0,0 +1,248 @@
+#!/bin/bash
+#
+# sync_geosector.sh - Synchronise les backups geosector depuis PM7 vers maria3 (IN3) et maria4 (IN4)
+#
+# Ce script :
+# 1. Trouve le dernier backup chiffré de geosector sur PM7
+# 2. Le déchiffre et décompresse localement
+# 3. Le transfère et l'importe dans IN3/maria3/geosector
+# 4. Le transfère et l'importe dans IN4/maria4/geosector
+#
+# Installation: /var/pierre/bat/sync_geosector.sh
+# Usage: ./sync_geosector.sh [--force] [--date YYYYMMDD_HH]
+#
+
+set -uo pipefail
+# Note: Removed -e to allow script to continue on sync errors
+# Errors are handled explicitly with ERROR_COUNT
+
+# Configuration
+SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
+CONFIG_FILE="$SCRIPT_DIR/d6back.yaml"
+BACKUP_DIR="/var/pierre/back/IN3/nx4/sql"
+ENC_KEY_FILE="/home/pierre/.key_enc"
+SSH_KEY="/home/pierre/.ssh/backup_key"
+TEMP_DIR="/tmp/geosector_sync"
+LOG_FILE="/var/pierre/bat/logs/sync_geosector.log"
+RECAP_FILE="/tmp/sync_geosector_recap_$$.txt"
+
+# Load email config from d6back.yaml
+if [[ -f "$CONFIG_FILE" ]]; then
+ EMAIL_TO=$(yq '.global.email_to // "support@unikoffice.com"' "$CONFIG_FILE" | tr -d '"')
+ BACKUP_SERVER=$(yq '.global.backup_server // "BACKUP"' "$CONFIG_FILE" | tr -d '"')
+else
+ EMAIL_TO="support@unikoffice.com"
+ BACKUP_SERVER="BACKUP"
+fi
+
+# Serveurs cibles
+IN3_HOST="195.154.80.116"
+IN3_USER="pierre"
+IN3_CONTAINER="maria3"
+
+IN4_HOST="51.159.7.190"
+IN4_USER="pierre"
+IN4_CONTAINER="maria4"
+
+# Credentials MariaDB
+DB_USER="root"
+IN3_DB_PASS="MyAlpLocal,90b" # maria3
+IN4_DB_PASS="MyAlpLocal,90b" # maria4
+DB_NAME="geosector"
+
+# Fonctions utilitaires
+log() {
+ echo "[$(date '+%Y-%m-%d %H:%M:%S')] $*" | tee -a "$LOG_FILE"
+}
+
+error() {
+ log "ERROR: $*"
+ exit 1
+}
+
+cleanup() {
+ if [[ -d "$TEMP_DIR" ]]; then
+ log "Nettoyage de $TEMP_DIR"
+ rm -rf "$TEMP_DIR"
+ fi
+ rm -f "$RECAP_FILE"
+}
+
+trap cleanup EXIT
+
+# Lecture de la clé de chiffrement
+if [[ ! -f "$ENC_KEY_FILE" ]]; then
+ error "Clé de chiffrement non trouvée: $ENC_KEY_FILE"
+fi
+ENC_KEY=$(cat "$ENC_KEY_FILE")
+
+# Parsing des arguments
+FORCE=0
+SPECIFIC_DATE=""
+
+while [[ $# -gt 0 ]]; do
+ case $1 in
+ --force)
+ FORCE=1
+ shift
+ ;;
+ --date)
+ SPECIFIC_DATE="$2"
+ shift 2
+ ;;
+ *)
+ echo "Usage: $0 [--force] [--date YYYYMMDD_HH]"
+ exit 1
+ ;;
+ esac
+done
+
+# Trouver le fichier backup
+if [[ -n "$SPECIFIC_DATE" ]]; then
+ BACKUP_FILE="$BACKUP_DIR/geosector_${SPECIFIC_DATE}.sql.gz.enc"
+ if [[ ! -f "$BACKUP_FILE" ]]; then
+ error "Backup non trouvé: $BACKUP_FILE"
+ fi
+else
+ # Chercher le plus récent
+ BACKUP_FILE=$(find "$BACKUP_DIR" -name "geosector_*.sql.gz.enc" -type f -printf '%T@ %p\n' | sort -rn | head -1 | cut -d' ' -f2-)
+ if [[ -z "$BACKUP_FILE" ]]; then
+ error "Aucun backup geosector trouvé dans $BACKUP_DIR"
+ fi
+fi
+
+BACKUP_BASENAME=$(basename "$BACKUP_FILE")
+log "Backup sélectionné: $BACKUP_BASENAME"
+
+# Initialiser le fichier récapitulatif
+echo "SYNC GEOSECTOR REPORT - $(hostname) - $(date '+%d.%m.%Y %H')h" > "$RECAP_FILE"
+echo "========================================" >> "$RECAP_FILE"
+echo "" >> "$RECAP_FILE"
+echo "Backup source: $BACKUP_BASENAME" >> "$RECAP_FILE"
+echo "" >> "$RECAP_FILE"
+
+# Créer le répertoire temporaire
+mkdir -p "$TEMP_DIR"
+DECRYPTED_FILE="$TEMP_DIR/geosector.sql"
+
+# Étape 1: Déchiffrer et décompresser
+log "Déchiffrement et décompression du backup..."
+if ! openssl enc -aes-256-cbc -d -pass pass:"$ENC_KEY" -pbkdf2 -in "$BACKUP_FILE" | gunzip > "$DECRYPTED_FILE"; then
+ error "Échec du déchiffrement/décompression"
+fi
+
+FILE_SIZE=$(du -h "$DECRYPTED_FILE" | cut -f1)
+log "Fichier SQL déchiffré: $FILE_SIZE"
+
+echo "Decrypted SQL size: $FILE_SIZE" >> "$RECAP_FILE"
+echo "" >> "$RECAP_FILE"
+
+# Compteur d'erreurs
+ERROR_COUNT=0
+
+# Fonction pour synchroniser vers un serveur
+sync_to_server() {
+ local HOST=$1
+ local USER=$2
+ local CONTAINER=$3
+ local DB_PASS=$4
+ local SERVER_NAME=$5
+
+ log "=== Synchronisation vers $SERVER_NAME ($HOST) ==="
+ echo "TARGET: $SERVER_NAME ($HOST/$CONTAINER)" >> "$RECAP_FILE"
+
+ # Test de connexion SSH
+ if ! ssh -i "$SSH_KEY" -o ConnectTimeout=10 "$USER@$HOST" "echo 'SSH OK'" &>/dev/null; then
+ log "ERROR: Impossible de se connecter à $HOST via SSH"
+ echo " ✗ SSH connection FAILED" >> "$RECAP_FILE"
+ ((ERROR_COUNT++))
+ return 1
+ fi
+
+ # Import dans MariaDB
+ log "Import dans $SERVER_NAME/$CONTAINER/geosector..."
+
+ # Drop et recréer la base sur le serveur distant
+ if ! ssh -i "$SSH_KEY" "$USER@$HOST" "incus exec $CONTAINER --project default -- mariadb -u root -p'$DB_PASS' -e 'DROP DATABASE IF EXISTS $DB_NAME; CREATE DATABASE $DB_NAME CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;'"; then
+ log "ERROR: Échec de la création de la base sur $SERVER_NAME"
+ echo " ✗ Database creation FAILED" >> "$RECAP_FILE"
+ ((ERROR_COUNT++))
+ return 1
+ fi
+
+ # Filtrer et importer le SQL (enlever CREATE DATABASE et USE avec timestamp)
+ log "Filtrage et import du SQL..."
+ if ! sed -e '/^CREATE DATABASE.*geosector_[0-9]/d' \
+ -e '/^USE.*geosector_[0-9]/d' \
+ "$DECRYPTED_FILE" | \
+ ssh -i "$SSH_KEY" "$USER@$HOST" "incus exec $CONTAINER --project default -- mariadb -u root -p'$DB_PASS' $DB_NAME"; then
+ log "ERROR: Échec de l'import sur $SERVER_NAME"
+ echo " ✗ SQL import FAILED" >> "$RECAP_FILE"
+ ((ERROR_COUNT++))
+ return 1
+ fi
+
+ log "$SERVER_NAME: Import réussi"
+ echo " ✓ Import SUCCESS" >> "$RECAP_FILE"
+ echo "" >> "$RECAP_FILE"
+}
+
+# Synchronisation vers IN3/maria3
+sync_to_server "$IN3_HOST" "$IN3_USER" "$IN3_CONTAINER" "$IN3_DB_PASS" "IN3/maria3"
+
+# Synchronisation vers IN4/maria4
+sync_to_server "$IN4_HOST" "$IN4_USER" "$IN4_CONTAINER" "$IN4_DB_PASS" "IN4/maria4"
+
+# Finaliser le récapitulatif
+echo "========================================" >> "$RECAP_FILE"
+echo "COMPLETED: $(date '+%d.%m.%Y %H:%M')" >> "$RECAP_FILE"
+
+# Préparer le sujet email avec date
+DATE_SUBJECT=$(date '+%d.%m.%Y %H')
+
+# Envoyer l'email récapitulatif
+if [[ $ERROR_COUNT -gt 0 ]]; then
+ log "Total errors: $ERROR_COUNT"
+
+ # Ajouter les erreurs au récap
+ echo "" >> "$RECAP_FILE"
+ echo "ERRORS DETECTED: $ERROR_COUNT" >> "$RECAP_FILE"
+ echo "----------------------------" >> "$RECAP_FILE"
+ grep -i "ERROR" "$LOG_FILE" | tail -20 >> "$RECAP_FILE"
+
+ # Envoyer email avec ERROR dans le sujet
+ log "Sending ERROR email to $EMAIL_TO (Errors found: $ERROR_COUNT)"
+ if command -v msmtp &> /dev/null; then
+ {
+ echo "To: $EMAIL_TO"
+ echo "Subject: Sync${BACKUP_SERVER} ERROR $DATE_SUBJECT"
+ echo ""
+ cat "$RECAP_FILE"
+ } | msmtp "$EMAIL_TO"
+ log "ERROR email sent successfully to $EMAIL_TO"
+ else
+ log "WARNING: msmtp not found - ERROR email NOT sent"
+ fi
+
+ log "=== Synchronisation terminée avec des erreurs ==="
+ exit 1
+else
+ log "=== Synchronisation terminée avec succès ==="
+ log "Les bases geosector sur maria3 et maria4 sont à jour avec le backup $BACKUP_BASENAME"
+
+ # Envoyer email de succès
+ log "Sending SUCCESS recap email to $EMAIL_TO"
+ if command -v msmtp &> /dev/null; then
+ {
+ echo "To: $EMAIL_TO"
+ echo "Subject: Sync${BACKUP_SERVER} $DATE_SUBJECT"
+ echo ""
+ cat "$RECAP_FILE"
+ } | msmtp "$EMAIL_TO"
+ log "SUCCESS recap email sent successfully to $EMAIL_TO"
+ else
+ log "WARNING: msmtp not found - SUCCESS recap email NOT sent"
+ fi
+
+ exit 0
+fi
diff --git a/api/TODO-API.md b/api/TODO-API.md
index 0cb1da2e..3d029eff 100644
--- a/api/TODO-API.md
+++ b/api/TODO-API.md
@@ -5,21 +5,25 @@
### 🔴 PRIORITÉ HAUTE
#### 1. Système de backup pour les suppressions (DELETE)
+
**Demandé le :** 20/08/2025
**Objectif :** Sauvegarder toutes les données supprimées (soft delete) dans un fichier SQL pour pouvoir les restaurer en cas d'erreur humaine.
**Détails techniques :**
+
- Créer un système de backup automatique lors de chaque DELETE
- Stocker les données dans un fichier SQL avec structure permettant la réintégration facile
- Format suggéré : `/backups/deleted/{année}/{mois}/deleted_{table}_{YYYYMMDD}.sql`
**Tables concernées :**
+
- `ope_pass` (passages) - DELETE /passages/{id}
- `users` (utilisateurs) - DELETE /users/{id}
- `operations` (opérations) - DELETE /operations/{id}
- `ope_sectors` (secteurs) - DELETE /sectors/{id}
**Structure du backup suggérée :**
+
```sql
-- Backup deletion: ope_pass
-- Date: 2025-08-20 14:30:45
@@ -54,16 +58,20 @@ INSERT INTO ope_pass_backup (
```
**Fonctionnalités à implémenter :**
+
1. **Service de backup** : `BackupService.php`
+
- Méthode `backupDeletedRecord($table, $id, $data)`
- Génération automatique du SQL de restauration
- Rotation des fichiers (garder 90 jours)
2. **Intégration dans les controllers**
+
- Ajouter l'appel au BackupService avant chaque soft delete
- Logger l'emplacement du backup
3. **Interface de restauration** (optionnel)
+
- Endpoint GET /api/backups/deleted pour lister les backups
- Endpoint POST /api/backups/restore/{backup_id} pour restaurer
@@ -71,6 +79,7 @@ INSERT INTO ope_pass_backup (
- Script PHP : `php scripts/restore_deleted.php --table=ope_pass --id=19500576`
**Avantages :**
+
- Traçabilité complète des suppressions
- Restauration rapide en cas d'erreur
- Audit trail pour conformité
@@ -78,33 +87,1132 @@ INSERT INTO ope_pass_backup (
---
+### 🔴 PRIORITÉ HAUTE
+
+#### 2. Migration des bases de données vers container maria3 centralisé
+
+**Demandé le :** 07/10/2025
+**Objectif :** Migrer les bases de données locales des containers dva-geo et rca-geo vers un container MariaDB centralisé maria3 sur le même host IN3.
+
+**Architecture actuelle :**
+
+- **dva-geo** : MariaDB local avec base `geo_app` (localhost)
+- **rca-geo** : MariaDB local avec base `geo_app` (localhost)
+- **maria3** : Container MariaDB 11.4 existant (IP: 13.23.33.4) - utilisé uniquement pour la base `adresses`
+
+**Architecture cible :**
+
+- **maria3** : Container centralisé avec :
+ - Base `dva_geo` pour l'environnement DEV
+ - Base `rca_geo` pour l'environnement RECETTE
+ - Base `adresses` (déjà existante)
+- **dva-geo** : Suppression du serveur MariaDB local
+- **rca-geo** : Suppression du serveur MariaDB local
+
+**Avantages :**
+
+- ✅ Centralisation des bases de données
+- ✅ Facilite les sauvegardes
+- ✅ Réduction de la consommation mémoire des containers API
+- ✅ Séparation claire des responsabilités (API vs DB)
+- ✅ Préparation pour architecture production (IN4/maria4/pra_geo)
+
+---
+
+#### 📋 TODOLIST DÉTAILLÉE - ENVIRONNEMENT DVA-GEO (DEV)
+
+##### Phase 1️⃣ : Préparation et sauvegarde (AVANT migration)
+
+- [x] **1.1** Vérifier l'état actuel de la base dans dva-geo
+
+ ```bash
+ incus exec dva-geo -- mysql -u root -pMyAlpineDb.90b -e "SHOW DATABASES;"
+ incus exec dva-geo -- mysql -u root -pMyAlpineDb.90b geo_app -e "SHOW TABLES;"
+ incus exec dva-geo -- mysql -u root -pMyAlpineDb.90b geo_app -e "SELECT COUNT(*) FROM users;"
+ ```
+
+- [x] **1.2** Créer une sauvegarde complète de la base actuelle
+
+ ```bash
+ # Dump avec skip-lock-tables (vue problématique v_stripe_amicale_dashboard)
+ incus exec dva-geo -- mariadb-dump -u root -pMyAlpineDb.90b --skip-lock-tables geo_app > /var/back/dva_geo_backup_20251007.sql
+ ```
+
+- [x] **1.3** Vérifier l'intégrité de la sauvegarde
+
+ ```bash
+ ls -lh /var/back/dva_geo_backup_20251007.sql
+ # Taille : 1.2GB (1217046899 bytes)
+ ```
+
+- [x] **1.4** Supprimer la vue problématique et refaire un dump propre
+
+ ```bash
+ # Suppression de la vue cassée
+ incus exec dva-geo -- mysql -u root -pMyAlpineDb.90b geo_app -e "DROP VIEW IF EXISTS v_stripe_amicale_dashboard;"
+
+ # Nouveau dump complet et propre
+ incus exec dva-geo -- mariadb-dump -u root -pMyAlpineDb.90b geo_app > /var/back/dva_geo_backup_final_20251007.sql
+ ```
+
+##### Phase 2️⃣ : Configuration de maria3
+
+- [x] **2.1** Se connecter à maria3 et créer la base dva_geo
+
+ ```bash
+ incus exec maria3 -- mysql -u root -p'MyAlpLocal,90b' -e "CREATE DATABASE IF NOT EXISTS dva_geo CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;"
+ ```
+
+- [x] **2.2** Créer l'utilisateur dédié dva_geo_user
+
+ ```bash
+ incus exec maria3 -- mysql -u root -p'MyAlpLocal,90b' -e "
+ CREATE USER IF NOT EXISTS 'dva_geo_user'@'%' IDENTIFIED BY 'CBq9tKHj6PGPZuTmAHV7';
+ GRANT ALL PRIVILEGES ON dva_geo.* TO 'dva_geo_user'@'%';
+ FLUSH PRIVILEGES;"
+ ```
+
+- [x] **2.3** Vérifier les permissions
+
+ ```bash
+ incus exec maria3 -- mysql -u root -p'MyAlpLocal,90b' -e "SHOW GRANTS FOR 'dva_geo_user'@'%';"
+ # Résultat : ALL PRIVILEGES sur dva_geo.*
+ ```
+
+- [x] **2.4** Tester la connexion avec le nouvel utilisateur depuis dva-geo
+ ```bash
+ incus exec dva-geo -- mysql -h 13.23.33.4 -u dva_geo_user -p'CBq9tKHj6PGPZuTmAHV7' -e "SHOW DATABASES;"
+ # ✅ Connexion réussie
+ ```
+
+##### Phase 3️⃣ : Migration des données
+
+- [x] **3.1** Copier le dump depuis le host vers maria3
+
+ ```bash
+ incus file push /var/back/dva_geo_backup_final_20251007.sql maria3/tmp/
+ # ✅ Fichier transféré
+ ```
+
+- [x] **3.2** Importer le dump dans maria3
+
+ ```bash
+ incus exec maria3 -- mysql -u root -p'MyAlpLocal,90b' dva_geo < /tmp/dva_geo_backup_final_20251007.sql
+ # ✅ Import réussi
+ ```
+
+- [x] **3.3** Vérifier l'importation
+
+ ```bash
+ incus exec maria3 -- mysql -u root -p'MyAlpLocal,90b' dva_geo -e "SHOW TABLES;"
+ incus exec maria3 -- mysql -u root -p'MyAlpLocal,90b' dva_geo -e "SELECT COUNT(*) FROM users;"
+ incus exec maria3 -- mysql -u root -p'MyAlpLocal,90b' dva_geo -e "SELECT COUNT(*) FROM operations;"
+ incus exec maria3 -- mysql -u root -p'MyAlpLocal,90b' dva_geo -e "SELECT COUNT(*) FROM ope_pass;"
+ # ✅ Toutes les tables présentes
+ ```
+
+- [x] **3.4** Comparer les comptages avec la base source
+
+ ```bash
+ # Base source (dva-geo localhost)
+ incus exec dva-geo -- mysql -u root -pMyAlpineDb.90b geo_app -e "
+ SELECT 'users' as table_name, COUNT(*) as count FROM users
+ UNION ALL SELECT 'operations', COUNT(*) FROM operations
+ UNION ALL SELECT 'ope_pass', COUNT(*) FROM ope_pass
+ UNION ALL SELECT 'entites', COUNT(*) FROM entites;"
+
+ # Base cible (maria3)
+ incus exec maria3 -- mysql -u root -p'MyAlpLocal,90b' dva_geo -e "
+ SELECT 'users' as table_name, COUNT(*) as count FROM users
+ UNION ALL SELECT 'operations', COUNT(*) FROM operations
+ UNION ALL SELECT 'ope_pass', COUNT(*) FROM ope_pass
+ UNION ALL SELECT 'entites', COUNT(*) FROM entites;"
+
+ # ✅ Comptages identiques confirmés
+ ```
+
+##### Phase 4️⃣ : Configuration de l'API
+
+- [x] **4.1** Mettre à jour AppConfig.php dans dva-geo
+
+ ```php
+ // Fichier: src/Config/AppConfig.php (local)
+ // Lignes 153-165
+
+ // Configuration maria3 activée (migration effectuée le 07/10/2025)
+ 'database' => [
+ 'host' => '13.23.33.4', // Container maria3 sur IN3
+ 'name' => 'dva_geo',
+ 'username' => 'dva_geo_user',
+ 'password' => 'CBq9tKHj6PGPZuTmAHV7',
+ ],
+ ```
+
+- [x] **4.2** Déployer la nouvelle configuration
+
+ ```bash
+ # Depuis /home/pierre/dev/geosector/api
+ ./deploy-api.sh
+ # ✅ Déploiement réussi sur dva-geo
+ ```
+
+- [x] **4.3** Redémarrer PHP-FPM
+ ```bash
+ incus exec dva-geo -- rc-service php-fpm83 restart
+ # ✅ PHP-FPM redémarré
+ ```
+
+##### Phase 5️⃣ : Tests de l'API
+
+- [x] **5.1** Tester la connexion API à la base
+
+ ```bash
+ curl -X GET https://dapp.geosector.fr/api/health
+ # ✅ API opérationnelle
+ ```
+
+- [x] **5.2** Tester l'authentification
+
+ ```bash
+ curl -X POST https://dapp.geosector.fr/api/login \
+ -H "Content-Type: application/json" \
+ -d '{"username":"pv_admin","password":"mot_de_passe"}'
+ # ✅ Login réussi
+ ```
+
+- [x] **5.3** Tester la récupération de données
+
+ ```bash
+ curl -X GET https://dapp.geosector.fr/api/users \
+ -H "Authorization: Bearer {session_id}"
+ # ✅ Données récupérées depuis maria3
+ ```
+
+- [x] **5.4** Tester l'application Flutter
+
+ ```bash
+ # Test depuis l'application mobile Flutter
+ # ✅ Connexion, récupération données, création passages : OK
+ ```
+
+- [x] **5.5** Vérifier les logs
+ ```bash
+ incus exec dva-geo -- tail -50 /var/www/geosector/api/logs/app.log
+ incus exec dva-geo -- tail -50 /var/log/nginx/dva-api-error.log
+ # ✅ Aucune erreur détectée
+ ```
+
+##### Phase 6️⃣ : Nettoyage et suppression de MariaDB local
+
+⚠️ **Migration validée et fonctionnelle** - Nettoyage effectué le 07/10/2025
+
+- [x] **6.1** Faire une dernière sauvegarde de sécurité
+
+ ```bash
+ incus exec dva-geo -- mariadb-dump -u root -pMyAlpineDb.90b --skip-lock-tables geo_app > /var/back/dva_geo_FINAL_backup_$(date +%Y%m%d_%H%M%S).sql
+ # ✅ Sauvegarde finale créée
+ ```
+
+- [x] **6.2** Arrêter le serveur MariaDB local dans dva-geo
+
+ ```bash
+ incus exec dva-geo -- rc-service mariadb stop
+ # ✅ MariaDB arrêté
+ ```
+
+- [x] **6.3** Désactiver le démarrage automatique de MariaDB
+
+ ```bash
+ incus exec dva-geo -- rc-update del mariadb default
+ # ✅ MariaDB retiré du runlevel default
+ ```
+
+- [x] **6.4** Désinstaller MariaDB
+
+ ```bash
+ incus exec dva-geo -- apk del mariadb mariadb-client mariadb-common
+ # ✅ MariaDB désinstallé du container
+ ```
+
+- [x] **6.5** Archiver les données MariaDB locales
+
+ ```bash
+ incus exec dva-geo -- tar -czf /var/back/mysql_data_archive_$(date +%Y%m%d).tar.gz /var/lib/mysql
+ # ✅ Archive créée
+ ```
+
+- [x] **6.6** Supprimer les données MariaDB locales
+ ```bash
+ incus exec dva-geo -- rm -rf /var/lib/mysql /run/mysqld
+ # ✅ Données supprimées, espace disque libéré
+ ```
+
+##### Phase 7️⃣ : Documentation
+
+- [x] **7.1** Mettre à jour TODO-API.md avec toutes les commandes
+
+ - Toutes les phases documentées avec commandes réelles
+ - Statut mis à jour : ✅ TERMINÉ
+
+- [x] **7.2** Documenter les problèmes rencontrés
+
+ - **Problème 1** : Vue `v_stripe_amicale_dashboard` cassée
+ - **Solution** : Suppression de la vue avant dump final
+ - **Fichier** : `scripts/migrations/stripe_tables.sql` (ligne 183)
+ - **Problème 2** : Erreur LOCK TABLES lors du dump initial
+ - **Solution** : Ajout de `--skip-lock-tables` au mysqldump
+
+- [ ] **7.3** Mettre à jour TECHBOOK.md avec la nouvelle architecture
+ - Section "Base de données" (lignes 136-154)
+ - Mettre à jour le tableau pour refléter la configuration DVA-GEO → maria3
+
+---
+
+#### 📋 PROCHAINES ÉTAPES (APRÈS DVA-GEO)
+
+Une fois la migration DVA-GEO validée et stable :
+
+### ✅ Migration RCA-GEO (RECETTE) - TERMINÉE
+
+**Statut :** ✅ **MIGRATION COMPLÉTÉE AVEC SUCCÈS** (16/10/2025)
+
+**Préparation effectuée le 07/10/2025** :
+
+- ✅ Base `rca_geo` créée dans maria3 (vide)
+- ✅ Utilisateur `rca_geo_user` créé avec ALL PRIVILEGES
+- ✅ Vue problématique `v_stripe_amicale_dashboard` supprimée de la base source (rca-geo localhost)
+
+**Migration réalisée le 16/10/2025** :
+
+- ✅ Dump de la base `geo_app` depuis rca-geo (localhost)
+- ✅ Import dans maria3 `rca_geo`
+- ✅ Validation des comptages (toutes les tables migrées correctement)
+- ✅ Modification AppConfig.php lignes 113-128
+- ✅ Déploiement manuel sur rca-geo
+- ✅ Tests complets sur https://rapp.geosector.fr/api/
+- ✅ Suppression complète de MariaDB local dans rca-geo
+- ✅ Suppression des dossiers `/var/lib/mysql` et `/run/mysqld`
+
+**Configuration finale** :
+
+- User : `rca_geo_user`
+- Password : `UPf3C0cQ805LypyM71iW`
+- Host : `13.23.33.4` (maria3 sur IN3)
+- Base : `rca_geo`
+- Privileges : ALL PRIVILEGES sur rca_geo.\*
+
+**Résultat** :
+
+- Application fonctionnelle en RECETTE avec base centralisée sur maria3
+- Container rca-geo allégé (plus de serveur MariaDB local)
+- Architecture cohérente DVA-GEO et RCA-GEO → maria3
+
+---
+
+### 🔵 Création environnement PRODUCTION (PLANIFIÉ JEUDI 10/10/2025 16h)
+
+**Architecture cible** :
+
+- Serveur IN4 (51.159.7.190)
+- Container `pra-geo` pour l'API (exporté depuis dva-geo)
+- Container `maria4` pour la base de données
+- Base : `pra_geo` avec utilisateur dédié
+- Données dupliquées depuis rca_geo (IN3/maria3)
+
+---
+
+#### 📋 TODOLIST DÉTAILLÉE - ENVIRONNEMENT PRODUCTION (PRA-GEO)
+
+##### Phase 0️⃣ : Préparation du serveur IN4
+
+- [x] **0.1** Vérifier l'accès SSH au serveur IN4
+
+ ```bash
+ ssh root@51.159.7.190
+ # ✅ Serveur accessible
+ ```
+
+- [x] **0.2** Vérifier Incus sur IN4
+
+ ```bash
+ ssh root@51.159.7.190 "incus list"
+ # ✅ Incus opérationnel
+ ```
+
+- [x] **0.3** Vérifier l'espace disque disponible
+
+ ```bash
+ ssh root@51.159.7.190 "df -h"
+ # ✅ Espace suffisant
+ ```
+
+- [x] **0.4** Préparer le répertoire de transfert
+ ```bash
+ ssh root@51.159.7.190 "mkdir -p /var/back/imports"
+ # ✅ Répertoire créé
+ ```
+
+##### Phase 1️⃣ : Export du container dva-geo depuis IN3
+
+- [x] **1.1** Export dva-geo réalisé
+
+ ```bash
+ # ✅ Container dva-geo exporté depuis IN3
+ # Note: Export effectué, détails exacts non documentés
+ ```
+
+- [x] **1.2** Snapshot et archive créés
+ ```bash
+ # ✅ Archive dva-geo créée et transférée vers IN4
+ ```
+
+##### Phase 2️⃣ : Transfert vers IN4
+
+- [x] **2.1** Archive transférée vers IN4
+ ```bash
+ # ✅ Archive transférée sur IN4
+ ```
+
+##### Phase 3️⃣ : Import et configuration de pra-geo sur IN4
+
+- [x] **3.1** Container pra-geo importé et lancé
+
+ ```bash
+ # ✅ Container pra-geo créé sur IN4
+ # IP: 13.23.33.22 (réseau Incus)
+ ```
+
+- [x] **3.2** Configuration réseau vérifiée
+
+ ```bash
+ # ✅ Réseau configuré, ping et connectivité OK
+ ```
+
+- [x] **3.3** Container pra-geo opérationnel
+
+ ```bash
+ # ✅ Container démarré et accessible
+ ```
+
+- [x] **3.4** Client MariaDB installé
+ ```bash
+ # ✅ mariadb-client installé sur pra-geo
+ ```
+
+##### Phase 4️⃣ : Création du container maria4 sur IN4
+
+- [x] **4.1** Container maria4 créé
+
+ ```bash
+ # ✅ Container maria4 créé sur IN4
+ # IP: 13.23.33.4 (même IP que maria3 pour cohérence)
+ ```
+
+- [x] **4.2** MariaDB installé et initialisé
+
+ ```bash
+ # ✅ MariaDB 11.4 installé et démarré
+ # ✅ Mot de passe root: MyAlpLocal,90b
+ ```
+
+- [x] **4.3** Connexions distantes autorisées
+
+ ```bash
+ # ✅ bind-address = 0.0.0.0
+ # ✅ MariaDB redémarré
+ ```
+
+- [x] **4.4** Base pra_geo créée
+
+ ```bash
+ # ✅ Base: pra_geo (utf8mb4_unicode_ci)
+ # ✅ User: pra_geo_user / d2jAAGGWi8fxFrWgXjOA
+ # ✅ ALL PRIVILEGES accordés
+ ```
+
+- [x] **4.5** Base adresses présente
+
+ ```bash
+ # ✅ Base adresses avec tables par département (cp01, cp02, etc.)
+ # ✅ User: adr_geo_user@13.23.33.2% / d66,AdrGeoPrd.User
+ # ✅ SELECT privileges accordés
+ ```
+
+- [x] **4.6** Firewall UFW configuré
+
+ ```bash
+ # ✅ UFW: allow from 13.23.33.0/24 to any port 3306
+ # ✅ Connexions depuis pra-geo (13.23.33.22) opérationnelles
+ ```
+
+- [x] **4.7** Tests de connexion réussis
+ ```bash
+ # ✅ pra-geo → maria4 (pra_geo): OK - 24 090 users
+ # ✅ pra-geo → maria4 (adresses): OK - Tables accessibles
+ ```
+
+##### Phase 5️⃣ : Migration des données depuis dva_geo (IN3/maria3)
+
+- [x] **5.1** Dump dva_geo réalisé
+
+ ```bash
+ # ✅ Dump de dva_geo (sans vue problématique v_stripe_amicale_dashboard)
+ # Source: dva_geo (maria3 sur IN3)
+ ```
+
+- [x] **5.2** Données importées dans pra_geo
+
+ ```bash
+ # ✅ Import réussi dans maria4/pra_geo
+ ```
+
+- [x] **5.3** Vérification des données
+ ```bash
+ # ✅ 24 090 utilisateurs dans pra_geo
+ # ✅ Tables operations, ope_pass, entites présentes
+ ```
+
+**Note** : La base `pra_geo` a été initialisée depuis `dva_geo` (DEV) et non depuis `rca_geo` (REC).
+Pour la mise en production finale, il faudra probablement migrer depuis `rca_geo` (données de recette validées).
+
+##### Phase 6️⃣ : Configuration de l'API pour PRODUCTION
+
+- [x] **6.1** AppConfig.php modifié localement
+
+ ```php
+ // Configuration PRODUCTION (lignes 84-111)
+ 'database' => [
+ 'host' => '13.23.33.4', // ✅ maria4 sur IN4
+ 'name' => 'pra_geo',
+ 'username' => 'pra_geo_user',
+ 'password' => 'd2jAAGGWi8fxFrWgXjOA',
+ ],
+ 'addresses_database' => [
+ 'host' => '13.23.33.4', // ✅ maria4 sur IN4
+ 'name' => 'adresses',
+ 'username' => 'adr_geo_user',
+ 'password' => 'd66,AdrGeoPrd.User',
+ ],
+ ```
+
+- [ ] **6.2** Déployer en PRODUCTION
+
+ ```bash
+ # ⏳ EN ATTENTE validation client
+ ./deploy-api.sh pra
+ ```
+
+- [ ] **6.3** Redémarrer PHP-FPM
+ ```bash
+ # ⏳ À faire après déploiement
+ ssh root@51.159.7.190 "incus exec pra-geo -- rc-service php-fpm83 restart"
+ ```
+
+##### Phase 7️⃣ : Configuration DNS et reverse proxy
+
+- [ ] **7.1** Vérifier la configuration NGINX sur IN4
+
+ ```bash
+ ssh root@51.159.7.190 "cat /etc/nginx/sites-available/app.geosector.fr"
+ ```
+
+- [ ] **7.2** Configurer le reverse proxy vers pra-geo
+
+ ```nginx
+ # /etc/nginx/sites-available/app.geosector.fr
+ upstream pra_geo_backend {
+ server 13.23.34.43:80;
+ }
+
+ server {
+ listen 443 ssl http2;
+ server_name app.geosector.fr;
+
+ ssl_certificate /etc/letsencrypt/live/app.geosector.fr/fullchain.pem;
+ ssl_certificate_key /etc/letsencrypt/live/app.geosector.fr/privkey.pem;
+
+ location / {
+ proxy_pass http://pra_geo_backend;
+ proxy_set_header Host $host;
+ proxy_set_header X-Real-IP $remote_addr;
+ proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
+ proxy_set_header X-Forwarded-Proto $scheme;
+ }
+ }
+ ```
+
+- [ ] **7.3** Activer le site et recharger NGINX
+
+ ```bash
+ ssh root@51.159.7.190 "ln -s /etc/nginx/sites-available/app.geosector.fr /etc/nginx/sites-enabled/"
+ ssh root@51.159.7.190 "nginx -t"
+ ssh root@51.159.7.190 "systemctl reload nginx"
+ ```
+
+- [ ] **7.4** Vérifier le certificat SSL
+ ```bash
+ ssh root@51.159.7.190 "certbot certificates | grep app3.geosector.fr"
+ # Si pas de certificat, le créer :
+ # certbot certonly --nginx -d app3.geosector.fr
+ ```
+
+##### Phase 8️⃣ : Tests de l'API PRODUCTION
+
+- [ ] **8.1** Tester le endpoint health
+
+ ```bash
+ curl -X GET https://app3.geosector.fr/api/health
+ # ✅ Doit retourner {"status":"ok"}
+ ```
+
+- [ ] **8.2** Tester l'authentification
+
+ ```bash
+ curl -X POST https://app3.geosector.fr/api/login \
+ -H "Content-Type: application/json" \
+ -d '{"username":"test_user","password":"test_pass"}'
+ # ✅ Login doit fonctionner
+ ```
+
+- [ ] **8.3** Tester la récupération de données
+
+ ```bash
+ curl -X GET https://app3.geosector.fr/api/users \
+ -H "Authorization: Bearer {session_id}"
+ # ✅ Données récupérées depuis maria4/pra_geo
+ ```
+
+- [ ] **8.4** Tester depuis l'application Flutter
+
+ ```bash
+ # Configurer l'app Flutter avec l'URL de production
+ # Tester : Login, récupération opérations, création passage
+ # ✅ Toutes les fonctionnalités opérationnelles
+ ```
+
+- [ ] **8.5** Vérifier les logs
+ ```bash
+ ssh root@51.159.7.190 "incus exec pra-geo -- tail -50 /var/www/geosector/api/logs/app.log"
+ ssh root@51.159.7.190 "incus exec pra-geo -- tail -50 /var/log/nginx/pra-api-error.log"
+ # ✅ Aucune erreur critique
+ ```
+
+##### Phase 9️⃣ : Sécurisation et monitoring
+
+- [ ] **9.1** Configurer les sauvegardes automatiques de maria4
+
+ ```bash
+ # Créer un script de backup quotidien
+ ssh root@51.159.7.190 "cat > /root/backup-maria4.sh << 'EOF'
+ #!/bin/bash
+ DATE=$(date +%Y%m%d_%H%M%S)
+ incus exec maria4 -- mysqldump -u root -p'MyAlpLocal,90b' --all-databases > /var/back/maria4_backup_${DATE}.sql
+ # Garder les 30 derniers jours
+ find /var/back/maria4_backup_*.sql -mtime +30 -delete
+ EOF"
+ ssh root@51.159.7.190 "chmod +x /root/backup-maria4.sh"
+ ```
+
+- [ ] **9.2** Configurer la tâche CRON pour les backups
+
+ ```bash
+ ssh root@51.159.7.190 "crontab -l | { cat; echo '0 2 * * * /root/backup-maria4.sh'; } | crontab -"
+ ```
+
+- [ ] **9.3** Configurer les logs de rotation
+
+ ```bash
+ # Vérifier que logrotate est configuré pour NGINX et PHP
+ ssh root@51.159.7.190 "cat /etc/logrotate.d/nginx"
+ ```
+
+- [ ] **9.4** Configurer le monitoring (optionnel)
+
+ ```bash
+ # Installer un monitoring basique (htop, iotop, etc.)
+ ssh root@51.159.7.190 "apt install htop iotop iftop -y"
+ ```
+
+- [ ] **9.5** Documenter les IPs et accès
+
+ ```bash
+ # Créer un fichier README sur IN4
+ ssh root@51.159.7.190 "cat > /root/PRODUCTION_INFO.txt << 'EOF'
+ ========================================
+ ENVIRONNEMENT PRODUCTION GEOSECTOR
+ ========================================
+
+ Serveur : IN4 (51.159.7.190)
+ Date de création : $(date +%Y-%m-%d)
+
+ Containers :
+ - pra-geo (API) : 13.23.34.43
+ - maria4 (DB) : 13.23.34.4
+
+ Base de données :
+ - Base : pra_geo
+ - User : pra_geo_user
+ - Password : d2jAAGGWi8fxFrWgXjOA
+
+ URLs :
+ - API : https://app3.geosector.fr/api/
+ - Flutter : https://app3.geosector.fr/
+
+ Backups :
+ - Quotidien à 2h : /var/back/maria4_backup_*.sql
+ - Rétention : 30 jours
+ ========================================
+ EOF"
+ ```
+
+##### Phase 🔟 : Documentation finale
+
+- [ ] **10.1** Mettre à jour TECHBOOK.md
+
+ - Ajouter la configuration PRODUCTION dans le tableau (ligne 142)
+ - Documenter l'architecture complète 3 environnements
+
+- [ ] **10.2** Mettre à jour ce TODO-API.md
+
+ - Marquer toutes les étapes comme complétées
+ - Statut : ✅ TERMINÉ
+
+- [ ] **10.3** Créer un document de procédures d'urgence
+ - Rollback en cas de problème
+ - Contacts et escalade
+ - Procédure de restauration depuis backup
+
+---
+
+**Date de création TODO :** 07/10/2025
+**Date planifiée d'exécution :** Jeudi 10/10/2025 à 16h
+**Date de début :** 07/10/2025
+**Statut :** 🟡 EN COURS - Phases 0 à 5 complétées, en attente validation client
+**Durée estimée :** 3-4 heures
+
+---
+
+#### 🔐 Informations sensibles (à ne pas commiter)
+
+**Container maria3 (IN3)** :
+
+- IP interne : 13.23.33.4
+- Root password : `MyAlpLocal,90b`
+- Port : 3306
+
+**Utilisateurs bases de données** :
+
+- DEV : `dva_geo_user` / `CBq9tKHj6PGPZuTmAHV7`
+- REC : `rca_geo_user` / `UPf3C0cQ805LypyM71iW` (à créer)
+- PROD : `pra_geo_user` / mot de passe à générer
+
+---
+
+**Date de création :** 07/10/2025
+**Statut :** ✅ TERMINÉ - Migration DVA-GEO complétée avec succès
+**Dernière mise à jour :** 07/10/2025
+
+---
+
+### 🟢 Migration de données PM7 → PRA-GEO (PRODUCTION)
+
+#### 3. Script de migration depuis backup PM7 restauré
+
+**Demandé le :** 07/10/2025
+**Objectif :** Créer un script PHP standalone pour migrer les données depuis un backup PM7 (restauré dans maria4) vers la base pra_geo.
+
+**Contexte :**
+
+- Backup quotidien de PM7 (11.1.2.17) : `geosector_YYYYMMDD.sql.tar.gz.enc`
+- Processus : Déchiffrement → SCP vers IN4 → Import dans maria4 → Migration vers pra_geo
+- Script exécuté depuis le container **pra-geo** (pas maria4, car PHP nécessaire)
+
+**Architecture de migration :**
+
+```
+PM7 (11.1.2.17) - Backup nocturne chiffré
+ ↓ Déchiffrement (decpm7.sh)
+ ↓ SCP vers IN4:/var/back/
+ ↓ incus file push vers maria4:/var/back/
+ ↓ Décompression et import
+maria4 (IN4) - Base geosector_YYYYMMDD
+ ↓ Migration (script PHP depuis pra-geo)
+maria4 (IN4) - Base pra_geo
+```
+
+**Fichier créé :** `scripts/php/migrate_from_backup.php`
+
+**Fonctionnalités implémentées :**
+
+✅ **1. Respect des contraintes FK**
+
+- Ordre de migration : x\_\* → entites → users → operations → ope_sectors → ope_pass → medias
+- Gestion des dépendances : x_devises → x_pays → x_regions → x_departements → x_villes
+
+✅ **2. Chiffrement AES-256-CBC**
+
+- Utilisation de `ApiService::encryptData()` et `ApiService::encryptSearchableData()`
+- Tables concernées : entites (name, phone, email, iban, bic), users (name, email, phone), ope_pass (name, email, phone)
+
+✅ **3. Mappings de champs**
+
+- `rowid` → `id`
+- `active` → `chk_active`
+- `date_creat` → `created_at`
+- `date_modif` → `updated_at`
+- `libelle` → `encrypted_name` (entites)
+- `nom` → `encrypted_name` (users, passages)
+
+✅ **4. Détection mobile/fixe**
+
+- Numéros commençant par 06/07 → champ `encrypted_mobile`
+- Autres numéros → champ `encrypted_phone`
+
+✅ **5. Support deux modes**
+
+- **Global** : Migration de toutes les amicales
+- **Entity** : Migration d'une amicale spécifique (avec tables de référence)
+
+✅ **6. Optimisations**
+
+- Traitement par lots de 1000 pour `ope_pass` (table volumineuse)
+- `ON DUPLICATE KEY UPDATE` pour idempotence
+- Logging détaillé avec progression
+
+**Utilisation :**
+
+```bash
+# Sur IN4, dans le container pra-geo
+cd /var/www/geosector/api
+
+# Migration globale (toutes les amicales)
+php scripts/php/migrate_from_backup.php \
+ --source-db=geosector_20251007 \
+ --target-db=pra_geo \
+ --mode=global
+
+# Migration d'une amicale spécifique
+php scripts/php/migrate_from_backup.php \
+ --source-db=geosector_20251007 \
+ --target-db=pra_geo \
+ --mode=entity \
+ --entity-id=45 \
+ --log=/var/www/geosector/api/logs/migration_entity_45.log
+```
+
+**Tables migrées (dans l'ordre) :**
+
+1. `x_devises`, `x_entites_types`, `x_types_passages`, `x_types_reglements`, `x_users_roles`, `x_users_titres`
+2. `x_pays`, `x_regions`, `x_departements`, `x_villes`
+3. `entites` (avec chiffrement)
+4. `users` (avec chiffrement et détection mobile)
+5. `operations`
+6. `ope_sectors`
+7. `sectors_adresses`
+8. `ope_users`
+9. `ope_users_sectors`
+10. `ope_pass` (avec chiffrement, traitement par lots)
+11. `ope_pass_histo`
+12. `medias`
+
+**Configuration connexion :**
+
+- Host : `13.23.33.4` (maria4 sur IN4)
+- Base source : `geosector_YYYYMMDD` (user root)
+- Base cible : `pra_geo` (user pra_geo_user)
+
+**Documentation :** `scripts/README-migration.md`
+
+**Statut :** ✅ TERMINÉ - Script standalone créé et prêt à être testé
+**Date de réalisation :** 07/10/2025
+
+---
+
+#### 🔧 Correction critique du mapping des secteurs
+
+**Date :** 08/10/2025
+**Problème identifié :** Bug majeur dans la migration des secteurs
+
+**Contexte du bug :**
+Le script initial lisait depuis la table `ope_sectors` dans la base SOURCE, alors que cette table n'existe QUE dans la base CIBLE. Dans la base source (geosector), les secteurs sont stockés dans la table `sectors`.
+
+**Architecture des secteurs :**
+
+```
+SOURCE (geosector_YYYYMMDD):
+ - sectors (rowid, libelle, sector, color)
+ - ope_users_sectors (fk_operation, fk_user, fk_sector → sectors.rowid)
+ - ope_pass (fk_sector → sectors.rowid)
+ - sectors_adresses (fk_sector → sectors.rowid)
+
+CIBLE (pra_geo):
+ - ope_sectors (id AUTO_INCREMENT, fk_operation, fk_old_sector, libelle, sector, color)
+ - ope_users_sectors (fk_sector → ope_sectors.id NOUVEAU)
+ - ope_pass (fk_sector → ope_sectors.id NOUVEAU)
+ - sectors_adresses (fk_sector → ope_sectors.id NOUVEAU)
+```
+
+**Corrections apportées :**
+
+✅ **1. Migration ope_sectors (lignes 650-772)**
+
+- Lecture depuis `sectors` (source) via JOIN avec `ope_users_sectors`
+- Création dans `ope_sectors` (cible) avec ID auto-increment
+- Génération d'un mapping : `fk_operation . '_' . old_sector_id → new_sector_id`
+- Stockage dans `$this->sectorMapping` pour utilisation par les tables suivantes
+
+✅ **2. Migration sectors_adresses (lignes 774-881)**
+
+- Utilise le mapping pour remplacer `fk_sector` ancien par nouveau ID
+- Ignore les adresses dont le secteur n'a pas été migré (compteur `$skipped`)
+
+✅ **3. Migration ope_users_sectors (lignes 952-1040)**
+
+- Utilise le mapping pour remplacer `fk_sector` ancien par nouveau ID
+- Ignore les associations dont le secteur n'a pas été migré
+
+✅ **4. Migration ope_pass (lignes 1042-1200)**
+
+- Ajout du mapping au début de la boucle (lignes 1120-1131)
+- Utilise le mapping pour remplacer `fk_sector` ancien par nouveau ID
+- Ignore les passages dont le secteur n'a pas été migré
+
+**Exemple de mapping créé :**
+
+```php
+$this->sectorMapping = [
+ '12345_789' => 1001, // opération 12345, ancien secteur 789 → nouveau secteur 1001
+ '12345_790' => 1002,
+ '12346_789' => 1003, // Même ancien secteur, mais autre opération = autre nouveau ID
+];
+```
+
+**Impact :** CRITIQUE - Sans cette correction, la migration aurait échoué complètement
+
+**Fichiers modifiés :**
+
+- `scripts/php/migrate_from_backup.php` (corrections lignes 650-1200)
+
+**Statut :** ✅ CORRIGÉ - Prêt pour tests
+**Date de correction :** 08/10/2025
+
+---
+
+#### 📋 Préparation migration en batch (406 entités éligibles)
+
+**Date :** 08/10/2025
+**Objectif :** Migrer progressivement les entités actives depuis geosector_20251008
+
+**Critères de sélection des entités :**
+
+- Plus de 4 utilisateurs actifs
+- Au moins 1 opération créée après le 2023-08-01
+- Code postal renseigné (nettoyage de 3 entités de test)
+
+**Fichiers créés :**
+
+1. **`scripts/migrations_entites.json`** (406 entités)
+
+ - Liste complète des entités éligibles avec statistiques
+ - Format : entity_id, code_postal, nom, ville, nb_users, nb_operations, nb_passages
+
+2. **`scripts/migrate_batch.sh`** (script bash orchestrateur)
+ - Migration entité par entité avec progression
+ - Options : `--start N`, `--limit N`, `--dry-run`, `--continue`
+ - Logs détaillés par entité + log global
+ - Gestion d'erreurs et possibilité de reprise
+ - Résumé final avec durée, succès, erreurs
+
+**Utilisation du script batch :**
+
+```bash
+# Test avec 1 entité
+./migrate_batch.sh --start 1 --limit 1
+
+# Migration de 50 entités
+./migrate_batch.sh --start 1 --limit 50
+
+# Migration complète (406 entités) avec continuité sur erreur
+./migrate_batch.sh --continue
+
+# Mode simulation (dry-run)
+./migrate_batch.sh --dry-run
+```
+
+**Statistiques des 406 entités :**
+
+- Total utilisateurs : ~15 000 - 20 000 (estimation)
+- Total passages : ~500 000 - 1 000 000 (estimation)
+- Durée estimée migration complète : 8-12 heures
+
+**Statut :** ✅ PRÉPARÉ - Scripts prêts, tests à effectuer
+**Date de préparation :** 08/10/2025
+
+---
+
+#### 🔧 Correction critique des doublons ope_users et ope_users_sectors
+
+**Date :** 10/10/2025
+**Problème identifié :** Doublons massifs dans les tables `ope_users` et `ope_users_sectors` lors de la migration
+
+**Diagnostic :**
+
+- Table `ope_users` : 186+ doublons pour la même paire (fk_operation, fk_user)
+- Table `ope_users_sectors` : Risque de doublons sur (fk_operation, fk_user, fk_sector)
+- Cause : Absence de contraintes UNIQUE sur ces combinaisons de FK
+- Impact : Le `ON DUPLICATE KEY UPDATE` ne fonctionnait pas, permettant la création de lignes en double
+
+**Corrections appliquées :**
+
+1. **Mise à jour de la structure de référence** (`geo_app_structure.sql`) :
+
+ - Ligne 403 : Ajout de `UNIQUE KEY idx_operation_user (fk_operation, fk_user)` sur `ope_users`
+ - Ligne 430 : Ajout de `UNIQUE KEY idx_operation_user_sector (fk_operation, fk_user, fk_sector)` sur `ope_users_sectors`
+
+2. **Correction du code PHP** (`migrate_from_backup.php`) :
+
+ ```php
+ // AVANT (lignes 1125-1140) - INCORRECT
+ $sql = "SELECT ou.rowid, ou.fk_operation, ou.fk_user, ...
+ INSERT INTO ope_users (id, fk_operation, fk_user, ...) VALUES (:id, ...
+
+ // APRÈS (lignes 1125-1191) - CORRECT
+ $sql = "SELECT DISTINCT ou.fk_operation, ou.fk_user, ... // Suppression de rowid
+ INSERT INTO ope_users (fk_operation, fk_user, ...) VALUES (... // Suppression de id, utilise auto-increment
+ ```
+
+3. **Requêtes SQL pour appliquer les contraintes** :
+
+ ```sql
+ -- Vérifier les doublons existants
+ SELECT fk_operation, fk_user, COUNT(*) as count
+ FROM ope_users GROUP BY fk_operation, fk_user HAVING count > 1;
+
+ -- Ajouter les contraintes UNIQUE
+ ALTER TABLE ope_users
+ ADD UNIQUE KEY `idx_operation_user` (`fk_operation`, `fk_user`);
+
+ ALTER TABLE ope_users_sectors
+ ADD UNIQUE KEY `idx_operation_user_sector` (`fk_operation`, `fk_user`, `fk_sector`);
+
+ -- Vérifier
+ SHOW INDEX FROM ope_users WHERE Key_name = 'idx_operation_user';
+ SHOW INDEX FROM ope_users_sectors WHERE Key_name = 'idx_operation_user_sector';
+ ```
+
+**À appliquer sur tous les environnements :**
+
+- DEV (dva_geo sur maria3/IN3) : mysql -h 13.23.33.4 -u dva_geo_user
+- REC (rca_geo sur maria3/IN3) : mysql -h 13.23.33.4 -u rca_geo_user
+- PROD (pra_geo sur maria4/IN4) : mysql -h 13.23.33.4 -u pra_geo_user
+
+**Procédure de re-migration après correction :**
+
+```bash
+# Supprimer les données de l'entité avant re-migration
+php scripts/php/migrate_from_backup.php \
+ --source-db=geosector_20251008 \
+ --mode=entity \
+ --entity-id=5 \
+ --delete-before
+```
+
+**Fichiers modifiés :**
+
+- `scripts/php/geo_app_structure.sql` (lignes 403, 430)
+- `scripts/php/migrate_from_backup.php` (lignes 1125-1191)
+- `scripts/README-migration.md` (section Corrections Critiques ajoutée)
+- `scripts/CORRECTIONS_MIGRATE.md` (correction #15 documentée)
+
+**Statut :** ✅ CORRIGÉ - Code et structure mis à jour, contraintes à appliquer sur bases
+**Date de correction :** 10/10/2025
+
+---
+
+#### 🎯 Planning de déploiement (09-10/10/2025)
+
+**Jeudi 09/10/2025 - Environnement DEV (dva-geo)**
+
+- [ ] Vider la base `dva_geo` (données de test)
+- [ ] Déployer les scripts corrigés (`deploy-api.sh`)
+- [ ] Test migration 1 entité pilote (#1178 - 5 users, 1317 passages)
+- [ ] Validation des mappings secteurs
+- [ ] Si OK : Migration batch des 406 entités
+- [ ] Vérification intégrité données (comptages, FK)
+- [ ] Tests fonctionnels API + Flutter
+
+**Jeudi 09/10/2025 PM - Environnement REC (rca-geo)**
+
+- [ ] Dump `dva_geo` → Import dans `rca_geo`
+- [ ] Tests de non-régression complets
+- [ ] Validation client sur REC
+- [ ] Tests de charge (si applicable)
+
+**Vendredi 10/10/2025 - Environnement PROD (pra-geo)**
+
+- [ ] Sauvegarde complète `pra_geo` (avant migration)
+- [ ] Dump `rca_geo` → Import dans `pra_geo`
+- [ ] Tests de smoke en production
+- [ ] Monitoring des performances
+- [ ] Documentation post-migration
+
+**Rollback plan :**
+
+- Restauration depuis sauvegarde pré-migration : 15-30 minutes
+- Scripts de vérification disponibles dans `scripts/php/verify_migration_structure.php`
+
+**Statut :** 📅 PLANIFIÉ - Démarrage jeudi 09/10/2025 matin
+**Durée estimée totale :** 2 jours
+
+---
+
### 🟡 PRIORITÉ MOYENNE
-#### 2. Amélioration des logs
+#### 4. Amélioration des logs
+
- Ajouter plus de contexte dans les logs
- Rotation automatique des logs
- Dashboard de monitoring
-#### 3. Optimisation des performances
+#### 5. Optimisation des performances
+
- Cache des requêtes fréquentes
- Index sur les tables volumineuses
- Pagination optimisée
-#### 4. Sécurisation des clés Stripe par environnement
-**Objectif :** Étudier une approche plus sécurisée pour stocker les clés Stripe
+#### 6. Sécurisation des clés Stripe par environnement
-**Problème actuel :**
-- Toutes les clés (DEV, REC, PROD) sont dans un seul fichier `AppConfig.php`
-- Les clés PRODUCTION sont visibles dans le code DEV/REC
-- Risque si accès au container DEV → exposition des clés PROD
+**Statut :** ✅ **PARTIELLEMENT RÉSOLU** (16/10/2025)
+
+**Ce qui a été fait :**
+
+- ✅ Configuration complète des clés API Stripe par environnement dans `AppConfig.php`
+ - DEV : Clés TEST Pierre (compte plateforme développeur)
+ - RECETTE : Clés TEST Client (compte plateforme client mode test)
+ - PRODUCTION : Clés LIVE Client (compte plateforme client mode live)
+- ✅ Configuration des webhooks Stripe avec secrets dédiés
+ - RECETTE : `webhook-rca` → `whsec_avExshr0MeWTI7wXP8478XVUkrbYG8hs`
+ - PRODUCTION : `webhook-pra` → `whsec_gFnA6pR92RLdbAS2T6CSC18xsSdNBZHR`
+- ✅ Mise à jour de la version API Stripe : `2025-08-27.basil`
+- ✅ Correction du bug StripeWebhookController (ligne 46)
+- ✅ Correction de l'URL webhook : `/api/stripe/webhooks` (avec 's')
+- ✅ Documentation complète dans `TECHBOOK.md`
+
+**Ce qui reste (amélioration future) :**
+
+- ⏳ Migration vers variables d'environnement pour sécurité renforcée
+- ⏳ Isolation des secrets par container (actuellement tous visibles dans AppConfig.php)
+
+**Solutions à étudier pour amélioration future :**
-**Solutions à étudier :**
1. **Variables d'environnement** (`.env` par container)
+
- Fichier `.env.dev`, `.env.rec`, `.env.prod`
- Chargement dynamique selon l'environnement
- Exclusion des `.env` du versionning Git
2. **Fichiers de config séparés**
+
- `config/stripe.dev.php`, `config/stripe.rec.php`, `config/stripe.prod.php`
- Déploiement sélectif selon l'environnement
- Non versionnés (ajoutés au .gitignore)
@@ -119,12 +1227,1463 @@ INSERT INTO ope_pass_backup (
### 🟢 PRIORITÉ BASSE
-#### 5. Documentation API
+#### 7. Amélioration de la suppression des utilisateurs
+
+**Demandé le :** 19/10/2025
+**Objectif :** Empêcher la suppression d'un utilisateur ayant des passages et gérer correctement le soft delete.
+
+**Problème actuel :**
+
+- Erreur SQL lors de `DELETE /api/users/{id}` si l'utilisateur a des passages dans `ope_pass`
+- Contrainte FK `ope_pass_ibfk_3` empêche la suppression (comportement attendu)
+- Génère des emails d'alerte "GEOSECTOR SECURITY" de type SQL_ERROR
+
+**Solution à implémenter (côté Flutter) :**
+
+1. **Vérification avant suppression**
+
+ - Endpoint suggéré : `GET /api/users/{id}/can-delete`
+ - Retourne : total passages, passages opérations actives, passages opérations inactives
+
+2. **Logique métier Flutter**
+
+ - Si passages sur **opération active** → Demander réassignation à un autre user
+ - Si passages uniquement sur **opérations inactives** → Proposer soft delete
+ - Si aucun passage → Autoriser suppression physique
+
+3. **Soft delete (recommandé)**
+ - Ajouter `deleted_at TIMESTAMP NULL` dans table `users`
+ - Modifier `UserController::deleteUser()` pour faire un soft delete
+ - Les utilisateurs soft-deleted ne sont plus affichés mais conservés pour l'historique
+
+**Côté API (préparation) :**
+
+- [ ] Ajouter endpoint `GET /api/users/{id}/can-delete` pour vérification
+- [ ] Ajouter colonne `deleted_at` dans table `users`
+- [ ] Modifier `UserController::deleteUser()` pour gérer le soft delete
+- [ ] Adapter les requêtes SQL pour filtrer `WHERE deleted_at IS NULL`
+
+**Côté Flutter (à implémenter par le développeur) :**
+
+- [ ] Appeler endpoint de vérification avant suppression
+- [ ] Afficher dialogue selon le contexte (réassignation vs soft delete)
+- [ ] Gérer le workflow de réassignation si nécessaire
+
+**Note :** En attendant l'implémentation, les emails SQL_ERROR pour `DELETE /api/users/*` sont normaux et attendus (violation de contrainte FK légitime).
+
+**Statut :** 📝 NOTÉ - À implémenter côté Flutter en priorité
+**Date :** 19/10/2025
+
+---
+
+### 🔴 PRIORITÉ HAUTE
+
+#### 8. Connexion de l'API à un broker Mosquitto MQTT
+
+**Demandé le :** 08/11/2025
+**Objectif :** Intégrer un système de communication MQTT permettant à l'API de publier et recevoir des messages en temps réel pour notifier les clients (applications Flutter) des changements de données.
+
+**Contexte :**
+- Les applications Flutter doivent être notifiées en temps réel lors de modifications de données (passages, secteurs, opérations)
+- Alternative moderne aux webhooks pour la communication bidirectionnelle
+- Permet de réduire le polling constant des applications mobiles
+- Architecture publish/subscribe pour une meilleure scalabilité
+
+**Cas d'usage principaux :**
+```
+Création/Modification passage
+ ↓ API publie sur MQTT: geosector/{entity_id}/passages/{operation_id}
+ ↓ Payload JSON avec les données du passage
+ ↓ Applications Flutter abonnées reçoivent la notification
+ ↓ Mise à jour automatique de l'UI sans refresh manuel
+
+Modification secteur
+ ↓ API publie sur MQTT: geosector/{entity_id}/sectors/{operation_id}
+ ↓ Applications concernées rechargent les données
+
+Changement statut opération
+ ↓ API publie sur MQTT: geosector/{entity_id}/operations/{operation_id}
+ ↓ Notification push vers tous les membres
+```
+
+---
+
+#### 📋 Plan d'action détaillé
+
+##### Étape 1 : Installation et configuration Mosquitto
+
+**A. Installation du broker Mosquitto** (sur serveur DVA/RCA/PROD) :
+
+```bash
+# Sur le host ou dans un container dédié
+apk add mosquitto mosquitto-clients # Alpine
+# ou
+apt install mosquitto mosquitto-clients # Debian/Ubuntu
+
+# Démarrer et activer au boot
+rc-service mosquitto start # Alpine
+rc-update add mosquitto default
+# ou
+systemctl start mosquitto # Systemd
+systemctl enable mosquitto
+```
+
+**B. Configuration Mosquitto** (`/etc/mosquitto/mosquitto.conf`) :
+
+```conf
+# Listeners
+listener 1883 0.0.0.0
+protocol mqtt
+
+listener 8883 0.0.0.0
+protocol mqtt
+certfile /etc/mosquitto/certs/cert.pem
+keyfile /etc/mosquitto/certs/key.pem
+cafile /etc/mosquitto/certs/ca.pem
+
+# WebSockets (pour Flutter Web)
+listener 9001
+protocol websockets
+
+# Authentification
+allow_anonymous false
+password_file /etc/mosquitto/passwd
+
+# ACL (Access Control List)
+acl_file /etc/mosquitto/acl.conf
+
+# Persistence
+persistence true
+persistence_location /var/lib/mosquitto/
+
+# Logs
+log_dest file /var/log/mosquitto/mosquitto.log
+log_type all
+log_timestamp true
+```
+
+**C. Créer les utilisateurs MQTT** :
+
+```bash
+# Utilisateur pour l'API
+mosquitto_passwd -c /etc/mosquitto/passwd geosector_api
+# Password: généré de manière sécurisée
+
+# Utilisateur pour les applications Flutter
+mosquitto_passwd /etc/mosquitto/passwd geosector_client
+```
+
+**D. Configuration ACL** (`/etc/mosquitto/acl.conf`) :
+
+```conf
+# API peut publier et s'abonner à tous les topics geosector
+user geosector_api
+topic readwrite geosector/#
+
+# Clients peuvent seulement s'abonner
+user geosector_client
+topic read geosector/#
+```
+
+**E. Tests de connexion** :
+
+```bash
+# Test publication
+mosquitto_pub -h localhost -p 1883 -u geosector_api -P password \
+ -t "geosector/test" -m "Test message"
+
+# Test souscription
+mosquitto_sub -h localhost -p 1883 -u geosector_client -P password \
+ -t "geosector/#"
+```
+
+**Fichiers impactés :**
+- [ ] Installer Mosquitto sur DVA (13.23.33.43)
+- [ ] Installer Mosquitto sur RCA (13.23.33.23)
+- [ ] Installer Mosquitto sur PROD (13.23.33.22 / IN4)
+- [ ] Configurer firewall (port 1883, 8883, 9001)
+
+---
+
+##### Étape 2 : Installation bibliothèque PHP MQTT
+
+```bash
+# Via Composer
+composer require php-mqtt/client
+```
+
+**Fichiers impactés :**
+- [ ] Modifier `composer.json`
+- [ ] Exécuter sur DVA, RCA, PROD
+
+---
+
+##### Étape 3 : Configuration AppConfig.php
+
+Ajouter la configuration MQTT :
+
+```php
+// Dans src/Config/AppConfig.php
+'mqtt' => [
+ 'enabled' => true,
+ 'broker' => [
+ 'host' => 'localhost', // ou IP du container mosquitto
+ 'port' => 1883,
+ 'username' => 'geosector_api',
+ 'password' => '', // À définir par environnement
+ 'client_id' => 'geosector_api_' . gethostname(),
+ ],
+ 'tls' => [
+ 'enabled' => false, // true en production
+ 'port' => 8883,
+ 'ca_file' => '/etc/mosquitto/certs/ca.pem',
+ ],
+ 'topics' => [
+ 'passages' => 'geosector/{entity_id}/passages/{operation_id}',
+ 'sectors' => 'geosector/{entity_id}/sectors/{operation_id}',
+ 'operations' => 'geosector/{entity_id}/operations/{operation_id}',
+ 'users' => 'geosector/{entity_id}/users',
+ 'system' => 'geosector/system',
+ ],
+ 'qos' => 1, // Quality of Service (0, 1, ou 2)
+ 'retain' => false,
+],
+```
+
+**Fichiers impactés :**
+- [ ] Modifier `src/Config/AppConfig.php` (3 environnements)
+
+---
+
+##### Étape 4 : Créer le service MqttService
+
+Créer `src/Services/MqttService.php` :
+
+**Méthodes principales :**
+
+```php
+namespace App\Services;
+
+use PhpMqtt\Client\MqttClient;
+use PhpMqtt\Client\ConnectionSettings;
+use App\Services\LogService;
+use Exception;
+
+class MqttService {
+ private static ?self $instance = null;
+ private ?MqttClient $client = null;
+ private array $config;
+ private bool $connected = false;
+
+ private function __construct() {
+ $this->config = AppConfig::getInstance()->get('mqtt');
+ }
+
+ public static function getInstance(): self {
+ if (self::$instance === null) {
+ self::$instance = new self();
+ }
+ return self::$instance;
+ }
+
+ /**
+ * Connecte au broker MQTT
+ */
+ public function connect(): bool
+
+ /**
+ * Déconnecte du broker MQTT
+ */
+ public function disconnect(): void
+
+ /**
+ * Publie un message sur un topic
+ * @param string $topic Topic MQTT
+ * @param array $payload Données à publier (sera converti en JSON)
+ * @param int $qos Quality of Service (0, 1, 2)
+ * @param bool $retain Retenir le message
+ */
+ public function publish(string $topic, array $payload, int $qos = 1, bool $retain = false): bool
+
+ /**
+ * Publie une notification de passage créé/modifié
+ */
+ public function publishPassageUpdate(int $entityId, int $operationId, array $passageData): bool
+
+ /**
+ * Publie une notification de secteur créé/modifié
+ */
+ public function publishSectorUpdate(int $entityId, int $operationId, array $sectorData): bool
+
+ /**
+ * Publie une notification d'opération créée/modifiée
+ */
+ public function publishOperationUpdate(int $entityId, int $operationId, array $operationData): bool
+
+ /**
+ * Publie un message système (maintenance, déconnexion forcée, etc.)
+ */
+ public function publishSystemMessage(string $message, array $metadata = []): bool
+
+ /**
+ * Génère le topic en remplaçant les placeholders
+ */
+ private function buildTopic(string $topicTemplate, array $params): string
+
+ /**
+ * Vérifie si le service est activé et connecté
+ */
+ public function isAvailable(): bool
+}
+```
+
+**Gestion des erreurs :**
+- Reconnexion automatique en cas de déconnexion
+- Timeout configurable
+- Logs détaillés des publications
+- Mode dégradé si MQTT indisponible (ne pas bloquer l'API)
+
+**Fichiers impactés :**
+- [ ] Créer `src/Services/MqttService.php`
+
+---
+
+##### Étape 5 : Intégrer MQTT dans les Controllers
+
+**A. PassageController** - Notifier création/modification passage :
+
+```php
+// Dans PassageController::create() après ligne 602
+if ($passageId) {
+ // Publier notification MQTT (non bloquant)
+ try {
+ $mqttService = MqttService::getInstance();
+ if ($mqttService->isAvailable()) {
+ $mqttService->publishPassageUpdate(
+ $entityId,
+ $operationId,
+ [
+ 'passage_id' => $passageId,
+ 'fk_type' => $fkType,
+ 'montant' => $montant,
+ 'action' => 'created',
+ 'timestamp' => time()
+ ]
+ );
+ }
+ } catch (Exception $e) {
+ // Logger mais ne pas bloquer l'API
+ LogService::log('Erreur publication MQTT passage', [
+ 'level' => 'warning',
+ 'error' => $e->getMessage()
+ ]);
+ }
+}
+```
+
+**B. SectorController** - Notifier création/modification secteur :
+
+```php
+// Dans SectorController::create() après ligne 531
+$mqttService = MqttService::getInstance();
+if ($mqttService->isAvailable()) {
+ $mqttService->publishSectorUpdate(
+ $entityId,
+ $operationId,
+ [
+ 'sector_id' => $sectorId,
+ 'libelle' => $data['libelle'],
+ 'passages_created' => $passagesCreated,
+ 'action' => 'created',
+ 'timestamp' => time()
+ ]
+ );
+}
+```
+
+**C. OperationController** - Notifier changements opération :
+
+```php
+// Notifications pour :
+// - Changement de statut (chk_active)
+// - Ajout/retrait d'utilisateurs
+// - Modification des dates
+```
+
+**Fichiers impactés :**
+- [ ] Modifier `src/Controllers/PassageController.php`
+- [ ] Modifier `src/Controllers/SectorController.php`
+- [ ] Modifier `src/Controllers/OperationController.php`
+- [ ] Modifier `src/Controllers/UserController.php` (optionnel)
+
+---
+
+##### Étape 6 : Structure des messages MQTT
+
+**Format JSON standardisé :**
+
+```json
+{
+ "event": "passage.created",
+ "entity_id": 5,
+ "operation_id": 12345,
+ "timestamp": 1699456789,
+ "data": {
+ "passage_id": 19500576,
+ "fk_type": 1,
+ "montant": 10.00,
+ "encrypted_name": "...",
+ "rue": "Rue Example",
+ "ville": "Rennes"
+ },
+ "metadata": {
+ "user_id": 9999985,
+ "api_version": "3.3.5"
+ }
+}
+```
+
+**Types d'événements :**
+- `passage.created`
+- `passage.updated`
+- `passage.deleted`
+- `sector.created`
+- `sector.updated`
+- `sector.deleted`
+- `operation.created`
+- `operation.updated`
+- `operation.status_changed`
+- `user.added`
+- `user.removed`
+- `system.maintenance`
+
+---
+
+##### Étape 7 : Sécurisation
+
+**A. Topics hiérarchiques par entité :**
+```
+geosector/{entity_id}/* → Seuls les membres de cette entité peuvent s'abonner
+```
+
+**B. Validation côté API :**
+- Vérifier que l'utilisateur appartient à l'entité avant publication
+- Ne jamais publier de données sensibles non chiffrées
+
+**C. Authentification JWT pour Flutter :**
+```json
+{
+ "mqtt_token": "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9...",
+ "mqtt_host": "mqtt.geosector.fr",
+ "mqtt_port": 8883,
+ "topics": ["geosector/5/#"]
+}
+```
+
+**Fichiers impactés :**
+- [ ] Créer endpoint `POST /api/mqtt/token` (génération token MQTT)
+- [ ] Modifier ACL Mosquitto pour utiliser JWT
+
+---
+
+##### Étape 8 : Tests et validation
+
+**Tests sur DVA :**
+- [ ] Publier message test via API
+- [ ] S'abonner avec `mosquitto_sub` et vérifier réception
+- [ ] Créer passage et vérifier publication MQTT
+- [ ] Créer secteur et vérifier publication MQTT
+- [ ] Vérifier logs MQTT dans `/var/log/mosquitto/`
+- [ ] Tester reconnexion après redémarrage broker
+
+**Tests de charge :**
+- [ ] 100 publications simultanées
+- [ ] 50 clients connectés en parallèle
+- [ ] Vérifier latence (doit être < 100ms)
+
+**Tests de non-régression :**
+- [ ] API fonctionne si MQTT désactivé
+- [ ] API fonctionne si broker MQTT inaccessible (mode dégradé)
+
+**Fichiers impactés :**
+- [ ] Créer `scripts/test/test_mqtt_integration.php`
+
+---
+
+##### Étape 9 : Intégration côté Flutter (information)
+
+**Package Flutter à utiliser :**
+```yaml
+dependencies:
+ mqtt_client: ^10.0.0
+```
+
+**Exemple de souscription :**
+```dart
+final client = MqttServerClient('mqtt.geosector.fr', '');
+await client.connect(username, password);
+
+client.subscribe('geosector/$entityId/passages/#', MqttQos.atLeastOnce);
+
+client.updates!.listen((List> messages) {
+ final recMess = messages[0].payload as MqttPublishMessage;
+ final payload = MqttPublishPayload.bytesToStringAsString(recMess.payload.message);
+ final data = jsonDecode(payload);
+
+ // Mettre à jour l'UI en temps réel
+ if (data['event'] == 'passage.created') {
+ _refreshPassages();
+ }
+});
+```
+
+---
+
+##### Étape 10 : Documentation
+
+**Mettre à jour la documentation :**
+- [ ] `docs/TECHBOOK.md` - Section "MQTT Real-Time Communication"
+- [ ] `TODO-API.md` - Marquer la tâche comme terminée
+- [ ] Créer `docs/MQTT-INTEGRATION.md` avec :
+ - Architecture MQTT
+ - Topics disponibles
+ - Format des messages
+ - Exemples de souscription (PHP, Flutter, JavaScript)
+ - Configuration ACL et sécurité
+
+---
+
+#### 📊 Résumé des impacts
+
+**Infrastructure :**
+- Broker Mosquitto installé sur chaque environnement
+- Ports ouverts : 1883 (MQTT), 8883 (MQTTS), 9001 (WebSockets)
+
+**Services créés :**
+- `MqttService.php` : Publication et gestion des messages
+
+**Controllers modifiés :**
+- `PassageController.php` : Publication événements passages
+- `SectorController.php` : Publication événements secteurs
+- `OperationController.php` : Publication événements opérations
+
+**Configuration modifiée :**
+- `AppConfig.php` : Section 'mqtt'
+- `composer.json` : Dépendance `php-mqtt/client`
+
+**Nouveaux endpoints :**
+- `POST /api/mqtt/token` : Génération token JWT pour Flutter
+
+---
+
+#### ⚠️ Points d'attention
+
+1. **Performance** : MQTT est asynchrone, ne doit JAMAIS bloquer l'API
+2. **Mode dégradé** : L'API doit fonctionner même si MQTT est HS
+3. **Sécurité** : ACL strictes, authentification obligatoire
+4. **Scalabilité** : Prévoir un broker MQTT dédié si > 1000 clients
+5. **Monitoring** : Logs Mosquitto à surveiller (connexions, déconnexions, erreurs)
+6. **Certificats SSL** : Obligatoires en production (Let's Encrypt)
+7. **QoS** : Utiliser QoS 1 (at least once) pour garantir la livraison
+
+---
+
+#### 🔧 Architecture déployée
+
+**Environnement DVA (IN3) :**
+```
+Container dva-geo (13.23.33.43)
+ ├── API PHP (nginx + php-fpm)
+ └── Mosquitto broker (1883, 8883, 9001)
+```
+
+**Environnement RCA (IN3) :**
+```
+Container rca-geo (13.23.33.23)
+ ├── API PHP
+ └── Mosquitto broker
+```
+
+**Environnement PROD (IN4) :**
+```
+Container pra-geo (13.23.33.22)
+ ├── API PHP
+ └── Mosquitto broker
+```
+
+**Alternative (scalable) :**
+```
+Container mqtt-broker (dédié)
+ ├── Mosquitto avec authentification JWT
+ └── Monitoring (Prometheus + Grafana)
+```
+
+---
+
+**Statut :** 📋 PLANIFIÉ - En attente de validation pour démarrage
+**Date de création :** 08/11/2025
+**Durée estimée :** 2-3 jours (installation + développement + tests)
+**Dépendances :** Aucune
+**Priorité :** HAUTE - Améliore significativement l'expérience utilisateur
+
+---
+
+#### 9. Envoi de SMS de reçu en alternative à l'email
+
+**Demandé le :** 08/11/2025
+**Objectif :** Permettre l'envoi de SMS de reçu au contributeur lors de la création/modification de passages de type 1 ou 5, comme alternative à l'email.
+
+**Contexte :**
+- La table `entites` possède déjà le champ `chk_accept_sms` (boolean)
+- Actuellement, les reçus sont envoyés par email pour les passages `fk_type = 1` ou `5`
+- Besoin d'envoyer un SMS de reçu si numéro mobile valide (10 chiffres commençant par 06 ou 07)
+- Comptabilisation par entité pour facturation ultérieure
+
+**Cas d'usage :**
+```
+Création passage avec fk_type = 1 ou 5
+ ↓ Si entite.chk_accept_sms = 1
+ ↓ Si encrypted_phone est mobile (06/07...)
+ ↓ Envoyer SMS reçu au lieu d'email
+ ↓ Comptabiliser dans sms_sent_count
+```
+
+---
+
+#### 📋 Plan d'action détaillé
+
+##### Étape 1 : Modifications base de données
+
+**A. Table `entites`** - Ajouter compteur et gestion pack SMS :
+
+```sql
+ALTER TABLE entites
+ADD COLUMN sms_sent_count INT UNSIGNED DEFAULT 0 COMMENT 'Nombre total de SMS envoyés pour facturation',
+ADD COLUMN sms_credits INT UNSIGNED DEFAULT 0 COMMENT 'Crédits SMS disponibles (si pack dédié)',
+ADD COLUMN sms_pack_type ENUM('global', 'entity') DEFAULT 'global' COMMENT 'Type de pack SMS (global partagé ou dédié)',
+ADD KEY idx_chk_accept_sms (chk_accept_sms);
+```
+
+**B. Nouvelle table `sms_log`** - Historique des envois pour audit :
+
+```sql
+CREATE TABLE IF NOT EXISTS sms_log (
+ id INT UNSIGNED AUTO_INCREMENT PRIMARY KEY,
+ fk_entite INT UNSIGNED NOT NULL,
+ fk_passage INT UNSIGNED NULL COMMENT 'Passage concerné par le SMS',
+ phone VARCHAR(20) NOT NULL COMMENT 'Numéro destinataire (format international)',
+ message TEXT NOT NULL COMMENT 'Contenu du SMS envoyé',
+ status ENUM('pending', 'sent', 'failed', 'error') DEFAULT 'pending',
+ ovh_message_id VARCHAR(100) NULL COMMENT 'ID retourné par API OVH',
+ credits_used DECIMAL(5,2) DEFAULT 1.00 COMMENT 'Crédits SMS consommés',
+ error_message TEXT NULL,
+ sent_at TIMESTAMP NULL,
+ created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
+
+ KEY idx_fk_entite (fk_entite),
+ KEY idx_fk_passage (fk_passage),
+ KEY idx_status (status),
+ KEY idx_sent_at (sent_at),
+
+ CONSTRAINT sms_log_ibfk_1 FOREIGN KEY (fk_entite) REFERENCES entites(id) ON DELETE CASCADE,
+ CONSTRAINT sms_log_ibfk_2 FOREIGN KEY (fk_passage) REFERENCES ope_pass(id) ON DELETE SET NULL
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
+```
+
+**C. Table de configuration `sms_config`** :
+
+```sql
+CREATE TABLE IF NOT EXISTS sms_config (
+ id INT UNSIGNED AUTO_INCREMENT PRIMARY KEY,
+ config_key VARCHAR(50) UNIQUE NOT NULL,
+ config_value TEXT NOT NULL,
+ description VARCHAR(255) NULL,
+ updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
+
+ KEY idx_config_key (config_key)
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
+
+-- Configuration initiale
+INSERT INTO sms_config (config_key, config_value, description) VALUES
+('ovh_service_name', '', 'Nom du service SMS OVH'),
+('ovh_app_key', '', 'Application Key OVH'),
+('ovh_app_secret', '', 'Application Secret OVH (chiffré)'),
+('ovh_consumer_key', '', 'Consumer Key OVH (chiffré)'),
+('global_sms_credits', '0', 'Crédits SMS globaux disponibles'),
+('default_sender', 'GEOSECTOR', 'Nom d\'expéditeur (11 chars max)'),
+('sms_enabled', '1', 'Activation globale du système SMS'),
+('sms_receipt_template', 'Merci pour votre don de {montant}EUR par {reglement}. Recu fiscal disponible sur demande. {entite_nom}', 'Template du SMS de reçu');
+```
+
+**Fichiers impactés :**
+- [ ] Créer `scripts/migrations/add_sms_receipts.sql`
+- [ ] Appliquer sur DVA, REC, PROD
+
+---
+
+##### Étape 2 : Configuration AppConfig.php
+
+Ajouter la configuration OVH SMS :
+
+```php
+// Dans src/Config/AppConfig.php
+'sms' => [
+ 'enabled' => true,
+ 'provider' => 'ovh',
+ 'ovh' => [
+ 'endpoint' => 'ovh-eu',
+ 'application_key' => '', // Chargé depuis sms_config
+ 'application_secret' => '', // Chargé depuis sms_config
+ 'consumer_key' => '', // Chargé depuis sms_config
+ 'service_name' => '', // Chargé depuis sms_config
+ ],
+ 'default_sender' => 'GEOSECTOR',
+ 'max_length' => 160,
+],
+```
+
+**Fichiers impactés :**
+- [ ] Modifier `src/Config/AppConfig.php` (3 environnements)
+
+---
+
+##### Étape 3 : Installation SDK OVH
+
+```bash
+composer require ovh/ovh
+```
+
+**Fichiers impactés :**
+- [ ] Modifier `composer.json`
+- [ ] Exécuter sur DVA, REC, PROD
+
+---
+
+##### Étape 4 : Nouveau service SmsService
+
+Créer `src/Services/SmsService.php` avec les méthodes principales :
+
+- `sendReceiptSms()` - Envoie un SMS de reçu après un passage payé
+- `canEntitySendSms()` - Vérifie si l'entité peut envoyer des SMS (crédits, autorisation)
+- `isValidMobile()` - Valide un numéro mobile français (06/07)
+- `formatPhoneNumber()` - Formate au format international (+33...)
+- `generateReceiptMessage()` - Génère le message depuis template
+- `incrementSmsCounter()` - Incrémente le compteur de facturation
+- `decrementCredits()` - Décremente les crédits (global ou entity)
+- `logSms()` - Enregistre dans sms_log
+
+**Template de message par défaut :**
+```
+Merci pour votre don de {montant}EUR par {reglement}. Recu fiscal disponible sur demande. {entite_nom}
+```
+
+**Fichiers impactés :**
+- [ ] Créer `src/Services/SmsService.php`
+
+---
+
+##### Étape 5 : Modifier PassageController
+
+**Modifications dans `PassageController::create()` (lignes 606-639) :**
+
+Ajouter logique de décision SMS vs Email :
+
+```php
+$fkType = isset($data['fk_type']) ? (int)$data['fk_type'] : 0;
+if ($fkType === 1 || $fkType === 5) {
+ // Récupérer l'entity_id de l'opération
+ $stmtEntity = $this->db->prepare('SELECT fk_entite FROM operations WHERE id = ?');
+ $stmtEntity->execute([$operationId]);
+ $entityId = (int)$stmtEntity->fetchColumn();
+
+ // Vérifier si l'entité accepte les SMS
+ $stmtEntite = $this->db->prepare('SELECT chk_accept_sms FROM entites WHERE id = ?');
+ $stmtEntite->execute([$entityId]);
+ $acceptSms = (bool)$stmtEntite->fetchColumn();
+
+ // Déterminer si on envoie SMS ou Email
+ $shouldSendSms = false;
+ $phone = null;
+
+ if ($acceptSms && !empty($encryptedPhone)) {
+ $phone = ApiService::decryptData($encryptedPhone);
+ $smsService = new \App\Services\SmsService($this->db);
+ if ($smsService->isValidMobile($phone)) {
+ $shouldSendSms = true;
+ }
+ }
+
+ if ($shouldSendSms && $phone) {
+ // ENVOI SMS via register_shutdown_function
+ // Récupérer type de règlement pour le message
+ // Appeler sendReceiptSms()
+ } else {
+ // ENVOI EMAIL (logique existante)
+ }
+}
+```
+
+**Même logique dans `PassageController::update()` si changement de type.**
+
+**Fichiers impactés :**
+- [ ] Modifier `src/Controllers/PassageController.php`
+
+---
+
+##### Étape 6 : Endpoints API pour gestion SMS (optionnel)
+
+**Créer `SmsController.php`** avec :
+
+- `GET /api/sms/stats/{entity_id}` - Statistiques SMS d'une entité
+- `GET /api/sms/history/{entity_id}` - Historique des envois
+- `GET /api/sms/credits` - Solde crédits global et par entité
+- `PUT /api/sms/config` - Configuration SMS (admin)
+
+**Fichiers impactés :**
+- [ ] Créer `src/Controllers/SmsController.php` (optionnel)
+- [ ] Modifier `src/Core/Router.php` (optionnel)
+
+---
+
+##### Étape 7 : Tests et validation
+
+**Tests sur DVA :**
+
+- [ ] Créer entité test avec `chk_accept_sms = 1`
+- [ ] Créer passage `fk_type = 1` avec mobile valide (06/07)
+- [ ] Vérifier envoi SMS via OVH
+- [ ] Vérifier enregistrement dans `sms_log`
+- [ ] Vérifier incrémentation de `sms_sent_count`
+- [ ] Vérifier débit des crédits
+
+**Tests de non-régression :**
+
+- [ ] Entité avec `chk_accept_sms = 0` → Email uniquement
+- [ ] Numéro fixe (01-05, 08, 09) → Email uniquement
+- [ ] Sans téléphone mais avec email → Email uniquement
+- [ ] Passage `fk_type = 2, 3, 4` → Pas d'envoi
+
+**Fichiers impactés :**
+- [ ] Créer `scripts/test/test_sms_receipts.php`
+
+---
+
+##### Étape 8 : Documentation
+
+**Mettre à jour la documentation :**
+
+- [ ] `docs/TECHBOOK.md` - Section "Envoi de SMS de reçu"
+- [ ] `TODO-API.md` - Marquer la tâche comme terminée
+- [ ] Créer `docs/SMS-RECEIPTS.md` avec :
+ - Configuration OVH SMS
+ - Template de message personnalisable
+ - Gestion des crédits (global vs entité)
+ - Facturation basée sur `sms_sent_count`
+
+---
+
+#### 📊 Résumé des impacts
+
+**Tables créées :**
+- `sms_log` : Historique des envois
+- `sms_config` : Configuration OVH
+
+**Tables modifiées :**
+- `entites` : +3 colonnes (sms_sent_count, sms_credits, sms_pack_type)
+
+**Services créés :**
+- `SmsService.php` : Gestion envoi SMS via OVH
+
+**Controllers modifiés :**
+- `PassageController.php` : Logique SMS vs Email selon contexte
+
+**Configuration modifiée :**
+- `AppConfig.php` : Section 'sms'
+- `composer.json` : Dépendance `ovh/ovh`
+
+---
+
+#### ⚠️ Points d'attention
+
+1. **Validation numéro mobile** : Uniquement 06/07 (10 chiffres)
+2. **Longueur SMS** : 160 caractères max (template à optimiser)
+3. **Coût** : ~0.035€/SMS en France
+4. **Priorité** : SMS si mobile valide, sinon Email en fallback
+5. **Comptabilisation** : `sms_sent_count` pour facturation
+6. **Credentials OVH** : À ne JAMAIS commiter, stockés en base chiffrés
+7. **Template personnalisable** : Dans `sms_config.sms_receipt_template`
+
+---
+
+#### 🔧 Configuration OVH requise
+
+**Étapes préalables :**
+
+1. Créer application OVH : https://eu.api.ovh.com/createApp/
+2. Générer Consumer Key avec droits `/sms/*`
+3. Commander service SMS OVH
+4. Créditer le compte (minimum 100-200 SMS pour tests)
+
+**Stockage en base :**
+
+```sql
+UPDATE sms_config SET config_value = 'sms-ab123456-1' WHERE config_key = 'ovh_service_name';
+UPDATE sms_config SET config_value = 'xxxxx' WHERE config_key = 'ovh_app_key';
+UPDATE sms_config SET config_value = 'xxxxx' WHERE config_key = 'ovh_app_secret';
+UPDATE sms_config SET config_value = 'xxxxx' WHERE config_key = 'ovh_consumer_key';
+UPDATE sms_config SET config_value = '1000' WHERE config_key = 'global_sms_credits';
+```
+
+---
+
+**Statut :** 📋 PLANIFIÉ - En attente de démarrage
+**Date de création :** 08/11/2025
+**Durée estimée :** 2-3 jours (développement + tests + configuration OVH)
+**Dépendances :** Compte OVH avec service SMS actif
+
+---
+
+#### 9. Intégration de la base "batiments" dans la gestion des secteurs
+
+**Demandé le :** 07/11/2025
+**Objectif :** Intégrer la base de données "batiments" pour identifier les immeubles et créer automatiquement un passage par logement lors de la création/modification des secteurs.
+
+**Contexte :**
+
+- **Nouvelle base** : `batiments` (même host/credentials que base `adresses`)
+- **Structure** : Tables par département `bat01`, `bat02`, etc.
+- **Lien** : `bat{dept}.cle_interop_adr` → `cp{dept}.id`
+- **Contenu** : Uniquement les immeubles avec métadonnées (nb_niveau, nb_log, residence, etc.)
+
+**Structure de la table bat{dept} :**
+
+```sql
+batiment_groupe_id VARCHAR(50) PRIMARY KEY
+code_departement_insee VARCHAR(5)
+cle_interop_adr VARCHAR(50) -- Lien vers cp{dept}.id
+nb_niveau INT -- Nombre d'étages
+nb_log INT -- Nombre de logements
+nb_pdl_tot INT -- Compteurs électriques
+annee_construction INT
+residence VARCHAR(200) -- Nom de la copropriété
+usage_principal VARCHAR(100)
+altitude_sol_mean DECIMAL(10,2)
+gps_lat DECIMAL(10,7)
+gps_lng DECIMAL(10,7)
+```
+
+**Filtre appliqué lors de l'import :**
+
+- `usage_principal IN ('Résidentiel individuel', 'Résidentiel collectif', 'Secondaire', 'Tertiaire')`
+- `nb_log > 1` (immeubles uniquement, pas les maisons individuelles)
+
+---
+
+#### 📋 Plan d'action détaillé
+
+##### Étape 1 : Modifications base de données
+
+**A. Table `sectors_adresses`** - Ajouter colonnes bâtiment :
+
+```sql
+ALTER TABLE sectors_adresses
+ADD COLUMN fk_batiment VARCHAR(50) NULL COMMENT 'batiment_groupe_id' AFTER fk_adresse,
+ADD COLUMN fk_habitat TINYINT NULL COMMENT '1=individuel, 2=collectif',
+ADD COLUMN nb_niveau INT NULL COMMENT 'Nombre d\'étages',
+ADD COLUMN nb_log INT NULL COMMENT 'Nombre de logements',
+ADD COLUMN residence VARCHAR(200) NULL COMMENT 'Nom copropriété',
+ADD COLUMN alt_sol DECIMAL(10,2) NULL COMMENT 'Altitude sol',
+ADD KEY idx_fk_batiment (fk_batiment),
+ADD KEY idx_fk_habitat (fk_habitat);
+```
+
+**B. Table `ope_pass`** - Vérifier/modifier colonne habitat :
+
+```sql
+-- Vérifier si fk_habitat existe déjà
+SHOW COLUMNS FROM ope_pass LIKE 'fk_habitat';
+
+-- Si nécessaire, modifier la colonne
+ALTER TABLE ope_pass
+MODIFY COLUMN fk_habitat TINYINT NULL COMMENT '1=individuel, 2=collectif';
+
+-- Ajouter index si manquant
+ALTER TABLE ope_pass ADD KEY idx_fk_habitat (fk_habitat);
+```
+
+**Fichiers impactés :**
+
+- [ ] Créer `scripts/migrations/add_batiments_integration.sql`
+- [ ] Appliquer sur DVA, REC, PROD
+
+---
+
+##### Étape 2 : Configuration AppConfig.php
+
+Ajouter la configuration de la base batiments :
+
+```php
+// Dans src/Config/AppConfig.php
+'buildings_database' => [
+ 'host' => '13.23.33.46', // DVA: 13.23.33.46
+ 'name' => 'batiments', // RCA: 13.23.33.36
+ 'username' => 'adr_geo_user', // PROD: 13.23.33.26
+ 'password' => 'd66,AdrGeoDev.User',
+],
+```
+
+**Fichiers impactés :**
+
+- [ ] Modifier `src/Config/AppConfig.php` (3 environnements DEV/REC/PROD)
+
+---
+
+##### Étape 3 : Nouveau service BuildingService
+
+Créer `src/Services/BuildingService.php` :
+
+**Méthodes à implémenter :**
+
+```php
+class BuildingService {
+ private \PDO $dbBuildings;
+
+ public function __construct()
+ {
+ // Connexion PDO vers base "batiments"
+ $config = AppConfig::getInstance()->get('buildings_database');
+ $this->dbBuildings = new \PDO(...);
+ }
+
+ /**
+ * Récupère les bâtiments dans un polygone
+ * @param array $coordinates Format [[lat, lng], ...]
+ * @param int|null $entityId Pour déterminer les départements
+ * @return array Liste des bâtiments avec métadonnées
+ */
+ public function getBuildingsInPolygon(array $coordinates, ?int $entityId = null): array
+
+ /**
+ * Compte les bâtiments dans un polygone
+ */
+ public function countBuildingsInPolygon(array $coordinates, ?int $entityId = null): int
+
+ /**
+ * Vérifie si la connexion à la base batiments est active
+ */
+ public function isConnected(): bool
+}
+```
+
+**Logique similaire à AddressService :**
+
+- Détection automatique des départements touchés par le polygone
+- Requête SQL spatiale sur toutes les tables `bat{dept}` concernées
+- Utilisation de `ST_Contains()` pour filtrer les bâtiments
+
+**Fichiers impactés :**
+
+- [ ] Créer `src/Services/BuildingService.php`
+
+---
+
+##### Étape 4 : Enrichir AddressService
+
+Modifier `src/Services/AddressService.php` :
+
+**Nouvelle méthode à ajouter :**
+
+```php
+/**
+ * Enrichit les adresses avec les données bâtiment si disponibles
+ * @param array $addresses Liste des adresses depuis getAddressesInPolygon()
+ * @param int|null $entityId
+ * @return array Adresses enrichies avec fk_batiment, fk_habitat, nb_log, etc.
+ */
+public function enrichAddressesWithBuildings(array $addresses, ?int $entityId = null): array
+{
+ // Pour chaque adresse, chercher si cp{dept}.id correspond à bat{dept}.cle_interop_adr
+ // Enrichir avec : fk_batiment, fk_habitat, nb_niveau, nb_log, residence, alt_sol
+ // Retourner les adresses enrichies
+}
+```
+
+**Logique :**
+
+1. Grouper les adresses par département
+2. Pour chaque département, faire un JOIN entre `cp{dept}` et `bat{dept}`
+3. Enrichir les adresses avec les données bâtiment trouvées
+4. Définir `fk_habitat = 2` (collectif) si bâtiment trouvé, sinon `fk_habitat = 1` (individuel)
+
+**Fichiers impactés :**
+
+- [ ] Modifier `src/Services/AddressService.php`
+
+---
+
+##### Étape 5 : Modifier SectorController::create()
+
+**Modifications dans la méthode `create()` (ligne 88-534) :**
+
+```php
+// APRÈS ligne 289 - Récupération des adresses
+$addresses = $this->addressService->getAddressesInPolygon($coordinates, $entityId);
+
+// NOUVEAU : Enrichir avec données batiments
+$addresses = $this->addressService->enrichAddressesWithBuildings($addresses, $entityId);
+
+// MODIFIER ligne 292-311 : Stockage dans sectors_adresses avec nouvelles colonnes
+foreach ($addresses as $address) {
+ $stmtAddress->execute([
+ 'sector_id' => $sectorId,
+ 'address_id' => $address['id'],
+ 'numero' => $address['numero'],
+ 'rue' => $address['voie'],
+ 'rue_bis' => '',
+ 'cp' => $address['code_postal'],
+ 'ville' => $address['commune'],
+ 'gps_lat' => $address['latitude'],
+ 'gps_lng' => $address['longitude'],
+ // NOUVELLES COLONNES
+ 'fk_batiment' => $address['fk_batiment'] ?? null,
+ 'fk_habitat' => $address['fk_habitat'] ?? 1,
+ 'nb_niveau' => $address['nb_niveau'] ?? null,
+ 'nb_log' => $address['nb_log'] ?? null,
+ 'residence' => $address['residence'] ?? null,
+ 'alt_sol' => $address['alt_sol'] ?? null,
+ ]);
+}
+
+// MODIFIER ligne 314-377 : Création des passages selon fk_habitat
+foreach ($addresses as $address) {
+ // Exclure les adresses déjà utilisées par passages orphelins
+ if (in_array($address['id'], $addressesToExclude)) {
+ continue;
+ }
+
+ if ($address['fk_habitat'] == 2 && !empty($address['nb_log']) && $address['nb_log'] > 1) {
+ // IMMEUBLE : créer nb_log passages (1 par logement)
+ for ($appt = 1; $appt <= $address['nb_log']; $appt++) {
+ $passageStmt->execute([
+ 'operation_id' => $operationId,
+ 'sector_id' => $sectorId,
+ 'user_id' => $firstOpeUserId,
+ 'fk_adresse' => $address['id'],
+ 'numero' => $address['numero'],
+ 'rue' => $address['voie'],
+ 'rue_bis' => '',
+ 'ville' => $address['commune'],
+ 'gps_lat' => $address['latitude'],
+ 'gps_lng' => $address['longitude'],
+ 'fk_habitat' => 2,
+ 'appt' => $appt, // Numéro d'appartement
+ 'residence' => $address['residence'] ?? null,
+ 'user_creat' => $userId
+ ]);
+ $passagesCreated++;
+ }
+ } else {
+ // MAISON INDIVIDUELLE : 1 seul passage
+ $passageStmt->execute([
+ // ... même structure avec fk_habitat=1, appt=null
+ ]);
+ $passagesCreated++;
+ }
+}
+```
+
+**Fichiers impactés :**
+
+- [ ] Modifier `src/Controllers/SectorController.php` (méthode `create()`)
+
+---
+
+##### Étape 6 : Modifier SectorController::update()
+
+**Modifications dans la méthode `update()` (ligne 539-938) :**
+
+1. Ligne 702-773 : Mise à jour de `sectors_adresses` avec nouvelles colonnes
+2. Ligne 1243-1557 : Modifier `updatePassagesForSector()` pour gérer les immeubles
+
+**Dans `updatePassagesForSector()` :**
+
+```php
+// Ligne 1346-1350 : Récupérer adresses depuis sectors_adresses
+$addressesQuery = "SELECT * FROM sectors_adresses WHERE fk_sector = :sector_id";
+$addresses = $addressesStmt->fetchAll();
+
+// Ligne 1456-1495 : INSERT MULTIPLE - Adapter pour créer nb_log passages si immeuble
+foreach ($toInsert as $addr) {
+ if ($addr['fk_habitat'] == 2 && $addr['nb_log'] > 1) {
+ // Créer nb_log passages pour cet immeuble
+ for ($appt = 1; $appt <= $addr['nb_log']; $appt++) {
+ // Ajouter à la liste d'insertion
+ }
+ } else {
+ // Créer 1 seul passage
+ }
+}
+```
+
+**Fichiers impactés :**
+
+- [ ] Modifier `src/Controllers/SectorController.php` (méthodes `update()` et `updatePassagesForSector()`)
+
+---
+
+##### Étape 7 : Tests et validation
+
+**Tests sur DVA :**
+
+- [ ] Créer un secteur test contenant des immeubles connus
+- [ ] Vérifier le nombre de passages créés (doit correspondre au total de logements)
+- [ ] Valider les données dans `sectors_adresses` (colonnes batiments remplies)
+- [ ] Valider les données dans `ope_pass` (fk_habitat, appt, residence)
+- [ ] Tester la modification d'un secteur (agrandissement/réduction)
+- [ ] Vérifier le comportement avec passages orphelins
+
+**Tests de non-régression :**
+
+- [ ] Secteurs sans immeubles (maisons individuelles uniquement)
+- [ ] Secteurs multi-départements
+- [ ] Secteurs sans adresses (base adresses inaccessible)
+
+**Fichiers impactés :**
+
+- [ ] Créer `scripts/test/test_batiments_integration.php`
+
+---
+
+##### Étape 8 : Documentation
+
+**Mettre à jour la documentation :**
+
+- [ ] `docs/GESTION-SECTORS.md` - Section "Gestion des bâtiments"
+- [ ] `docs/TECHBOOK.md` - Section "Base de données"
+- [ ] `TODO-API.md` - Marquer la tâche comme terminée
+
+**Contenu à documenter :**
+
+- Structure de la table `bat{dept}`
+- Logique de création de passages pour immeubles (1 passage par logement)
+- Format de réponse API enrichi avec données bâtiments
+- Exemples de requêtes SQL pour interroger les bâtiments
+
+---
+
+#### 📊 Résumé des impacts
+
+**Tables modifiées :**
+
+- `sectors_adresses` : +6 colonnes
+- `ope_pass` : Modification colonne `fk_habitat` (si nécessaire)
+
+**Services créés/modifiés :**
+
+- **Nouveau** : `BuildingService.php`
+- **Modifié** : `AddressService.php` (enrichissement)
+
+**Controllers modifiés :**
+
+- `SectorController.php` : Méthodes `create()`, `update()`, `updatePassagesForSector()`
+
+**Configuration modifiée :**
+
+- `AppConfig.php` : Ajout `buildings_database`
+
+**Impact sur la création de secteur :**
+
+- **AVANT** : 1 passage par adresse
+- **APRÈS** : 1 passage par logement (immeubles = nb_log passages)
+- **Exemple** : Secteur avec 100 adresses dont 20 immeubles de 10 logements
+ - Avant : 100 passages
+ - Après : 80 maisons + (20 × 10 immeubles) = 280 passages
+
+---
+
+#### ⚠️ Points d'attention
+
+1. **Performance** : Les immeubles génèrent beaucoup plus de passages
+ - Optimiser les INSERT multiple dans `updatePassagesForSector()`
+ - Prévoir un timeout plus long pour les gros secteurs
+
+2. **Cohérence des données** : La base `batiments` doit être à jour
+ - Vérifier la présence des départements concernés
+ - Gérer le cas où la base batiments est inaccessible (fallback)
+
+3. **Migration des données existantes** : Les secteurs déjà créés
+ - Option 1 : Ne pas toucher aux secteurs existants
+ - Option 2 : Script de migration pour enrichir les secteurs existants
+
+4. **Synchronisation** : Base `batiments` vs `adresses`
+ - Le lien `cle_interop_adr` doit être valide
+ - Gérer les adresses sans bâtiment correspondant
+
+---
+
+**Statut :** 📋 PLANIFIÉ - En attente de validation pour démarrage
+**Date de création :** 07/11/2025
+**Durée estimée :** 2-3 jours (développement + tests)
+
+---
+
+## 🟡 PRIORITÉ MOYENNE
+
+### 9. Migration complète des Services vers namespace App\Services
+
+**Date :** 07/11/2025
+**Objectif :** Migrer tous les services vers le namespace `App\Services` pour améliorer l'organisation du code et l'autoloading.
+
+**Contexte :**
+Lors de l'intégration de la base batiments, nous avons commencé à migrer certains services vers le namespace `App\Services`. Cette migration partielle crée des conflits d'autoloading et des incohérences dans le code.
+
+**État actuel :**
+
+✅ **Services AVEC namespace App\Services (10)** :
+- AddressService.php
+- BuildingService.php
+- DepartmentBoundaryService.php
+- LogService.php
+- PasswordSecurityService.php
+- PDFGenerator.php
+- ReceiptPDFGenerator.php
+- ReceiptService.php
+- SimplePDF.php
+- StripeService.php
+
+❌ **Services SANS namespace (8 à migrer)** :
+- ApiService.php
+- BackupEncryptionService.php
+- EmailTemplates.php
+- EventLogService.php
+- ExportService.php
+- FileService.php
+- MigrationService.php
+- OperationDataService.php
+
+✅ **Services Security (tous OK)** :
+- Tous les 5 services dans `App\Services\Security` ont déjà le namespace
+
+**Plan de migration :**
+
+#### Étape 1 : Préparer l'autoloading PSR-4
+```json
+// composer.json - remplacer classmap par psr-4
+"autoload": {
+ "psr-4": {
+ "App\\": "src/"
+ }
+}
+```
+
+Puis exécuter sur DVA, RCA et PROD :
+```bash
+composer dump-autoload -o
+```
+
+#### Étape 2 : Migrer les 8 services restants
+
+Pour chaque service :
+
+1. **Ajouter le namespace** au début du fichier
+```php
+namespace App\Services;
+```
+
+2. **Ajouter les imports** pour les classes globales
+```php
+use PDO;
+use Exception;
+// etc.
+```
+
+3. **Vérifier les dépendances** : si le service utilise d'autres services, s'assurer que les imports sont corrects
+
+4. **Tester** : s'assurer que le service fonctionne correctement
+
+#### Étape 3 : Mettre à jour les fichiers qui utilisent ces services
+
+Pour chaque fichier utilisant les services migrés :
+- Ajouter `use App\Services\NomDuService;` si le fichier a un namespace différent
+- Ou utiliser le nom court si dans le même namespace
+
+#### Ordre de migration recommandé :
+
+1. **EmailTemplates.php** (utilisé par ApiService, ReceiptService)
+2. **ApiService.php** (utilisé par beaucoup de contrôleurs)
+3. **FileService.php** (utilisé par ExportService, ReceiptService)
+4. **ExportService.php** (autonome)
+5. **OperationDataService.php** (autonome)
+6. **BackupEncryptionService.php** (autonome)
+7. **EventLogService.php** (autonome)
+8. **MigrationService.php** (scripts uniquement)
+
+**Fichiers à vérifier après migration :**
+
+Controllers utilisant ces services :
+- `src/Controllers/*.php` - vérifier tous les imports
+- `scripts/cron/*.php` - vérifier les scripts cron
+- `scripts/php/*.php` - vérifier les scripts de migration
+
+**Tests à effectuer :**
+
+1. **Sur DVA** :
+ - Tester tous les endpoints principaux
+ - Vérifier les logs (pas d'erreur Class not found)
+ - Tester l'envoi d'emails (ApiService)
+ - Tester les exports (ExportService)
+ - Tester les reçus PDF (ReceiptService + EmailTemplates)
+
+2. **Sur RCA** :
+ - Tests de non-régression complets
+ - Validation par un utilisateur test
+
+3. **Sur PROD** :
+ - Déploiement progressif
+ - Monitoring des logs pendant 24h
+
+**Risques et mitigation :**
+
+⚠️ **Risque** : Erreur "Class not found" si l'autoloading ne fonctionne pas
+✅ **Mitigation** : Tester d'abord sur DVA avec rollback rapide possible
+
+⚠️ **Risque** : Services utilisés par scripts cron qui cassent
+✅ **Mitigation** : Lister tous les scripts et les tester individuellement
+
+⚠️ **Risque** : Cache d'autoload pas régénéré
+✅ **Mitigation** : Forcer `composer dump-autoload -o` sur chaque environnement
+
+**Impact sur les environnements :**
+
+- **DVA** : Test complet de la migration
+- **RCA** : Validation avant production
+- **PROD** : Déploiement final
+
+**Prérequis :**
+
+- ✅ Accès SSH aux 3 environnements (DVA, RCA, PROD)
+- ✅ Composer installé sur chaque environnement
+- ✅ Backup de la version actuelle du code
+
+---
+
+**Statut :** 🔄 EN COURS - Migration partielle effectuée (10/18 services)
+**Date de création :** 07/11/2025
+**Durée estimée :** 1 jour (migration + tests)
+**Priorité :** MOYENNE - Bloque le déploiement de l'intégration batiments
+
+---
+
+### 🟢 PRIORITÉ BASSE
+
+#### 10. Documentation API
+
- Génération automatique OpenAPI/Swagger
- Documentation interactive
- Exemples de code pour chaque endpoint
-#### 6. Tests automatisés
+#### 11. Tests automatisés
+
- Tests unitaires pour les services critiques
- Tests d'intégration pour les endpoints
- Tests de charge
@@ -139,4 +2698,4 @@ INSERT INTO ope_pass_backup (
---
-**Dernière mise à jour :** 20/08/2025
\ No newline at end of file
+**Dernière mise à jour :** 07/11/2025
diff --git a/api/composer.json b/api/composer.json
index d08c4970..01d724fe 100755
--- a/api/composer.json
+++ b/api/composer.json
@@ -8,13 +8,20 @@
"ext-openssl": "*",
"ext-pdo": "*",
"phpmailer/phpmailer": "^6.8",
- "phpoffice/phpspreadsheet": "^2.0",
+ "phpoffice/phpspreadsheet": "^5.0",
"setasign/fpdf": "^1.8",
+ "setasign/fpdi": "^2.6",
"stripe/stripe-php": "^17.6"
},
"autoload": {
+ "psr-4": {
+ "App\\": "src/"
+ },
"classmap": [
- "src/"
+ "src/Core/",
+ "src/Config/",
+ "src/Utils/",
+ "src/Controllers/LogController.php"
]
},
"config": {
diff --git a/api/composer.lock b/api/composer.lock
index 942ab502..a926d49d 100755
--- a/api/composer.lock
+++ b/api/composer.lock
@@ -4,7 +4,7 @@
"Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies",
"This file is @generated automatically"
],
- "content-hash": "155893f9be89bceda3639efbf19b14d1",
+ "content-hash": "936a7e1a35fde56354a4dea02b309267",
"packages": [
{
"name": "composer/pcre",
@@ -87,22 +87,22 @@
},
{
"name": "maennchen/zipstream-php",
- "version": "3.1.2",
+ "version": "3.2.0",
"source": {
"type": "git",
"url": "https://github.com/maennchen/ZipStream-PHP.git",
- "reference": "aeadcf5c412332eb426c0f9b4485f6accba2a99f"
+ "reference": "9712d8fa4cdf9240380b01eb4be55ad8dcf71416"
},
"dist": {
"type": "zip",
- "url": "https://api.github.com/repos/maennchen/ZipStream-PHP/zipball/aeadcf5c412332eb426c0f9b4485f6accba2a99f",
- "reference": "aeadcf5c412332eb426c0f9b4485f6accba2a99f",
+ "url": "https://api.github.com/repos/maennchen/ZipStream-PHP/zipball/9712d8fa4cdf9240380b01eb4be55ad8dcf71416",
+ "reference": "9712d8fa4cdf9240380b01eb4be55ad8dcf71416",
"shasum": ""
},
"require": {
"ext-mbstring": "*",
"ext-zlib": "*",
- "php-64bit": "^8.2"
+ "php-64bit": "^8.3"
},
"require-dev": {
"brianium/paratest": "^7.7",
@@ -111,7 +111,7 @@
"guzzlehttp/guzzle": "^7.5",
"mikey179/vfsstream": "^1.6",
"php-coveralls/php-coveralls": "^2.5",
- "phpunit/phpunit": "^11.0",
+ "phpunit/phpunit": "^12.0",
"vimeo/psalm": "^6.0"
},
"suggest": {
@@ -153,7 +153,7 @@
],
"support": {
"issues": "https://github.com/maennchen/ZipStream-PHP/issues",
- "source": "https://github.com/maennchen/ZipStream-PHP/tree/3.1.2"
+ "source": "https://github.com/maennchen/ZipStream-PHP/tree/3.2.0"
},
"funding": [
{
@@ -161,7 +161,7 @@
"type": "github"
}
],
- "time": "2025-01-27T12:07:53+00:00"
+ "time": "2025-07-17T11:15:13+00:00"
},
{
"name": "markbaker/complex",
@@ -272,16 +272,16 @@
},
{
"name": "phpmailer/phpmailer",
- "version": "v6.10.0",
+ "version": "v6.11.1",
"source": {
"type": "git",
"url": "https://github.com/PHPMailer/PHPMailer.git",
- "reference": "bf74d75a1fde6beaa34a0ddae2ec5fce0f72a144"
+ "reference": "d9e3b36b47f04b497a0164c5a20f92acb4593284"
},
"dist": {
"type": "zip",
- "url": "https://api.github.com/repos/PHPMailer/PHPMailer/zipball/bf74d75a1fde6beaa34a0ddae2ec5fce0f72a144",
- "reference": "bf74d75a1fde6beaa34a0ddae2ec5fce0f72a144",
+ "url": "https://api.github.com/repos/PHPMailer/PHPMailer/zipball/d9e3b36b47f04b497a0164c5a20f92acb4593284",
+ "reference": "d9e3b36b47f04b497a0164c5a20f92acb4593284",
"shasum": ""
},
"require": {
@@ -302,6 +302,7 @@
},
"suggest": {
"decomplexity/SendOauth2": "Adapter for using XOAUTH2 authentication",
+ "ext-imap": "Needed to support advanced email address parsing according to RFC822",
"ext-mbstring": "Needed to send email in multibyte encoding charset or decode encoded addresses",
"ext-openssl": "Needed for secure SMTP sending and DKIM signing",
"greew/oauth2-azure-provider": "Needed for Microsoft Azure XOAUTH2 authentication",
@@ -341,7 +342,7 @@
"description": "PHPMailer is a full-featured email creation and transfer class for PHP",
"support": {
"issues": "https://github.com/PHPMailer/PHPMailer/issues",
- "source": "https://github.com/PHPMailer/PHPMailer/tree/v6.10.0"
+ "source": "https://github.com/PHPMailer/PHPMailer/tree/v6.11.1"
},
"funding": [
{
@@ -349,24 +350,24 @@
"type": "github"
}
],
- "time": "2025-04-24T15:19:31+00:00"
+ "time": "2025-09-30T11:54:53+00:00"
},
{
"name": "phpoffice/phpspreadsheet",
- "version": "2.3.8",
+ "version": "5.1.0",
"source": {
"type": "git",
"url": "https://github.com/PHPOffice/PhpSpreadsheet.git",
- "reference": "7a700683743bf1c4a21837c84b266916f1aa7d25"
+ "reference": "fd26e45a814e94ae2aad0df757d9d1739c4bf2e0"
},
"dist": {
"type": "zip",
- "url": "https://api.github.com/repos/PHPOffice/PhpSpreadsheet/zipball/7a700683743bf1c4a21837c84b266916f1aa7d25",
- "reference": "7a700683743bf1c4a21837c84b266916f1aa7d25",
+ "url": "https://api.github.com/repos/PHPOffice/PhpSpreadsheet/zipball/fd26e45a814e94ae2aad0df757d9d1739c4bf2e0",
+ "reference": "fd26e45a814e94ae2aad0df757d9d1739c4bf2e0",
"shasum": ""
},
"require": {
- "composer/pcre": "^1 || ^2 || ^3",
+ "composer/pcre": "^1||^2||^3",
"ext-ctype": "*",
"ext-dom": "*",
"ext-fileinfo": "*",
@@ -395,9 +396,10 @@
"mitoteam/jpgraph": "^10.3",
"mpdf/mpdf": "^8.1.1",
"phpcompatibility/php-compatibility": "^9.3",
- "phpstan/phpstan": "^1.1",
- "phpstan/phpstan-phpunit": "^1.0",
- "phpunit/phpunit": "^9.6 || ^10.5",
+ "phpstan/phpstan": "^1.1 || ^2.0",
+ "phpstan/phpstan-deprecation-rules": "^1.0 || ^2.0",
+ "phpstan/phpstan-phpunit": "^1.0 || ^2.0",
+ "phpunit/phpunit": "^10.5",
"squizlabs/php_codesniffer": "^3.7",
"tecnickcom/tcpdf": "^6.5"
},
@@ -452,9 +454,9 @@
],
"support": {
"issues": "https://github.com/PHPOffice/PhpSpreadsheet/issues",
- "source": "https://github.com/PHPOffice/PhpSpreadsheet/tree/2.3.8"
+ "source": "https://github.com/PHPOffice/PhpSpreadsheet/tree/5.1.0"
},
- "time": "2025-02-08T03:01:45+00:00"
+ "time": "2025-09-04T05:34:49+00:00"
},
{
"name": "psr/http-client",
@@ -713,6 +715,78 @@
},
"time": "2023-06-26T14:44:25+00:00"
},
+ {
+ "name": "setasign/fpdi",
+ "version": "v2.6.4",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/Setasign/FPDI.git",
+ "reference": "4b53852fde2734ec6a07e458a085db627c60eada"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/Setasign/FPDI/zipball/4b53852fde2734ec6a07e458a085db627c60eada",
+ "reference": "4b53852fde2734ec6a07e458a085db627c60eada",
+ "shasum": ""
+ },
+ "require": {
+ "ext-zlib": "*",
+ "php": "^7.1 || ^8.0"
+ },
+ "conflict": {
+ "setasign/tfpdf": "<1.31"
+ },
+ "require-dev": {
+ "phpunit/phpunit": "^7",
+ "setasign/fpdf": "~1.8.6",
+ "setasign/tfpdf": "~1.33",
+ "squizlabs/php_codesniffer": "^3.5",
+ "tecnickcom/tcpdf": "^6.8"
+ },
+ "suggest": {
+ "setasign/fpdf": "FPDI will extend this class but as it is also possible to use TCPDF or tFPDF as an alternative. There's no fixed dependency configured."
+ },
+ "type": "library",
+ "autoload": {
+ "psr-4": {
+ "setasign\\Fpdi\\": "src/"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Jan Slabon",
+ "email": "jan.slabon@setasign.com",
+ "homepage": "https://www.setasign.com"
+ },
+ {
+ "name": "Maximilian Kresse",
+ "email": "maximilian.kresse@setasign.com",
+ "homepage": "https://www.setasign.com"
+ }
+ ],
+ "description": "FPDI is a collection of PHP classes facilitating developers to read pages from existing PDF documents and use them as templates in FPDF. Because it is also possible to use FPDI with TCPDF, there are no fixed dependencies defined. Please see suggestions for packages which evaluates the dependencies automatically.",
+ "homepage": "https://www.setasign.com/fpdi",
+ "keywords": [
+ "fpdf",
+ "fpdi",
+ "pdf"
+ ],
+ "support": {
+ "issues": "https://github.com/Setasign/FPDI/issues",
+ "source": "https://github.com/Setasign/FPDI/tree/v2.6.4"
+ },
+ "funding": [
+ {
+ "url": "https://tidelift.com/funding/github/packagist/setasign/fpdi",
+ "type": "tidelift"
+ }
+ ],
+ "time": "2025-08-05T09:57:14+00:00"
+ },
{
"name": "stripe/stripe-php",
"version": "v17.6.0",
diff --git a/api/config/nginx/pra-geo-http-only.conf b/api/config/nginx/pra-geo-http-only.conf
new file mode 100644
index 00000000..484b3d31
--- /dev/null
+++ b/api/config/nginx/pra-geo-http-only.conf
@@ -0,0 +1,200 @@
+# =============================================================================
+# Configuration NGINX PRODUCTION pour pra-geo (IN4)
+# Date: 2025-10-07
+# Environnement: PRODUCTION
+# Server: Container pra-geo (13.23.34.43)
+# Port: 80 uniquement (HTTP)
+# SSL/HTTPS: Géré par le reverse proxy NGINX sur le host IN4
+# =============================================================================
+
+# Site principal (web statique)
+server {
+ listen 80;
+ server_name geosector.fr;
+
+ root /var/www/geosector/web;
+ index index.html;
+
+ # Logs PRODUCTION
+ access_log /var/log/nginx/geosector-web_access.log combined;
+ error_log /var/log/nginx/geosector-web_error.log warn;
+
+ location / {
+ try_files $uri $uri/ /index.html;
+ }
+
+ # Assets statiques avec cache agressif
+ location ~* \.(jpg|jpeg|png|gif|ico|svg|webp|css|js|woff|woff2|ttf|eot)$ {
+ expires 1y;
+ add_header Cache-Control "public, immutable";
+ access_log off;
+ }
+
+ # Protection des fichiers sensibles
+ location ~ /\.(?!well-known) {
+ deny all;
+ access_log off;
+ log_not_found off;
+ }
+}
+
+# =============================================================================
+# APPLICATION FLUTTER + API PHP
+# =============================================================================
+
+server {
+ listen 80;
+ server_name app3.geosector.fr;
+
+ # Logs PRODUCTION
+ access_log /var/log/nginx/pra-app_access.log combined;
+ error_log /var/log/nginx/pra-app_error.log warn;
+
+ # Récupérer le vrai IP du client depuis le reverse proxy
+ set_real_ip_from 13.23.34.0/24; # Réseau Incus
+ set_real_ip_from 51.159.7.190; # IP publique IN4
+ real_ip_header X-Forwarded-For;
+ real_ip_recursive on;
+
+ # Taille maximale des uploads (pour les logos, exports, etc.)
+ client_max_body_size 10M;
+ client_body_buffer_size 128k;
+
+ # Timeouts optimisés pour PRODUCTION
+ client_body_timeout 30s;
+ client_header_timeout 30s;
+ send_timeout 60s;
+
+ # =============================================================================
+ # APPLICATION FLUTTER (contenu statique)
+ # =============================================================================
+ location / {
+ root /var/www/geosector/app;
+ index index.html;
+ try_files $uri $uri/ /index.html;
+
+ # Cache intelligent pour PRODUCTION
+ # HTML : pas de cache (pour déploiements)
+ location ~* \.html$ {
+ expires -1;
+ add_header Cache-Control "no-cache, no-store, must-revalidate";
+ }
+
+ # Assets Flutter (JS, CSS, fonts) avec hash : cache agressif
+ location ~* \.(js|css|woff|woff2|ttf|eot)$ {
+ expires 1y;
+ add_header Cache-Control "public, immutable";
+ access_log off;
+ }
+
+ # Images : cache longue durée
+ location ~* \.(jpg|jpeg|png|gif|ico|svg|webp)$ {
+ expires 30d;
+ add_header Cache-Control "public";
+ access_log off;
+ }
+ }
+
+ # =============================================================================
+ # API PHP (RESTful)
+ # =============================================================================
+ location /api/ {
+ root /var/www/geosector;
+
+ # CORS - Le reverse proxy IN4 ajoute déjà les headers CORS
+ # On les ajoute ici pour les requêtes internes si besoin
+
+ # Cache API : pas de cache (données dynamiques)
+ add_header Cache-Control "no-store, no-cache, must-revalidate" always;
+ add_header Pragma "no-cache" always;
+ add_header Expires "0" always;
+
+ # Rewrite vers index.php
+ try_files $uri $uri/ /api/index.php$is_args$args;
+
+ # Traitement PHP
+ location ~ ^/api/(.+\.php)$ {
+ root /var/www/geosector;
+
+ # FastCGI PHP-FPM
+ fastcgi_pass unix:/run/php-fpm83/php-fpm.sock;
+ fastcgi_index index.php;
+ include fastcgi_params;
+ fastcgi_param SCRIPT_FILENAME $request_filename;
+
+ # Variable d'environnement PRODUCTION
+ fastcgi_param APP_ENV "production";
+ fastcgi_param SERVER_NAME "app3.geosector.fr";
+
+ # Headers transmis à PHP (viennent du reverse proxy)
+ fastcgi_param HTTP_X_REAL_IP $http_x_real_ip;
+ fastcgi_param HTTP_X_FORWARDED_FOR $http_x_forwarded_for;
+ fastcgi_param HTTP_X_FORWARDED_PROTO $http_x_forwarded_proto;
+ fastcgi_param HTTPS $http_x_forwarded_proto;
+
+ # Timeouts pour opérations longues (sync, exports)
+ fastcgi_read_timeout 300;
+ fastcgi_send_timeout 300;
+ fastcgi_connect_timeout 60;
+
+ # Buffers optimisés
+ fastcgi_buffer_size 128k;
+ fastcgi_buffers 256 16k;
+ fastcgi_busy_buffers_size 256k;
+ fastcgi_temp_file_write_size 256k;
+ }
+ }
+
+ # =============================================================================
+ # UPLOADS ET MÉDIAS
+ # =============================================================================
+ location /api/uploads/ {
+ alias /var/www/geosector/api/uploads/;
+
+ # Cache pour les médias uploadés
+ expires 7d;
+ add_header Cache-Control "public";
+
+ # Sécurité : empêcher l'exécution de scripts
+ location ~ \.(php|phtml|php3|php4|php5|phps)$ {
+ deny all;
+ }
+ }
+
+ # =============================================================================
+ # SÉCURITÉ
+ # =============================================================================
+
+ # Bloquer l'accès aux fichiers sensibles
+ location ~ /\.(?!well-known) {
+ deny all;
+ access_log off;
+ log_not_found off;
+ }
+
+ # Bloquer l'accès aux fichiers de configuration
+ location ~* \.(env|sql|bak|backup|swp|config|conf|ini|log)$ {
+ deny all;
+ access_log off;
+ log_not_found off;
+ }
+
+ # Protection contre les requêtes invalides
+ if ($request_method !~ ^(GET|HEAD|POST|PUT|DELETE|PATCH|OPTIONS)$) {
+ return 405;
+ }
+
+ # =============================================================================
+ # MONITORING
+ # =============================================================================
+
+ # Endpoint de health check (accessible en interne)
+ location = /nginx-health {
+ access_log off;
+ allow 127.0.0.1;
+ allow 13.23.34.0/24; # Réseau interne Incus
+ deny all;
+ return 200 "healthy\n";
+ add_header Content-Type text/plain;
+ }
+}
diff --git a/api/config/nginx/pra-geo-production.conf b/api/config/nginx/pra-geo-production.conf
new file mode 100644
index 00000000..2a9504a5
--- /dev/null
+++ b/api/config/nginx/pra-geo-production.conf
@@ -0,0 +1,290 @@
+# =============================================================================
+# Configuration NGINX PRODUCTION pour pra-geo (IN4)
+# Date: 2025-10-07
+# Environnement: PRODUCTION
+# Server: IN4 (51.159.7.190)
+# =============================================================================
+
+# Site principal (redirection vers www ou app)
+server {
+ listen 80;
+ server_name geosector.fr;
+
+ # Redirection permanente vers HTTPS
+ return 301 https://$server_name$request_uri;
+}
+
+server {
+ listen 443 ssl http2;
+ server_name geosector.fr;
+
+ # Certificats SSL
+ ssl_certificate /etc/letsencrypt/live/geosector.fr/fullchain.pem;
+ ssl_certificate_key /etc/letsencrypt/live/geosector.fr/privkey.pem;
+
+ # Configuration SSL optimisée
+ ssl_protocols TLSv1.2 TLSv1.3;
+ ssl_ciphers 'ECDHE-RSA-AES128-GCM-SHA256:ECDHE-RSA-AES256-GCM-SHA384:ECDHE-RSA-CHACHA20-POLY1305';
+ ssl_prefer_server_ciphers on;
+ ssl_session_cache shared:SSL:10m;
+ ssl_session_timeout 10m;
+ ssl_stapling on;
+ ssl_stapling_verify on;
+
+ root /var/www/geosector/web;
+ index index.html;
+
+ # Logs PRODUCTION
+ access_log /var/log/nginx/geosector-web_access.log combined;
+ error_log /var/log/nginx/geosector-web_error.log warn;
+
+ # Headers de sécurité
+ add_header Strict-Transport-Security "max-age=31536000; includeSubDomains; preload" always;
+ add_header X-Frame-Options "SAMEORIGIN" always;
+ add_header X-Content-Type-Options "nosniff" always;
+ add_header X-XSS-Protection "1; mode=block" always;
+ add_header Referrer-Policy "strict-origin-when-cross-origin" always;
+
+ location / {
+ try_files $uri $uri/ /index.html;
+ }
+
+ # Assets statiques avec cache agressif
+ location ~* \.(jpg|jpeg|png|gif|ico|svg|webp|css|js|woff|woff2|ttf|eot)$ {
+ expires 1y;
+ add_header Cache-Control "public, immutable";
+ access_log off;
+ }
+
+ # Protection des fichiers sensibles
+ location ~ /\.(?!well-known) {
+ deny all;
+ access_log off;
+ log_not_found off;
+ }
+}
+
+# =============================================================================
+# APPLICATION FLUTTER + API PHP
+# =============================================================================
+
+# Redirection HTTP → HTTPS
+server {
+ listen 80;
+ server_name app3.geosector.fr;
+
+ # Permettre Let's Encrypt validation
+ location ^~ /.well-known/acme-challenge/ {
+ root /var/www/letsencrypt;
+ allow all;
+ }
+
+ # Redirection permanente vers HTTPS
+ return 301 https://$server_name$request_uri;
+}
+
+server {
+ listen 443 ssl http2;
+ server_name app3.geosector.fr;
+
+ # Certificats SSL
+ ssl_certificate /etc/letsencrypt/live/app3.geosector.fr/fullchain.pem;
+ ssl_certificate_key /etc/letsencrypt/live/app3.geosector.fr/privkey.pem;
+
+ # Configuration SSL optimisée (même que ci-dessus)
+ ssl_protocols TLSv1.2 TLSv1.3;
+ ssl_ciphers 'ECDHE-RSA-AES128-GCM-SHA256:ECDHE-RSA-AES256-GCM-SHA384:ECDHE-RSA-CHACHA20-POLY1305';
+ ssl_prefer_server_ciphers on;
+ ssl_session_cache shared:SSL:10m;
+ ssl_session_timeout 10m;
+ ssl_stapling on;
+ ssl_stapling_verify on;
+
+ # Logs PRODUCTION
+ access_log /var/log/nginx/pra-app_access.log combined;
+ error_log /var/log/nginx/pra-app_error.log warn;
+
+ # Headers de sécurité globaux
+ add_header Strict-Transport-Security "max-age=31536000; includeSubDomains; preload" always;
+ add_header X-Frame-Options "SAMEORIGIN" always;
+ add_header X-Content-Type-Options "nosniff" always;
+ add_header X-XSS-Protection "1; mode=block" always;
+ add_header Referrer-Policy "strict-origin-when-cross-origin" always;
+
+ # Taille maximale des uploads (pour les logos, exports, etc.)
+ client_max_body_size 10M;
+ client_body_buffer_size 128k;
+
+ # Timeouts optimisés pour PRODUCTION
+ client_body_timeout 30s;
+ client_header_timeout 30s;
+ send_timeout 60s;
+
+ # =============================================================================
+ # APPLICATION FLUTTER (contenu statique)
+ # =============================================================================
+ location / {
+ root /var/www/geosector/app;
+ index index.html;
+ try_files $uri $uri/ /index.html;
+
+ # Cache intelligent pour PRODUCTION
+ # HTML : pas de cache (pour déploiements)
+ location ~* \.html$ {
+ expires -1;
+ add_header Cache-Control "no-cache, no-store, must-revalidate";
+ }
+
+ # Assets Flutter (JS, CSS, fonts) avec hash : cache agressif
+ location ~* \.(js|css|woff|woff2|ttf|eot)$ {
+ expires 1y;
+ add_header Cache-Control "public, immutable";
+ access_log off;
+ }
+
+ # Images : cache longue durée
+ location ~* \.(jpg|jpeg|png|gif|ico|svg|webp)$ {
+ expires 30d;
+ add_header Cache-Control "public";
+ access_log off;
+ }
+ }
+
+ # =============================================================================
+ # API PHP (RESTful)
+ # =============================================================================
+ location /api/ {
+ root /var/www/geosector;
+
+ # CORS - Liste blanche des origines autorisées en PRODUCTION
+ set $cors_origin "";
+
+ # Autoriser uniquement les domaines de production
+ if ($http_origin ~* ^https://(app\.geosector\.fr|geosector\.fr)$) {
+ set $cors_origin $http_origin;
+ }
+
+ # Gestion des preflight requests (OPTIONS)
+ if ($request_method = 'OPTIONS') {
+ add_header 'Access-Control-Allow-Origin' $cors_origin always;
+ add_header 'Access-Control-Allow-Methods' 'GET, POST, PUT, DELETE, PATCH, OPTIONS' always;
+ add_header 'Access-Control-Allow-Headers' 'DNT,User-Agent,X-Requested-With,If-Modified-Since,Cache-Control,Content-Type,Range,Authorization' always;
+ add_header 'Access-Control-Allow-Credentials' 'true' always;
+ add_header 'Access-Control-Max-Age' 86400;
+ add_header 'Content-Type' 'text/plain; charset=utf-8';
+ add_header 'Content-Length' 0;
+ return 204;
+ }
+
+ # Headers CORS pour les requêtes normales
+ add_header 'Access-Control-Allow-Origin' $cors_origin always;
+ add_header 'Access-Control-Allow-Methods' 'GET, POST, PUT, DELETE, PATCH, OPTIONS' always;
+ add_header 'Access-Control-Allow-Headers' 'DNT,X-CustomHeader,Keep-Alive,User-Agent,X-Requested-With,If-Modified-Since,Cache-Control,Content-Type,Authorization' always;
+ add_header 'Access-Control-Allow-Credentials' 'true' always;
+
+ # Cache API : pas de cache (données dynamiques)
+ add_header Cache-Control "no-store, no-cache, must-revalidate, proxy-revalidate" always;
+ add_header Pragma "no-cache" always;
+ add_header Expires "0" always;
+
+ # Headers de sécurité spécifiques API
+ add_header X-Content-Type-Options "nosniff" always;
+ add_header X-Frame-Options "DENY" always;
+
+ # Rewrite vers index.php
+ try_files $uri $uri/ /api/index.php$is_args$args;
+
+ # Traitement PHP
+ location ~ ^/api/(.+\.php)$ {
+ root /var/www/geosector;
+
+ # FastCGI PHP-FPM
+ fastcgi_pass unix:/run/php-fpm83/php-fpm.sock;
+ fastcgi_index index.php;
+ include fastcgi_params;
+ fastcgi_param SCRIPT_FILENAME $request_filename;
+
+ # Variable d'environnement PRODUCTION
+ fastcgi_param APP_ENV "production";
+ fastcgi_param SERVER_NAME "app3.geosector.fr";
+
+ # Headers transmis à PHP
+ fastcgi_param HTTP_X_REAL_IP $remote_addr;
+ fastcgi_param HTTP_X_FORWARDED_FOR $proxy_add_x_forwarded_for;
+ fastcgi_param HTTP_X_FORWARDED_PROTO $scheme;
+
+ # Timeouts pour opérations longues (sync, exports)
+ fastcgi_read_timeout 300;
+ fastcgi_send_timeout 300;
+ fastcgi_connect_timeout 60;
+
+ # Buffers optimisés
+ fastcgi_buffer_size 128k;
+ fastcgi_buffers 256 16k;
+ fastcgi_busy_buffers_size 256k;
+ fastcgi_temp_file_write_size 256k;
+
+ # Headers CORS pour réponses PHP
+ add_header 'Access-Control-Allow-Origin' $cors_origin always;
+ add_header 'Access-Control-Allow-Credentials' 'true' always;
+ }
+ }
+
+ # =============================================================================
+ # UPLOADS ET MÉDIAS
+ # =============================================================================
+ location /api/uploads/ {
+ alias /var/www/geosector/api/uploads/;
+
+ # Cache pour les médias uploadés
+ expires 7d;
+ add_header Cache-Control "public";
+
+ # Sécurité : empêcher l'exécution de scripts
+ location ~ \.(php|phtml|php3|php4|php5|phps)$ {
+ deny all;
+ }
+ }
+
+ # =============================================================================
+ # SÉCURITÉ
+ # =============================================================================
+
+ # Bloquer l'accès aux fichiers sensibles
+ location ~ /\.(?!well-known) {
+ deny all;
+ access_log off;
+ log_not_found off;
+ }
+
+ # Bloquer l'accès aux fichiers de configuration
+ location ~* \.(env|sql|bak|backup|swp|config|conf|ini|log)$ {
+ deny all;
+ access_log off;
+ log_not_found off;
+ }
+
+ # Bloquer les user-agents malveillants
+ if ($http_user_agent ~* (bot|crawler|spider|scraper|wget|curl)) {
+ return 403;
+ }
+
+ # Protection contre les requêtes invalides
+ if ($request_method !~ ^(GET|HEAD|POST|PUT|DELETE|PATCH|OPTIONS)$) {
+ return 405;
+ }
+
+ # =============================================================================
+ # MONITORING
+ # =============================================================================
+
+ # Endpoint de health check (accessible uniquement en local)
+ location = /nginx-health {
+ access_log off;
+ allow 127.0.0.1;
+ allow 13.23.34.0/24; # Réseau interne Incus
+ deny all;
+ return 200 "healthy\n";
+ add_header Content-Type text/plain;
+ }
+}
diff --git a/api/deploy-api.sh b/api/deploy-api.sh
index 45c11a01..deffd8a9 100755
--- a/api/deploy-api.sh
+++ b/api/deploy-api.sh
@@ -75,8 +75,8 @@ cleanup_old_backups() {
"pra") prefix="api-pra-" ;;
esac
- echo_info "Cleaning old backups (keeping last 10)..."
- ls -t "${BACKUP_DIR}"/${prefix}*.tar.gz 2>/dev/null | tail -n +11 | xargs -r rm -f && {
+ echo_info "Cleaning old backups (keeping last 5)..."
+ ls -t "${BACKUP_DIR}"/${prefix}*.tar.gz 2>/dev/null | tail -n +6 | xargs -r rm -f && {
REMAINING_BACKUPS=$(ls "${BACKUP_DIR}"/${prefix}*.tar.gz 2>/dev/null | wc -l)
echo_info "Kept ${REMAINING_BACKUPS} backup(s) for ${TARGET_ENV}"
}
@@ -164,9 +164,12 @@ if [ "$SOURCE_TYPE" = "local_code" ]; then
--exclude='.gitignore' \
--exclude='.vscode' \
--exclude='logs' \
+ --exclude='sessions' \
+ --exclude='opendata' \
--exclude='*.template' \
--exclude='*.sh' \
--exclude='.env' \
+ --exclude='.env_marker' \
--exclude='*.log' \
--exclude='.DS_Store' \
--exclude='README.md' \
@@ -193,6 +196,8 @@ elif [ "$SOURCE_TYPE" = "remote_container" ]; then
incus exec ${SOURCE_CONTAINER} -- tar \
--exclude='logs' \
--exclude='uploads' \
+ --exclude='sessions' \
+ --exclude='opendata' \
-czf /tmp/${ARCHIVE_NAME} -C ${API_PATH} .
" || echo_error "Failed to create archive on remote"
@@ -254,18 +259,29 @@ if [ "$DEST_HOST" != "local" ]; then
# Déployer sur le container de destination
echo_info "Extracting on destination container..."
+
+ # Déterminer le nom de l'environnement pour le marqueur
+ case $TARGET_ENV in
+ "dev") ENV_MARKER="development" ;;
+ "rca") ENV_MARKER="recette" ;;
+ "pra") ENV_MARKER="production" ;;
+ esac
+
ssh -i ${HOST_KEY} -p ${HOST_PORT} ${HOST_USER}@${DEST_HOST} "
set -euo pipefail
-
+
# Pousser l'archive dans le container
incus project switch ${INCUS_PROJECT} &&
incus file push /tmp/${ARCHIVE_NAME} ${DEST_CONTAINER}/tmp/${ARCHIVE_NAME} &&
-
- # Nettoyer sélectivement (préserver logs et uploads)
- incus exec ${DEST_CONTAINER} -- find ${API_PATH} -mindepth 1 -maxdepth 1 ! -name 'uploads' ! -name 'logs' -exec rm -rf {} \; 2>/dev/null || true &&
-
+
+ # Nettoyer sélectivement (préserver logs, uploads et sessions)
+ incus exec ${DEST_CONTAINER} -- find ${API_PATH} -mindepth 1 -maxdepth 1 ! -name 'uploads' ! -name 'logs' ! -name 'sessions' -exec rm -rf {} \; 2>/dev/null || true &&
+
# Extraire l'archive
incus exec ${DEST_CONTAINER} -- tar -xzf /tmp/${ARCHIVE_NAME} -C ${API_PATH}/ &&
+
+ # Créer le marqueur d'environnement pour la détection CLI
+ incus exec ${DEST_CONTAINER} -- bash -c 'echo \"${ENV_MARKER}\" > ${API_PATH}/.env_marker' &&
# Permissions
incus exec ${DEST_CONTAINER} -- chown -R ${FINAL_OWNER}:${FINAL_GROUP} ${API_PATH} &&
@@ -273,24 +289,74 @@ if [ "$DEST_HOST" != "local" ]; then
incus exec ${DEST_CONTAINER} -- find ${API_PATH} -type f -exec chmod 644 {} \; &&
# Permissions spéciales pour logs
- incus exec ${DEST_CONTAINER} -- test -d ${API_PATH}/logs &&
- incus exec ${DEST_CONTAINER} -- chown -R ${FINAL_OWNER_LOGS}:${FINAL_GROUP_LOGS} ${API_PATH}/logs &&
- incus exec ${DEST_CONTAINER} -- chmod -R 755 ${API_PATH}/logs || true &&
+ incus exec ${DEST_CONTAINER} -- mkdir -p ${API_PATH}/logs/events &&
+ incus exec ${DEST_CONTAINER} -- chown -R ${FINAL_OWNER_LOGS}:${FINAL_GROUP} ${API_PATH}/logs &&
+ incus exec ${DEST_CONTAINER} -- find ${API_PATH}/logs -type d -exec chmod 750 {} \; &&
+ incus exec ${DEST_CONTAINER} -- find ${API_PATH}/logs -type f -exec chmod 640 {} \; &&
# Permissions spéciales pour uploads
- incus exec ${DEST_CONTAINER} -- test -d ${API_PATH}/uploads &&
- incus exec ${DEST_CONTAINER} -- chown -R ${FINAL_OWNER_LOGS}:${FINAL_GROUP_LOGS} ${API_PATH}/uploads &&
- incus exec ${DEST_CONTAINER} -- chmod -R 755 ${API_PATH}/uploads || true &&
-
- # Composer
- incus exec ${DEST_CONTAINER} -- bash -c 'cd ${API_PATH} && composer install --no-dev --optimize-autoloader' || echo 'Composer install skipped' &&
+ incus exec ${DEST_CONTAINER} -- mkdir -p ${API_PATH}/uploads &&
+ incus exec ${DEST_CONTAINER} -- chown -R ${FINAL_OWNER_LOGS}:${FINAL_GROUP} ${API_PATH}/uploads &&
+ incus exec ${DEST_CONTAINER} -- find ${API_PATH}/uploads -type d -exec chmod 750 {} \; &&
+ incus exec ${DEST_CONTAINER} -- find ${API_PATH}/uploads -type f -exec chmod 640 {} \; &&
+
+ # Permissions spéciales pour sessions
+ incus exec ${DEST_CONTAINER} -- mkdir -p ${API_PATH}/sessions &&
+ incus exec ${DEST_CONTAINER} -- chown -R ${FINAL_OWNER_LOGS}:${FINAL_GROUP} ${API_PATH}/sessions &&
+ incus exec ${DEST_CONTAINER} -- chmod 700 ${API_PATH}/sessions &&
+
+ # Composer (installation stricte - échec bloquant)
+ incus exec ${DEST_CONTAINER} -- bash -c 'cd ${API_PATH} && composer install --no-dev --optimize-autoloader' || { echo 'ERROR: Composer install failed'; exit 1; } &&
# Nettoyage
incus exec ${DEST_CONTAINER} -- rm -f /tmp/${ARCHIVE_NAME} &&
rm -f /tmp/${ARCHIVE_NAME}
" || echo_error "Deployment failed on destination"
-
+
echo_info "Remote backup saved: ${REMOTE_BACKUP_DIR} on ${DEST_CONTAINER}"
+
+ # Nettoyage des anciens backups sur le container distant
+ echo_info "Cleaning old backup directories on ${DEST_CONTAINER}..."
+ ssh -i ${HOST_KEY} -p ${HOST_PORT} ${HOST_USER}@${DEST_HOST} "
+ incus exec ${DEST_CONTAINER} -- bash -c 'rm -rf ${API_PATH}_backup_*'
+ " && echo_info "Old backups cleaned" || echo_warning "Could not clean old backups"
+
+ # =====================================
+ # Configuration des tâches CRON
+ # =====================================
+
+ echo_step "Configuring CRON tasks..."
+
+ ssh -i ${HOST_KEY} -p ${HOST_PORT} ${HOST_USER}@${DEST_HOST} "
+ incus exec ${DEST_CONTAINER} -- bash <<'EOFCRON'
+ # Sauvegarder les crons existants (hors geosector)
+ crontab -l 2>/dev/null | grep -v 'geosector/api/scripts/cron' > /tmp/crontab_backup || true
+
+ # Créer le nouveau crontab avec les tâches CRON pour l'API
+ cat /tmp/crontab_backup > /tmp/new_crontab
+ cat >> /tmp/new_crontab <<'EOF'
+
+# GEOSECTOR API - Email queue processing (every 5 minutes)
+*/5 * * * * /usr/bin/php /var/www/geosector/api/scripts/cron/process_email_queue.php >> /var/www/geosector/api/logs/email_queue.log 2>&1
+
+# GEOSECTOR API - Security data cleanup (daily at 2am)
+0 2 * * * /usr/bin/php /var/www/geosector/api/scripts/cron/cleanup_security_data.php >> /var/www/geosector/api/logs/cleanup_security.log 2>&1
+
+# GEOSECTOR API - Stripe devices update (weekly Sunday at 3am)
+0 3 * * 0 /usr/bin/php /var/www/geosector/api/scripts/cron/update_stripe_devices.php >> /var/www/geosector/api/logs/stripe_devices.log 2>&1
+EOF
+
+ # Installer le nouveau crontab
+ crontab /tmp/new_crontab
+
+ # Nettoyer
+ rm -f /tmp/crontab_backup /tmp/new_crontab
+
+ # Afficher les crons installés
+ echo 'CRON tasks installed:'
+ crontab -l | grep geosector
+EOFCRON
+ " && echo_info "CRON tasks configured successfully" || echo_warning "CRON configuration failed"
fi
# L'archive reste dans le dossier de backup, pas de nettoyage nécessaire
diff --git a/api/docs/EVENTS-LOG.md b/api/docs/EVENTS-LOG.md
new file mode 100644
index 00000000..e209da57
--- /dev/null
+++ b/api/docs/EVENTS-LOG.md
@@ -0,0 +1,495 @@
+# Système de logs d'événements JSONL
+
+## 📋 Vue d'ensemble
+
+Système de traçabilité des événements métier pour statistiques et audit, stocké en fichiers JSONL (JSON Lines) sans impact sur la base de données principale.
+
+**Créé le :** 26 Octobre 2025
+**Rétention :** 15 mois
+**Format :** JSONL (une ligne = un événement JSON)
+
+## 🎯 Objectifs
+
+### Événements tracés
+
+**Authentification**
+- Connexions réussies (user_id, entity_id, plateforme, IP)
+- Tentatives échouées (username, raison, IP, nb tentatives)
+
+**CRUD métier**
+- **Passages** : création, modification, suppression
+- **Secteurs** : création, modification, suppression
+- **Membres** : création, modification, suppression
+- **Entités** : création, modification, suppression
+
+### Cas d'usage
+
+**1. Admin entité**
+- Stats de son entité : connexions, passages, secteurs sur 1 jour/semaine/mois
+- Activité des membres de l'entité
+
+**2. Super-admin**
+- Stats globales : tous les passages modifiés sur 2 semaines
+- Événements toutes entités sur période donnée
+- Détection d'anomalies
+
+## 📁 Architecture de stockage
+
+### Structure des répertoires
+
+```
+/logs/events/
+ ├── 2025-10-26.jsonl # Fichier du jour (écriture append)
+ ├── 2025-10-25.jsonl
+ ├── 2025-10-24.jsonl
+ ├── 2025-09-30.jsonl
+ ├── 2025-09-29.jsonl.gz # Compression auto après 30 jours
+ └── archive/
+ ├── 2025-09.jsonl.gz # Archive mensuelle
+ ├── 2025-08.jsonl.gz
+ └── 2024-07.jsonl.gz # Supprimé auto après 15 mois
+```
+
+### Cycle de vie des fichiers
+
+| Âge | État | Taille estimée | Accès |
+|-----|------|----------------|-------|
+| 0-30 jours | `.jsonl` non compressé | 1-10 MB/jour | Lecture directe rapide |
+| 30 jours-15 mois | `.jsonl.gz` compressé | ~100 KB/jour | Décompression à la volée |
+| > 15 mois | Supprimé automatiquement | - | - |
+
+### Rotation et rétention
+
+**CRON mensuel** : `scripts/cron/rotate_event_logs.php`
+- **Fréquence** : 1er du mois à 3h00
+- **Actions** :
+ 1. Compresser les fichiers `.jsonl` de plus de 30 jours en `.jsonl.gz`
+ 2. Supprimer les fichiers `.jsonl.gz` de plus de 15 mois
+ 3. Logger le résumé de rotation
+
+**Commande manuelle** :
+```bash
+php scripts/cron/rotate_event_logs.php
+```
+
+## 📊 Format des événements
+
+### Structure commune
+
+Tous les événements partagent ces champs :
+
+```json
+{
+ "timestamp": "2025-10-26T14:32:15Z", // ISO 8601 UTC
+ "event": "nom_evenement", // Type d'événement
+ "user_id": 123, // ID utilisateur (si authentifié)
+ "entity_id": 5, // ID entité (si applicable)
+ "ip": "192.168.1.100", // IP client
+ "platform": "ios|android|web", // Plateforme
+ "app_version": "3.3.6" // Version app (mobile uniquement)
+}
+```
+
+### Événements d'authentification
+
+#### Login réussi
+```jsonl
+{"timestamp":"2025-10-26T14:32:15Z","event":"login_success","user_id":123,"entity_id":5,"platform":"ios","app_version":"3.3.6","ip":"192.168.1.100","username":"user123"}
+```
+
+#### Login échoué
+```jsonl
+{"timestamp":"2025-10-26T14:35:22Z","event":"login_failed","username":"test","reason":"invalid_password","ip":"192.168.1.101","attempt":3,"platform":"web"}
+```
+
+**Raisons possibles** : `invalid_password`, `user_not_found`, `account_inactive`, `blocked_ip`
+
+#### Logout
+```jsonl
+{"timestamp":"2025-10-26T16:45:00Z","event":"logout","user_id":123,"entity_id":5,"platform":"android","session_duration":7800}
+```
+
+### Événements Passages
+
+#### Création
+```jsonl
+{"timestamp":"2025-10-26T14:40:10Z","event":"passage_created","passage_id":45678,"user_id":123,"entity_id":5,"operation_id":789,"sector_id":12,"amount":50.00,"payment_type":"cash","platform":"android"}
+```
+
+#### Modification
+```jsonl
+{"timestamp":"2025-10-26T14:42:05Z","event":"passage_updated","passage_id":45678,"user_id":123,"entity_id":5,"changes":{"amount":{"old":50.00,"new":75.00},"payment_type":{"old":"cash","new":"stripe"}},"platform":"ios"}
+```
+
+#### Suppression
+```jsonl
+{"timestamp":"2025-10-26T14:45:30Z","event":"passage_deleted","passage_id":45678,"user_id":123,"entity_id":5,"operation_id":789,"deleted_by":123,"soft_delete":true,"platform":"web"}
+```
+
+### Événements Secteurs
+
+#### Création
+```jsonl
+{"timestamp":"2025-10-26T15:10:00Z","event":"sector_created","sector_id":456,"operation_id":789,"entity_id":5,"user_id":123,"sector_name":"Secteur A","platform":"web"}
+```
+
+#### Modification
+```jsonl
+{"timestamp":"2025-10-26T15:12:00Z","event":"sector_updated","sector_id":456,"operation_id":789,"entity_id":5,"user_id":123,"changes":{"sector_name":{"old":"Secteur A","new":"Secteur Alpha"}},"platform":"web"}
+```
+
+#### Suppression
+```jsonl
+{"timestamp":"2025-10-26T15:15:00Z","event":"sector_deleted","sector_id":456,"operation_id":789,"entity_id":5,"user_id":123,"deleted_by":123,"soft_delete":true,"platform":"web"}
+```
+
+### Événements Membres (Users)
+
+#### Création
+```jsonl
+{"timestamp":"2025-10-26T15:20:00Z","event":"user_created","new_user_id":789,"entity_id":5,"created_by":123,"role_id":1,"username":"newuser","platform":"web"}
+```
+
+#### Modification
+```jsonl
+{"timestamp":"2025-10-26T15:25:00Z","event":"user_updated","user_id":789,"entity_id":5,"updated_by":123,"changes":{"role_id":{"old":1,"new":2},"encrypted_phone":true},"platform":"web"}
+```
+
+**Note** : Les champs chiffrés sont indiqués par un booléen `true` sans exposer les valeurs
+
+#### Suppression
+```jsonl
+{"timestamp":"2025-10-26T15:30:00Z","event":"user_deleted","user_id":789,"entity_id":5,"deleted_by":123,"soft_delete":true,"platform":"web"}
+```
+
+### Événements Entités
+
+#### Création
+```jsonl
+{"timestamp":"2025-10-26T15:35:00Z","event":"entity_created","entity_id":25,"created_by":1,"entity_type_id":1,"postal_code":"75001","platform":"web"}
+```
+
+#### Modification
+```jsonl
+{"timestamp":"2025-10-26T15:40:00Z","event":"entity_updated","entity_id":25,"user_id":123,"updated_by":123,"changes":{"encrypted_name":true,"encrypted_email":true,"chk_stripe":{"old":0,"new":1}},"platform":"web"}
+```
+
+#### Suppression (rare)
+```jsonl
+{"timestamp":"2025-10-26T15:45:00Z","event":"entity_deleted","entity_id":25,"deleted_by":1,"soft_delete":true,"reason":"duplicate","platform":"web"}
+```
+
+### Événements Opérations
+
+#### Création
+```jsonl
+{"timestamp":"2025-10-26T16:00:00Z","event":"operation_created","operation_id":999,"entity_id":5,"created_by":123,"date_start":"2025-11-01","date_end":"2025-11-30","platform":"web"}
+```
+
+#### Modification
+```jsonl
+{"timestamp":"2025-10-26T16:05:00Z","event":"operation_updated","operation_id":999,"entity_id":5,"updated_by":123,"changes":{"date_end":{"old":"2025-11-30","new":"2025-12-15"},"chk_active":{"old":0,"new":1}},"platform":"web"}
+```
+
+#### Suppression
+```jsonl
+{"timestamp":"2025-10-26T16:10:00Z","event":"operation_deleted","operation_id":999,"entity_id":5,"deleted_by":123,"soft_delete":true,"platform":"web"}
+```
+
+## 🛠️ Implémentation
+
+### Service EventLogService.php
+
+**Emplacement** : `src/Services/EventLogService.php`
+
+**Méthodes publiques** :
+```php
+EventLogService::logLoginSuccess($userId, $entityId, $username)
+EventLogService::logLoginFailed($username, $reason, $attempt)
+EventLogService::logLogout($userId, $entityId, $sessionDuration)
+
+EventLogService::logPassageCreated($passageId, $operationId, $sectorId, $amount, $paymentType)
+EventLogService::logPassageUpdated($passageId, $changes)
+EventLogService::logPassageDeleted($passageId, $operationId, $softDelete)
+
+EventLogService::logSectorCreated($sectorId, $operationId, $sectorName)
+EventLogService::logSectorUpdated($sectorId, $operationId, $changes)
+EventLogService::logSectorDeleted($sectorId, $operationId, $softDelete)
+
+EventLogService::logUserCreated($newUserId, $entityId, $roleId, $username)
+EventLogService::logUserUpdated($userId, $changes)
+EventLogService::logUserDeleted($userId, $softDelete)
+
+EventLogService::logEntityCreated($entityId, $entityTypeId, $postalCode)
+EventLogService::logEntityUpdated($entityId, $changes)
+EventLogService::logEntityDeleted($entityId, $reason)
+
+EventLogService::logOperationCreated($operationId, $dateStart, $dateEnd)
+EventLogService::logOperationUpdated($operationId, $changes)
+EventLogService::logOperationDeleted($operationId, $softDelete)
+```
+
+**Enrichissement automatique** :
+- `timestamp` : Généré automatiquement (UTC)
+- `user_id`, `entity_id` : Récupérés depuis `Session`
+- `ip` : Récupérée via `ClientDetector`
+- `platform` : Détecté via `ClientDetector` (ios/android/web)
+- `app_version` : Extrait du User-Agent pour mobile
+
+### Intégration dans les Controllers
+
+**Exemple dans PassageController** :
+```php
+public function createPassage(Request $request, Response $response): void {
+ // ... validation et création ...
+
+ $passageId = $db->lastInsertId();
+
+ // Log de l'événement
+ EventLogService::logPassageCreated(
+ $passageId,
+ $data['fk_operation'],
+ $data['fk_sector'],
+ $data['montant'],
+ $data['fk_type_reglement']
+ );
+
+ // ... suite du code ...
+}
+```
+
+### Scripts d'analyse
+
+#### 1. Stats entité
+
+**Fichier** : `scripts/stats/entity_stats.php`
+
+**Usage** :
+```bash
+# Stats entité 5 sur 7 derniers jours
+php scripts/stats/entity_stats.php --entity-id=5 --days=7
+
+# Stats entité 5 entre deux dates
+php scripts/stats/entity_stats.php --entity-id=5 --from=2025-10-01 --to=2025-10-26
+
+# Résultat JSON
+{
+ "entity_id": 5,
+ "period": {"from": "2025-10-20", "to": "2025-10-26"},
+ "stats": {
+ "logins": {"success": 45, "failed": 2},
+ "passages": {"created": 120, "updated": 15, "deleted": 3},
+ "sectors": {"created": 2, "updated": 8, "deleted": 0},
+ "users": {"created": 1, "updated": 5, "deleted": 0}
+ },
+ "top_users": [
+ {"user_id": 123, "actions": 85},
+ {"user_id": 456, "actions": 42}
+ ]
+}
+```
+
+#### 2. Stats globales super-admin
+
+**Fichier** : `scripts/stats/global_stats.php`
+
+**Usage** :
+```bash
+# Tous les passages modifiés sur 2 semaines
+php scripts/stats/global_stats.php --event=passage_updated --days=14
+
+# Toutes les connexions échouées du mois
+php scripts/stats/global_stats.php --event=login_failed --month=2025-10
+
+# Résultat JSON
+{
+ "event": "passage_updated",
+ "period": {"from": "2025-10-13", "to": "2025-10-26"},
+ "total_events": 342,
+ "by_entity": [
+ {"entity_id": 5, "count": 120},
+ {"entity_id": 12, "count": 85},
+ {"entity_id": 18, "count": 67}
+ ],
+ "by_day": {
+ "2025-10-26": 45,
+ "2025-10-25": 38,
+ "2025-10-24": 52
+ }
+}
+```
+
+#### 3. Export CSV pour analyse externe
+
+**Fichier** : `scripts/stats/export_events_csv.php`
+
+**Usage** :
+```bash
+# Exporter toutes les connexions du mois en CSV
+php scripts/stats/export_events_csv.php \
+ --event=login_success \
+ --month=2025-10 \
+ --output=/tmp/logins_october.csv
+```
+
+### CRON de rotation
+
+**Fichier** : `scripts/cron/rotate_event_logs.php`
+
+**Configuration crontab** :
+```cron
+# Rotation des logs d'événements - 1er du mois à 3h
+0 3 1 * * cd /var/www/geosector/api && php scripts/cron/rotate_event_logs.php
+```
+
+**Actions** :
+1. Compresser fichiers > 30 jours : `gzip logs/events/2025-09-*.jsonl`
+2. Supprimer archives > 15 mois : `rm logs/events/*-2024-06-*.jsonl.gz`
+3. Logger résumé dans `logs/rotation.log`
+
+## 📈 Performances et volumétrie
+
+### Estimations
+
+**Volume quotidien moyen** (pour 50 entités actives) :
+- 500 connexions/jour = 500 lignes
+- 2000 passages créés/modifiés = 2000 lignes
+- 100 autres événements = 100 lignes
+- **Total : ~2600 événements/jour**
+
+**Taille fichier** :
+- 1 événement ≈ 200-400 bytes JSON
+- 2600 événements ≈ 0.8-1 MB/jour non compressé
+- Compression gzip : ratio ~10:1 → **~100 KB/jour compressé**
+
+**Rétention 15 mois** :
+- Non compressé (30 jours) : 30 MB
+- Compressé (14.5 mois) : 45 MB
+- **Total stockage : ~75 MB** pour 15 mois
+
+### Optimisation lecture
+
+**Lecture mono-fichier** : < 50ms pour analyser 1 jour (2600 événements)
+
+**Lecture période 7 jours** :
+- 7 fichiers × 1 MB = 7 MB à lire
+- Filtrage `jq` ou PHP : ~200-300ms
+
+**Lecture période 2 semaines (super-admin)** :
+- 14 fichiers × 1 MB = 14 MB à lire
+- Filtrage sur type événement : ~500ms
+
+**Lecture archive compressée** :
+- Décompression à la volée : +100-200ms
+- Total : ~700-800ms pour 1 mois compressé
+
+## 🔒 Sécurité et confidentialité
+
+### Données sensibles
+
+**❌ Jamais loggé en clair** :
+- Mots de passe
+- Contenu chiffré (noms, emails, téléphones, IBAN)
+- Tokens d'authentification
+
+**✅ Loggé** :
+- IDs (user_id, entity_id, passage_id, etc.)
+- Montants financiers
+- Dates et timestamps
+- Types de modifications (indicateur booléen pour champs chiffrés)
+
+### Exemple champ chiffré
+```json
+{
+ "event": "user_updated",
+ "changes": {
+ "encrypted_name": true, // Indique modification sans valeur
+ "encrypted_email": true,
+ "role_id": {"old": 1, "new": 2} // Champ non sensible = valeurs OK
+ }
+}
+```
+
+### Permissions d'accès
+
+**Fichiers logs** :
+- Propriétaire : `nginx:nginx`
+- Permissions : `0640` (lecture nginx, écriture nginx, aucun autre)
+- Dossier `/logs/events/` : `0750`
+
+**Scripts d'analyse** :
+- Exécution : root ou nginx uniquement
+- Pas d'accès direct via endpoints API (pour l'instant)
+
+## 🚀 Roadmap et évolutions futures
+
+### Phase 1 - MVP (actuel) ✅
+- [x] Architecture JSONL quotidienne
+- [x] Service EventLogService.php
+- [x] Intégration dans controllers (LoginController, PassageController, UserController, SectorController, OperationController, EntiteController)
+- [ ] CRON de rotation 15 mois
+- [ ] Scripts d'analyse de base
+
+### Phase 2 - Dashboards (Q1 2026)
+- [ ] Endpoints API : `GET /api/stats/entity/{id}`, `GET /api/stats/global`
+- [ ] Interface web admin : graphiques connexions, passages
+- [ ] Filtres avancés (période, plateforme, utilisateur)
+
+### Phase 3 - Alertes (Q2 2026)
+- [ ] Détection anomalies (pics de connexions échouées)
+- [ ] Alertes email super-admins
+- [ ] Seuils configurables par entité
+
+### Phase 4 - Migration TimescaleDB (si besoin)
+- [ ] Évaluation volume : si > 50k événements/jour
+- [ ] Import JSONL → TimescaleDB
+- [ ] Rétention hybride : 90j TimescaleDB, archives JSONL
+
+## 📝 Statut implémentation
+
+**Date : 28 Octobre 2025**
+
+### ✅ Terminé
+- Service `EventLogService.php` créé avec toutes les méthodes de logging
+- Intégration complète dans les 6 controllers principaux :
+ - **LoginController** : login réussi/échoué, logout
+ - **PassageController** : création, modification, suppression passages
+ - **UserController** : création, modification, suppression utilisateurs
+ - **SectorController** : création, modification, suppression secteurs
+ - **OperationController** : création, modification, suppression opérations
+ - **EntiteController** : création, modification entités
+- Enrichissement automatique : timestamp UTC, user_id, entity_id, IP, platform, app_version
+- Sécurité : champs sensibles loggés en booléen uniquement (pas de valeurs chiffrées)
+- Script de déploiement `deploy-api.sh` crée automatiquement `/logs/events/` avec permissions 0750
+
+### 🔄 En attente
+- Scripts d'analyse (`entity_stats.php`, `global_stats.php`, `export_events_csv.php`)
+- CRON de rotation 15 mois (`rotate_event_logs.php`)
+- Tests en environnement DEV
+
+## 📝 Checklist déploiement
+
+### Environnement DEV (dva-geo)
+- [x] Créer dossier `/logs/events/` (permissions 0750) - Intégré dans deploy-api.sh
+- [x] Déployer `EventLogService.php`
+- [ ] Déployer scripts stats et rotation
+- [ ] Configurer CRON rotation
+- [ ] Tests : générer événements manuellement
+- [ ] Valider format JSONL et rotation
+
+### Environnement RECETTE (rca-geo)
+- [ ] Déployer depuis DEV validé
+- [ ] Tests de charge : 10k événements/jour
+- [ ] Valider performances scripts d'analyse
+- [ ] Valider compression et suppression auto
+
+### Environnement PRODUCTION (pra-geo)
+- [ ] Déployer depuis RECETTE validée
+- [ ] Monitoring volumétrie
+- [ ] Backups quotidiens `/logs/events/` (via CRON général)
+
+---
+
+**Dernière mise à jour :** 28 Octobre 2025
+**Version :** 1.1
+**Statut :** ✅ Service implémenté et intégré - Scripts d'analyse à développer
diff --git a/api/docs/GESTION-SECTORS.md b/api/docs/GESTION-SECTORS.md
index 6c425f37..0f144a27 100644
--- a/api/docs/GESTION-SECTORS.md
+++ b/api/docs/GESTION-SECTORS.md
@@ -24,24 +24,78 @@ Ce document décrit le système de gestion des secteurs dans l'API Geosector, in
- Contient toutes les tables de l'application
- Tables concernées : `ope_sectors`, `sectors_adresses`, `ope_pass`, `ope_users_sectors`, `x_departements_contours`
-2. **Base adresses** (dans conteneurs Incus séparés)
- - DVA : `dva-maria` (13.23.33.46) - base `adresses`
- - RCA : `rca-maria` (13.23.33.36) - base `adresses`
- - PRA : `pra-maria` (13.23.33.26) - base `adresses`
- - Credentials : `adr_geo_user` / `d66,AdrGeoDev.User`
+2. **Base adresses** (dans conteneurs maria3/maria4)
+ - **DVA** : maria3 (13.23.33.4) - base `adresses`
+ - User : `adr_geo_user` / `d66,AdrGeoDev.User`
+ - **RCA** : maria3 (13.23.33.4) - base `adresses`
+ - User : `adr_geo_user` / `d66,AdrGeoRec.User`
+ - **PROD** : maria4 (13.23.33.4) - base `adresses`
+ - User : `adr_geo_user` / `d66,AdrGeoPrd.User`
- Tables par département : `cp22`, `cp23`, etc.
+3. **Base bâtiments** (dans conteneurs maria3/maria4)
+ - **DVA** : maria3 (13.23.33.4) - base `batiments`
+ - User : `adr_geo_user` / `d66,AdrGeoDev.User`
+ - **RCA** : maria3 (13.23.33.4) - base `batiments`
+ - User : `adr_geo_user` / `d66,AdrGeoRec.User`
+ - **PROD** : maria4 (13.23.33.4) - base `batiments`
+ - User : `adr_geo_user` / `d66,AdrGeoPrd.User`
+ - Tables par département : `bat22`, `bat23`, etc.
+ - Colonnes principales : `batiment_groupe_id`, `cle_interop_adr`, `nb_log`, `nb_niveau`, `residence`, `altitude_sol_mean`
+ - Lien avec adresses : `bat{dept}.cle_interop_adr = cp{dept}.id`
+
### Configuration
Dans `src/Config/AppConfig.php` :
```php
+// DÉVELOPPEMENT
'addresses_database' => [
- 'host' => '13.23.33.46', // Varie selon l'environnement
+ 'host' => '13.23.33.4', // Container maria3 sur IN3
'name' => 'adresses',
'username' => 'adr_geo_user',
'password' => 'd66,AdrGeoDev.User',
],
+
+// RECETTE
+'addresses_database' => [
+ 'host' => '13.23.33.4', // Container maria3 sur IN3
+ 'name' => 'adresses',
+ 'username' => 'adr_geo_user',
+ 'password' => 'd66,AdrGeoRec.User',
+],
+
+// PRODUCTION
+'addresses_database' => [
+ 'host' => '13.23.33.4', // Container maria4 sur IN4
+ 'name' => 'adresses',
+ 'username' => 'adr_geo_user',
+ 'password' => 'd66,AdrGeoPrd.User',
+],
+
+// DÉVELOPPEMENT - Bâtiments
+'buildings_database' => [
+ 'host' => '13.23.33.4', // Container maria3 sur IN3
+ 'name' => 'batiments',
+ 'username' => 'adr_geo_user',
+ 'password' => 'd66,AdrGeoDev.User',
+],
+
+// RECETTE - Bâtiments
+'buildings_database' => [
+ 'host' => '13.23.33.4', // Container maria3 sur IN3
+ 'name' => 'batiments',
+ 'username' => 'adr_geo_user',
+ 'password' => 'd66,AdrGeoRec.User',
+],
+
+// PRODUCTION - Bâtiments
+'buildings_database' => [
+ 'host' => '13.23.33.4', // Container maria4 sur IN4
+ 'name' => 'batiments',
+ 'username' => 'adr_geo_user',
+ 'password' => 'd66,AdrGeoPrd.User',
+],
```
## Gestion des contours départementaux
@@ -100,7 +154,7 @@ Vérifie les limites départementales des secteurs :
class DepartmentBoundaryService {
// Vérifie si un secteur est contenu dans un département
public function checkSectorInDepartment(array $sectorCoordinates, string $departmentCode): array
-
+
// Liste tous les départements touchés par un secteur
public function getDepartmentsForSector(array $sectorCoordinates): array
}
@@ -118,6 +172,46 @@ class DepartmentBoundaryService {
]
```
+### BuildingService
+
+Enrichit les adresses avec les données bâtiments :
+
+```php
+namespace App\Services;
+
+class BuildingService {
+ // Enrichit une liste d'adresses avec les métadonnées des bâtiments
+ public function enrichAddresses(array $addresses): array
+}
+```
+
+**Fonctionnement** :
+- Connexion à la base `batiments` externe
+- Interrogation des tables `bat{dept}` par département
+- JOIN sur `bat{dept}.cle_interop_adr = cp{dept}.id`
+- Ajout des métadonnées : `fk_batiment`, `fk_habitat`, `nb_niveau`, `nb_log`, `residence`, `alt_sol`
+- Fallback : `fk_habitat=1` (maison individuelle) si pas de bâtiment trouvé
+
+**Données retournées** :
+```php
+[
+ 'id' => 'cp22.123456',
+ 'numero' => '10',
+ 'voie' => 'Rue Victor Hugo',
+ 'code_postal' => '22000',
+ 'commune' => 'Saint-Brieuc',
+ 'latitude' => 48.5149,
+ 'longitude' => -2.7658,
+ // Données bâtiment enrichies :
+ 'fk_batiment' => 'BAT_123456', // null si maison
+ 'fk_habitat' => 2, // 1=individuel, 2=collectif
+ 'nb_niveau' => 4, // null si maison
+ 'nb_log' => 12, // null si maison
+ 'residence' => 'Résidence Les Pins', // '' si maison
+ 'alt_sol' => 25.5 // null si maison
+]
+```
+
## Processus de création de secteur
### 1. Structure du payload
@@ -150,13 +244,77 @@ class DepartmentBoundaryService {
- Recherche des passages avec `fk_sector = 0` dans le polygone
- Mise à jour de leur `fk_sector` vers le nouveau secteur
- Exclusion des passages ayant déjà une `fk_adresse`
-7. **Récupération** des adresses via `AddressService`
-8. **Stockage** des adresses dans `sectors_adresses`
-9. **Création** des passages dans `ope_pass` pour chaque adresse :
+7. **Récupération** des adresses via `AddressService::getAddressesInPolygon()`
+8. **Enrichissement** avec données bâtiments via `AddressService::enrichAddressesWithBuildings()`
+9. **Stockage** des adresses dans `sectors_adresses` avec colonnes bâtiment :
+ - `fk_batiment`, `fk_habitat`, `nb_niveau`, `nb_log`, `residence`, `alt_sol`
+10. **Création** des passages dans `ope_pass` :
+ - **Maisons individuelles** (fk_habitat=1) : 1 passage par adresse
+ - **Immeubles** (fk_habitat=2) : nb_log passages par adresse (1 par appartement)
+ - Champs ajoutés : `residence`, `appt` (numéro 1 à nb_log), `fk_habitat`
- Affectés au premier utilisateur de la liste
- Avec toutes les FK nécessaires (entité, opération, secteur, user)
- Données d'adresse complètes
-10. **Commit** de la transaction ou **rollback** en cas d'erreur
+11. **Commit** de la transaction ou **rollback** en cas d'erreur
+
+## Processus de modification de secteur
+
+### 1. Structure du payload UPDATE
+
+```json
+{
+ "libelle": "Secteur Centre-Ville Modifié",
+ "color": "#00FF00",
+ "sector": "48.117266/-1.6777926#48.118500/-1.6750000#...",
+ "users": [12, 34],
+ "chk_adresses_change": 1
+}
+```
+
+### 2. Paramètre chk_adresses_change
+
+**Valeurs** :
+- `0` : Ne pas recalculer les adresses et passages (modification simple)
+- `1` : Recalculer les adresses et passages (défaut)
+
+**Cas d'usage** :
+
+#### chk_adresses_change = 0
+Modification rapide sans toucher aux adresses/passages :
+- ✅ Modification du libellé
+- ✅ Modification de la couleur
+- ✅ Modification des coordonnées du polygone (visuel uniquement)
+- ✅ Modification des membres affectés
+- ❌ Pas de recalcul des adresses dans sectors_adresses
+- ❌ Pas de mise à jour des passages (orphelins, créés, supprimés)
+- ❌ **Réponse sans passages_sector** (tableau vide)
+
+**Utilité** : Permet aux admins de corriger rapidement un libellé, une couleur, ou d'ajuster légèrement le périmètre visuel sans déclencher un recalcul complet qui pourrait prendre plusieurs secondes.
+
+**Réponse API** :
+```json
+{
+ "status": "success",
+ "message": "Secteur modifié avec succès",
+ "sector": { "id": 123, "libelle": "...", "color": "...", "sector": "..." },
+ "passages_sector": [], // Vide car chk_adresses_change = 0
+ "passages_orphaned": 0,
+ "passages_deleted": 0,
+ "passages_updated": 0,
+ "passages_created": 0,
+ "passages_total": 0,
+ "users_sectors": [...]
+}
+```
+
+#### chk_adresses_change = 1 (défaut)
+Modification complète avec recalcul :
+- ✅ Modification du libellé/couleur/polygone
+- ✅ Modification des membres
+- ✅ Suppression et recréation de sectors_adresses
+- ✅ Application des règles de gestion des bâtiments
+- ✅ Mise en orphelin des passages hors périmètre
+- ✅ Création de nouveaux passages pour nouvelles adresses
### 3. Réponse API pour CREATE
@@ -287,14 +445,28 @@ $coordinates = [
### sectors_adresses
- `fk_sector` : Lien vers le secteur
-- `fk_address` : ID de l'adresse dans la base externe
-- `numero`, `voie`, `code_postal`, `commune`
-- `latitude`, `longitude`
+- `fk_adresse` : ID de l'adresse dans la base externe
+- `numero`, `rue`, `rue_bis`, `cp`, `ville`
+- `gps_lat`, `gps_lng`
+- **Colonnes bâtiment** :
+ - `fk_batiment` : ID bâtiment (VARCHAR 50, null si maison)
+ - `fk_habitat` : 1=individuel, 2=collectif (TINYINT UNSIGNED)
+ - `nb_niveau` : Nombre d'étages (INT, null)
+ - `nb_log` : Nombre de logements (INT, null)
+ - `residence` : Nom résidence/copropriété (VARCHAR 75)
+ - `alt_sol` : Altitude sol en mètres (DECIMAL 10,2, null)
### ope_pass (passages)
-- `fk_entite`, `fk_operation`, `fk_sector`, `fk_user`
-- `numero`, `voie`, `code_postal`, `commune`
-- `latitude`, `longitude`
+- `fk_operation`, `fk_sector`, `fk_user`, `fk_adresse`
+- `numero`, `rue`, `rue_bis`, `ville`
+- `gps_lat`, `gps_lng`
+- **Colonnes bâtiment** :
+ - `residence` : Nom résidence (VARCHAR 75)
+ - `appt` : Numéro appartement (VARCHAR 10, saisie libre)
+ - `niveau` : Étage (VARCHAR 10, saisie libre)
+ - `fk_habitat` : 1=individuel, 2=collectif (TINYINT UNSIGNED)
+- `fk_type` : Type passage (2=à faire, autres valeurs pour fait/refus)
+- `encrypted_name`, `encrypted_email`, `encrypted_phone` : Données cryptées
- `created_at`, `fk_user_creat`, `chk_active`
### ope_users_sectors
@@ -303,6 +475,103 @@ $coordinates = [
- `fk_sector` : Lien vers le secteur
- `created_at`, `fk_user_creat`, `chk_active`
+## Règles de gestion des bâtiments lors de l'UPDATE
+
+### Principe général
+
+Lors de la mise à jour d'un secteur, le système applique une logique intelligente pour gérer les passages en fonction du type d'habitat (maison/immeuble) et du nombre de logements.
+
+### Clé d'identification unique
+
+**Tous les passages** sont identifiés par la clé : `numero|rue|rue_bis|ville`
+
+Cette clé ne contient **pas** `residence` ni `appt` car ces champs sont en **saisie libre** par l'utilisateur.
+
+### Cas 1 : Maison individuelle (fk_habitat=1)
+
+#### Si 0 passage existant :
+```
+→ INSERT 1 nouveau passage
+ - fk_habitat = 1
+ - residence = ''
+ - appt = ''
+```
+
+#### Si 1+ passages existants :
+```
+→ UPDATE le premier passage
+ - fk_habitat = 1
+ - residence = ''
+→ Les autres passages restent INTACTS
+ (peuvent correspondre à plusieurs habitants saisis manuellement)
+```
+
+### Cas 2 : Immeuble (fk_habitat=2)
+
+#### Étape 1 : UPDATE systématique
+```
+→ UPDATE TOUS les passages existants à cette adresse
+ - fk_habitat = 2
+ - residence = sectors_adresses.residence (si non vide)
+```
+
+#### Étape 2a : Si nb_existants < nb_log (ex: 3 passages, nb_log=6)
+```
+→ INSERT (nb_log - nb_existants) nouveaux passages
+ - fk_habitat = 2
+ - residence = sectors_adresses.residence
+ - appt = '' (pas de numéro prédéfini)
+ - fk_type = 2 (à faire)
+
+Résultat : 6 passages total (3 conservés + 3 créés)
+```
+
+#### Étape 2b : Si nb_existants > nb_log (ex: 10 passages, nb_log=6)
+```
+→ DELETE max (nb_existants - nb_log) passages
+ Conditions de suppression :
+ - fk_type = 2 (à faire)
+ - ET encrypted_name vide (non visité)
+ - Tri par created_at ASC (les plus anciens d'abord)
+
+Résultat : Entre 6 et 10 passages (selon combien sont visités)
+```
+
+### Points importants
+
+✅ **Préservation des données utilisateur** :
+- `appt` et `niveau` ne sont **JAMAIS modifiés** (saisie libre conservée)
+- Les passages visités (encrypted_name rempli) ne sont **JAMAIS supprimés**
+
+✅ **Mise à jour conditionnelle** :
+- `residence` est mis à jour **uniquement si non vide** dans sectors_adresses
+- Permet de conserver une saisie manuelle si la base bâtiments n'a pas l'info
+
+✅ **Gestion des transitions** :
+- Une adresse peut passer de maison (fk_habitat=1) à immeuble (fk_habitat=2) ou inversement
+- La logique s'adapte automatiquement au nouveau type d'habitat
+
+✅ **Uniformisation GPS** :
+- **Tous les passages d'une même adresse partagent les mêmes coordonnées GPS** (gps_lat, gps_lng)
+- Ces coordonnées proviennent de `sectors_adresses` (enrichies depuis la base externe `adresses`)
+- Cette règle s'applique lors de la **création** et de la **mise à jour** avec `chk_adresses_change=1`
+- Garantit la cohérence géographique pour tous les passages d'un même immeuble
+
+### Exemple concret
+
+**Situation initiale** :
+- Adresse : "10 rue Victor Hugo, 22000 Saint-Brieuc"
+- 8 passages existants (dont 3 visités)
+- nb_log passe de 8 à 5
+
+**Actions** :
+1. UPDATE les 8 passages → fk_habitat=2, residence="Les Chênes"
+2. Tentative suppression de (8-5) = 3 passages
+3. Recherche des passages avec fk_type=2 ET encrypted_name vide
+4. Suppose 5 passages non visités trouvés
+5. Suppression des 3 plus anciens non visités
+6. **Résultat** : 5 passages restants (3 visités + 2 non visités)
+
## Logs et monitoring
Le système génère des logs détaillés pour :
diff --git a/api/docs/STRIPE-BACKEND-MIGRATION.md b/api/docs/STRIPE-BACKEND-MIGRATION.md
new file mode 100644
index 00000000..9973ccae
--- /dev/null
+++ b/api/docs/STRIPE-BACKEND-MIGRATION.md
@@ -0,0 +1,464 @@
+# 🔧 Migration Backend Stripe - Option A (Tout en 1)
+
+## 📋 Objectif
+
+Optimiser la création de compte Stripe Connect en **1 seule requête** côté Flutter qui crée :
+1. Le compte Stripe Connect
+2. La Location Terminal
+3. Le lien d'onboarding
+
+---
+
+## 🗄️ 1. Modification de la base de données
+
+### **Ajouter la colonne `stripe_location_id`**
+
+```sql
+ALTER TABLE amicales
+ADD COLUMN stripe_location_id VARCHAR(255) NULL
+AFTER stripe_id;
+```
+
+**Vérification** :
+```sql
+DESCRIBE amicales;
+```
+
+Doit afficher :
+```
++-------------------+--------------+------+-----+---------+-------+
+| Field | Type | Null | Key | Default | Extra |
++-------------------+--------------+------+-----+---------+-------+
+| stripe_id | varchar(255) | YES | | NULL | |
+| stripe_location_id| varchar(255) | YES | | NULL | |
++-------------------+--------------+------+-----+---------+-------+
+```
+
+---
+
+## 🔧 2. Modification de l'endpoint `POST /stripe/accounts`
+
+### **Fichier** : `app/Http/Controllers/StripeController.php` (ou similaire)
+
+### **Méthode** : `createAccount()` ou `store()`
+
+### **Code proposé** :
+
+```php
+validate([
+ 'fk_entite' => 'required|integer|exists:amicales,id',
+ 'return_url' => 'required|string|url',
+ 'refresh_url' => 'required|string|url',
+ ]);
+
+ $fkEntite = $request->fk_entite;
+ $amicale = Amicale::findOrFail($fkEntite);
+
+ // Vérifier si un compte existe déjà
+ if (!empty($amicale->stripe_id)) {
+ return $this->handleExistingAccount($amicale, $request);
+ }
+
+ DB::beginTransaction();
+
+ try {
+ // Configurer la clé Stripe (selon environnement)
+ \Stripe\Stripe::setApiKey(config('services.stripe.secret'));
+
+ // 1️⃣ Créer le compte Stripe Connect Express
+ $account = \Stripe\Account::create([
+ 'type' => 'express',
+ 'country' => 'FR',
+ 'email' => $amicale->email,
+ 'business_type' => 'non_profit', // ou 'company' selon le cas
+ 'business_profile' => [
+ 'name' => $amicale->name,
+ 'url' => config('app.url'),
+ ],
+ 'capabilities' => [
+ 'card_payments' => ['requested' => true],
+ 'transfers' => ['requested' => true],
+ ],
+ ]);
+
+ \Log::info('Stripe account created', [
+ 'amicale_id' => $amicale->id,
+ 'account_id' => $account->id,
+ ]);
+
+ // 2️⃣ Créer la Location Terminal pour Tap to Pay
+ $location = \Stripe\Terminal\Location::create([
+ 'display_name' => $amicale->name,
+ 'address' => [
+ 'line1' => $amicale->adresse1 ?: 'Non renseigné',
+ 'line2' => $amicale->adresse2,
+ 'city' => $amicale->ville ?: 'Non renseigné',
+ 'postal_code' => $amicale->code_postal ?: '00000',
+ 'country' => 'FR',
+ ],
+ ], [
+ 'stripe_account' => $account->id, // ← Important : Connect account
+ ]);
+
+ \Log::info('Stripe Terminal Location created', [
+ 'amicale_id' => $amicale->id,
+ 'location_id' => $location->id,
+ ]);
+
+ // 3️⃣ Créer le lien d'onboarding
+ $accountLink = \Stripe\AccountLink::create([
+ 'account' => $account->id,
+ 'refresh_url' => $request->refresh_url,
+ 'return_url' => $request->return_url,
+ 'type' => 'account_onboarding',
+ ]);
+
+ \Log::info('Stripe onboarding link created', [
+ 'amicale_id' => $amicale->id,
+ 'account_id' => $account->id,
+ ]);
+
+ // 4️⃣ Sauvegarder TOUT en base de données
+ $amicale->stripe_id = $account->id;
+ $amicale->stripe_location_id = $location->id;
+ $amicale->chk_stripe = true;
+ $amicale->save();
+
+ DB::commit();
+
+ // 5️⃣ Retourner TOUTES les informations
+ return response()->json([
+ 'success' => true,
+ 'account_id' => $account->id,
+ 'location_id' => $location->id,
+ 'onboarding_url' => $accountLink->url,
+ 'charges_enabled' => $account->charges_enabled,
+ 'payouts_enabled' => $account->payouts_enabled,
+ 'existing' => false,
+ 'message' => 'Compte Stripe Connect créé avec succès',
+ ], 201);
+
+ } catch (\Stripe\Exception\ApiErrorException $e) {
+ DB::rollBack();
+
+ \Log::error('Stripe API error', [
+ 'amicale_id' => $amicale->id,
+ 'error' => $e->getMessage(),
+ 'type' => get_class($e),
+ ]);
+
+ return response()->json([
+ 'success' => false,
+ 'message' => 'Erreur Stripe : ' . $e->getMessage(),
+ ], 500);
+
+ } catch (\Exception $e) {
+ DB::rollBack();
+
+ \Log::error('Stripe account creation failed', [
+ 'amicale_id' => $amicale->id,
+ 'error' => $e->getMessage(),
+ 'trace' => $e->getTraceAsString(),
+ ]);
+
+ return response()->json([
+ 'success' => false,
+ 'message' => 'Erreur lors de la création du compte Stripe',
+ ], 500);
+ }
+}
+
+/**
+ * Gérer le cas d'un compte Stripe existant
+ */
+private function handleExistingAccount(Amicale $amicale, Request $request)
+{
+ try {
+ \Stripe\Stripe::setApiKey(config('services.stripe.secret'));
+
+ // Récupérer les infos du compte existant
+ $account = \Stripe\Account::retrieve($amicale->stripe_id);
+
+ // Si pas de location_id, la créer maintenant
+ if (empty($amicale->stripe_location_id)) {
+ $location = \Stripe\Terminal\Location::create([
+ 'display_name' => $amicale->name,
+ 'address' => [
+ 'line1' => $amicale->adresse1 ?: 'Non renseigné',
+ 'city' => $amicale->ville ?: 'Non renseigné',
+ 'postal_code' => $amicale->code_postal ?: '00000',
+ 'country' => 'FR',
+ ],
+ ], [
+ 'stripe_account' => $amicale->stripe_id,
+ ]);
+
+ $amicale->stripe_location_id = $location->id;
+ $amicale->save();
+
+ \Log::info('Location created for existing account', [
+ 'amicale_id' => $amicale->id,
+ 'location_id' => $location->id,
+ ]);
+ }
+
+ // Si le compte est déjà complètement configuré
+ if ($account->charges_enabled && $account->payouts_enabled) {
+ return response()->json([
+ 'success' => true,
+ 'account_id' => $amicale->stripe_id,
+ 'location_id' => $amicale->stripe_location_id,
+ 'onboarding_url' => null,
+ 'charges_enabled' => true,
+ 'payouts_enabled' => true,
+ 'existing' => true,
+ 'message' => 'Compte Stripe déjà configuré et actif',
+ ]);
+ }
+
+ // Compte existant mais configuration incomplète : générer un nouveau lien
+ $accountLink = \Stripe\AccountLink::create([
+ 'account' => $amicale->stripe_id,
+ 'refresh_url' => $request->refresh_url,
+ 'return_url' => $request->return_url,
+ 'type' => 'account_onboarding',
+ ]);
+
+ return response()->json([
+ 'success' => true,
+ 'account_id' => $amicale->stripe_id,
+ 'location_id' => $amicale->stripe_location_id,
+ 'onboarding_url' => $accountLink->url,
+ 'charges_enabled' => $account->charges_enabled,
+ 'payouts_enabled' => $account->payouts_enabled,
+ 'existing' => true,
+ 'message' => 'Compte existant, configuration à finaliser',
+ ]);
+
+ } catch (\Exception $e) {
+ \Log::error('Error handling existing account', [
+ 'amicale_id' => $amicale->id,
+ 'error' => $e->getMessage(),
+ ]);
+
+ return response()->json([
+ 'success' => false,
+ 'message' => 'Erreur lors de la vérification du compte existant',
+ ], 500);
+ }
+}
+```
+
+---
+
+## 📡 3. Modification de l'endpoint `GET /stripe/accounts/{id}/status`
+
+Ajouter `location_id` dans la réponse :
+
+```php
+public function checkAccountStatus($amicaleId)
+{
+ $amicale = Amicale::findOrFail($amicaleId);
+
+ if (empty($amicale->stripe_id)) {
+ return response()->json([
+ 'has_account' => false,
+ 'account_id' => null,
+ 'location_id' => null,
+ 'charges_enabled' => false,
+ 'payouts_enabled' => false,
+ 'onboarding_completed' => false,
+ ]);
+ }
+
+ try {
+ \Stripe\Stripe::setApiKey(config('services.stripe.secret'));
+ $account = \Stripe\Account::retrieve($amicale->stripe_id);
+
+ return response()->json([
+ 'has_account' => true,
+ 'account_id' => $amicale->stripe_id,
+ 'location_id' => $amicale->stripe_location_id, // ← Ajouté
+ 'charges_enabled' => $account->charges_enabled,
+ 'payouts_enabled' => $account->payouts_enabled,
+ 'onboarding_completed' => $account->details_submitted,
+ ]);
+
+ } catch (\Exception $e) {
+ return response()->json([
+ 'has_account' => false,
+ 'error' => $e->getMessage(),
+ ], 500);
+ }
+}
+```
+
+---
+
+## 🗑️ 4. Endpoint à SUPPRIMER (devenu inutile)
+
+### **❌ `POST /stripe/locations`**
+
+Cet endpoint n'est plus nécessaire car la Location est créée automatiquement dans `POST /stripe/accounts`.
+
+**Option 1** : Supprimer complètement
+**Option 2** : Le garder pour compatibilité temporaire (si utilisé ailleurs)
+
+---
+
+## 📝 5. Modification du modèle Eloquent
+
+### **Fichier** : `app/Models/Amicale.php`
+
+Ajouter le champ `stripe_location_id` :
+
+```php
+protected $fillable = [
+ // ... autres champs
+ 'stripe_id',
+ 'stripe_location_id', // ← Ajouté
+ 'chk_stripe',
+];
+
+protected $casts = [
+ 'chk_stripe' => 'boolean',
+];
+```
+
+---
+
+## ✅ 6. Tests à effectuer
+
+### **Test 1 : Nouvelle amicale**
+```bash
+curl -X POST http://localhost/api/stripe/accounts \
+ -H "Content-Type: application/json" \
+ -H "Authorization: Bearer {token}" \
+ -d '{
+ "fk_entite": 123,
+ "return_url": "https://app.geosector.fr/stripe/success",
+ "refresh_url": "https://app.geosector.fr/stripe/refresh"
+ }'
+```
+
+**Réponse attendue** :
+```json
+{
+ "success": true,
+ "account_id": "acct_xxxxxxxxxxxxx",
+ "location_id": "tml_xxxxxxxxxxxxx",
+ "onboarding_url": "https://connect.stripe.com/setup/...",
+ "charges_enabled": false,
+ "payouts_enabled": false,
+ "existing": false,
+ "message": "Compte Stripe Connect créé avec succès"
+}
+```
+
+### **Test 2 : Amicale avec compte existant**
+```bash
+curl -X POST http://localhost/api/stripe/accounts \
+ -H "Content-Type: application/json" \
+ -H "Authorization: Bearer {token}" \
+ -d '{
+ "fk_entite": 456,
+ "return_url": "https://app.geosector.fr/stripe/success",
+ "refresh_url": "https://app.geosector.fr/stripe/refresh"
+ }'
+```
+
+**Réponse attendue** :
+```json
+{
+ "success": true,
+ "account_id": "acct_xxxxxxxxxxxxx",
+ "location_id": "tml_xxxxxxxxxxxxx",
+ "onboarding_url": null,
+ "charges_enabled": true,
+ "payouts_enabled": true,
+ "existing": true,
+ "message": "Compte Stripe déjà configuré et actif"
+}
+```
+
+### **Test 3 : Vérifier la BDD**
+```sql
+SELECT id, name, stripe_id, stripe_location_id, chk_stripe
+FROM amicales
+WHERE id = 123;
+```
+
+**Résultat attendu** :
+```
++-----+------------------+-------------------+-------------------+------------+
+| id | name | stripe_id | stripe_location_id| chk_stripe |
++-----+------------------+-------------------+-------------------+------------+
+| 123 | Pompiers Paris15 | acct_xxxxxxxxxxxxx| tml_xxxxxxxxxxxxx | 1 |
++-----+------------------+-------------------+-------------------+------------+
+```
+
+---
+
+## 🚀 7. Déploiement
+
+### **Étapes** :
+1. ✅ Appliquer la migration SQL
+2. ✅ Déployer le code Backend modifié
+3. ✅ Tester avec Postman/curl
+4. ✅ Déployer le code Flutter modifié
+5. ✅ Tester le flow complet depuis l'app
+
+---
+
+## 📊 Comparaison Avant/Après
+
+| Aspect | Avant | Après |
+|--------|-------|-------|
+| **Appels API Flutter → Backend** | 3 | 1 |
+| **Appels Backend → Stripe** | 3 | 3 (mais atomiques) |
+| **Latence totale** | ~3-5s | ~1-2s |
+| **Gestion erreurs** | Complexe | Simplifié avec transaction |
+| **Atomicité** | ❌ Non | ✅ Oui (DB transaction) |
+| **Location ID sauvegardé** | ❌ Non | ✅ Oui |
+
+---
+
+## 🎯 Bénéfices
+
+1. ✅ **Performance** : Latence divisée par 2-3
+2. ✅ **Fiabilité** : Transaction BDD garantit la cohérence
+3. ✅ **Simplicité** : Code Flutter plus simple
+4. ✅ **Maintenance** : Moins de code à maintenir
+5. ✅ **Traçabilité** : Logs centralisés côté Backend
+6. ✅ **Tap to Pay prêt** : `location_id` disponible immédiatement
+
+---
+
+## ⚠️ Points d'attention
+
+1. **Rollback** : Si la transaction échoue, rien n'est sauvegardé (bon comportement)
+2. **Logs** : Bien logger chaque étape pour le debug
+3. **Stripe Connect limitations** : Respecter les rate limits Stripe
+4. **Tests** : Tester avec des comptes Stripe de test d'abord
+
+---
+
+## 📚 Ressources
+
+- [Stripe Connect Express Accounts](https://stripe.com/docs/connect/express-accounts)
+- [Stripe Terminal Locations](https://stripe.com/docs/terminal/fleet/locations)
+- [Stripe Account Links](https://stripe.com/docs/connect/account-links)
diff --git a/api/docs/TECHBOOK.md b/api/docs/TECHBOOK.md
index 5802ac61..3f876d4d 100755
--- a/api/docs/TECHBOOK.md
+++ b/api/docs/TECHBOOK.md
@@ -1,1385 +1,175 @@
-# Documentation Technique API RESTful PHP 8.3
+# Documentation Technique API GeoSector
-## Table des matières
+## 🏗️ Infrastructure
-1. [Structure du projet](#structure-du-projet)
-2. [Configuration du serveur](#configuration-du-serveur)
-3. [Flux d'une requête](#flux-dune-requête)
-4. [Architecture des composants](#architecture-des-composants)
-5. [Base de données](#base-de-données)
-6. [Sécurité](#sécurité)
-7. [Gestion des mots de passe (NIST SP 800-63B)](#gestion-des-mots-de-passe-nist-sp-800-63b)
-8. [Endpoints API](#endpoints-api)
-9. [Paiements Stripe Connect](#paiements-stripe-connect)
-10. [Changements récents](#changements-récents)
+**Stack** : PHP 8.3 | NGINX | MariaDB 10.11 | Alpine Linux (Incus)
-## Structure du projet
+| Environnement | Container | DB (IP) | URL | Serveur |
+|---------------|-----------|---------|-----|---------|
+| **DEV** | dva-geo | maria3 (13.23.33.4) | https://dapp.geosector.fr | IN3 (195.154.80.116) |
+| **REC** | rca-geo | maria3 (13.23.33.4) | https://rapp.geosector.fr | IN3 (195.154.80.116) |
+| **PROD** | pra-geo | maria4 (13.23.33.4) | https://app3.geosector.fr | IN4 (51.159.7.190) |
-```plaintext
-/api/
-├── docs/
-│ └── TECHBOOK.md
-├── src/
-│ ├── Controllers/
-│ │ └── UserController.php
-│ ├── Core/
-│ │ ├── Router.php
-│ │ ├── Request.php
-│ │ ├── Response.php
-│ │ ├── Session.php
-│ │ └── Database.php
-│ └── Config/
-│ └── config.php
-├── index.php
-└── .htaccess
+**Architecture** : MVC sans framework | Point d'entrée : `index.php` | Config : `AppConfig.php` singleton
+
+## 🗄️ Base de données
+
+### Modèle d'isolation par opération (CRITIQUE)
+
+**Principe** : Chaque opération est un **univers fermé** isolé de la table centrale `users`.
+
+```
+users (table centrale, conservée après suppression opération)
+ └── ope_users (id, fk_user, fk_operation) ← PIVOT par opération
+ ├── ope_users_sectors (fk_user → ope_users.id)
+ └── ope_pass (fk_user → ope_users.id)
```
-## Configuration du serveur
+**⚠️ IMPORTANT pour Flutter** :
+- **`users.id`** = Identifiant global (gestion membres, login)
+- **`ope_users.id`** = Identifiant opération (passages, secteurs)
+- **Flutter doit gérer les 2 IDs** lors du login et des CRUD
-### Prérequis
-
-- Debian 12
-- NGINX
-- PHP 8.3-FPM
-- MariaDB 10.11
-
-### Configuration NGINX
-
-Le serveur NGINX est configuré pour rediriger toutes les requêtes vers le point d'entrée `index.php` de l'API.
-
-### Configuration PHP-FPM
-
-PHP-FPM est configuré pour gérer les processus PHP avec des paramètres optimisés pour une API.
-
-## Flux d'une requête
-
-Exemple détaillé du parcours d'une requête POST /api/users :
-
-1. **Entrée de la requête**
-
- - La requête arrive sur le serveur NGINX
- - NGINX redirige vers PHP-FPM via le socket unix
- - Le fichier .htaccess redirige vers index.php
-
-2. **Initialisation (index.php)**
-
- - Chargement des dépendances
- - Initialisation de la configuration
- - Démarrage de la session
- - Configuration des headers CORS
- - Initialisation du routeur
-
-3. **Routage**
-
- - Le Router analyse la méthode HTTP (POST)
- - Analyse de l'URI (/api/users)
- - Correspondance avec les routes enregistrées
- - Instanciation du Controller approprié
-
-4. **Traitement (UserController)**
-
- - Vérification de l'authentification
- - Récupération des données JSON
- - Validation des données reçues
- - Traitement métier
- - Interaction avec la base de données
- - Préparation de la réponse
-
-5. **Réponse**
- - Formatage de la réponse en JSON
- - Configuration des headers de réponse
- - Envoi au client
-
-## Architecture des composants
-
-### Core Components
-
-#### Router
-
-- Gère le routage des requêtes
-- Associe les URLs aux Controllers
-- Gère les paramètres d'URL
-- Dispatch vers les méthodes appropriées
-
-#### Request
-
-- Parse les données entrantes
-- Gère les différents types de contenu
-- Nettoie et valide les entrées
-- Fournit une interface unifiée pour accéder aux données
-
-#### Response
-
-- Formate les réponses en JSON
-- Gère les codes HTTP
-- Configure les headers de réponse
-- Assure la cohérence des réponses
-
-#### Session
-
-- Gère l'état des sessions
-- Stocke les données d'authentification
-- Vérifie les permissions
-- Sécurise les données de session
-
-#### Database
-
-- Gère la connexion à MariaDB
-- Fournit une interface PDO
-- Gère le pool de connexions
-- Assure la sécurité des requêtes
-
-## Base de données
-
-### Architecture des containers MariaDB
-
-Depuis janvier 2025, les bases de données sont hébergées dans des containers MariaDB dédiés :
-
-| Environnement | Container API | Container DB | Serveur | IP DB | Nom BDD | Utilisateur | Source des données |
-|---------------|--------------|--------------|---------|-------|---------|-------------|-------------------|
-| **DEV** | dva-geo | maria3 | IN3 | 13.23.33.4 | dva_geo | dva_geo_user | Migré depuis dva-geo/geo_app |
-| **RECETTE** | rca-geo | maria3 | IN3 | 13.23.33.4 | rca_geo | rca_geo_user | Migré depuis rca-geo/geo_app |
-| **PRODUCTION** | pra-geo | maria4 | IN4 | 13.23.33.4 | pra_geo | pra_geo_user | **Dupliqué depuis maria3/rca_geo** |
-
-**Note importante :** La base de production `pra_geo` est créée en dupliquant `rca_geo` depuis IN3/maria3 vers IN4/maria4.
-
-**Avantages de cette architecture :**
-- Isolation des données par environnement
-- Performances optimisées (containers dédiés)
-- Sauvegardes indépendantes
-- Maintenance simplifiée
-- Séparation physique Production/Recette (serveurs différents)
-
-**Migration :** Utiliser le script `scripts/migrate_to_maria_containers.sh` pour migrer les données.
-
-### Structure des tables principales
-
-#### Table `users`
-- `encrypted_user_name` : Identifiant de connexion chiffré (unique)
-- `encrypted_email` : Email chiffré (unique)
-- `user_pass_hash` : Hash du mot de passe
-- `encrypted_name`, `encrypted_phone`, `encrypted_mobile` : Données personnelles chiffrées
-- Autres champs : `first_name`, `sect_name`, `fk_role`, `fk_entite`, etc.
-
-#### Table `entites` (Amicales)
-- `chk_mdp_manuel` (DEFAULT 0) : Gestion manuelle des mots de passe
-- `chk_username_manuel` (DEFAULT 0) : Gestion manuelle des identifiants
-- `chk_stripe` : Activation des paiements Stripe
-- Données chiffrées : `encrypted_name`, `encrypted_email`, `encrypted_phone`, etc.
-
-#### Table `medias`
-- `support` : Type de support (entite, user, operation, passage)
-- `support_id` : ID de l'élément associé
-- `file_category` : Catégorie (logo, export, carte, etc.)
-- `file_path` : Chemin complet du fichier
-- `processed_width/height` : Dimensions après traitement
-- Utilisée pour stocker les logos des entités
-
-### Chiffrement des données
-
-Toutes les données sensibles sont chiffrées avec AES-256-CBC :
-- Emails, noms, téléphones
-- Identifiants de connexion
-- Informations bancaires (IBAN, BIC)
-
-### Migration de base de données
-
-Script SQL pour ajouter les nouveaux champs :
-
-```sql
--- Ajout de la gestion manuelle des usernames
-ALTER TABLE `entites`
-ADD COLUMN `chk_username_manuel` tinyint(1) unsigned NOT NULL DEFAULT 0
-COMMENT 'Gestion des usernames manuelle (1) ou automatique (0)'
-AFTER `chk_mdp_manuel`;
-
--- Index pour optimiser la vérification d'unicité
-ALTER TABLE `users`
-ADD INDEX `idx_encrypted_user_name` (`encrypted_user_name`);
-```
-
-## Sécurité
-
-### Mesures implémentées
-
-- Validation stricte des entrées
-- Protection contre les injections SQL (PDO)
-- Hachage sécurisé des mots de passe
-- Headers de sécurité HTTP
-- Gestion des CORS
-- Session sécurisée
-- Authentification requise
-- Chiffrement AES-256 des données sensibles
-- Envoi séparé des identifiants par email
-
-## Gestion des mots de passe (NIST SP 800-63B)
-
-### Vue d'ensemble
-
-L'API implémente un système de gestion des mots de passe conforme aux recommandations NIST SP 800-63B, avec quelques adaptations spécifiques demandées par le client.
-
-### Service PasswordSecurityService
-
-Le service `PasswordSecurityService` (`src/Services/PasswordSecurityService.php`) gère :
-- Validation des mots de passe selon NIST
-- Vérification contre les bases de données de mots de passe compromis (HIBP)
-- Génération de mots de passe sécurisés
-- Estimation de la force des mots de passe
-
-### Conformités NIST respectées
-
-| Recommandation NIST | Notre Implémentation | Status |
-|-------------------|---------------------|--------|
-| **Longueur minimale : 8 caractères** | ✅ MIN = 8 caractères | ✅ CONFORME |
-| **Longueur maximale : 64 caractères minimum** | ✅ MAX = 64 caractères | ✅ CONFORME |
-| **Accepter TOUS les caractères ASCII imprimables** | ✅ Aucune restriction sur les caractères | ✅ CONFORME |
-| **Accepter les espaces** | ✅ Espaces acceptés (début, milieu, fin) | ✅ CONFORME |
-| **Accepter Unicode (émojis, accents, etc.)** | ✅ Support UTF-8 avec `mb_strlen()` | ✅ CONFORME |
-| **Vérifier contre les mots de passe compromis** | ✅ API Have I Been Pwned avec k-anonymity | ✅ CONFORME |
-| **Pas d'obligation de composition** | ✅ Pas d'erreur si manque majuscules/chiffres/spéciaux | ✅ CONFORME |
-| **Pas de changement périodique forcé** | ✅ Aucune expiration automatique | ✅ CONFORME |
-| **Permettre les phrases de passe** | ✅ "Mon chat Félix a 3 ans!" accepté | ✅ CONFORME |
-
-### Déviations par choix du client
-
-| Recommandation NIST | Notre Implémentation | Raison |
-|-------------------|---------------------|--------|
-| **Email unique par compte** | ❌ Plusieurs comptes par email autorisés | Demande client |
-| **Mot de passe ≠ identifiant** | ❌ Mot de passe = identifiant autorisé | Demande client |
-| **Vérifier contexte utilisateur** | ❌ Pas de vérification nom/email dans mdp | Demande client |
-
-### Vérification contre les mots de passe compromis
-
-#### Have I Been Pwned (HIBP) API
-
-L'implémentation utilise l'API HIBP avec la technique **k-anonymity** pour préserver la confidentialité :
-
-1. **Hash SHA-1** du mot de passe
-2. **Envoi des 5 premiers caractères** du hash à l'API
-3. **Comparaison locale** avec les suffixes retournés
-4. **Aucun mot de passe en clair** n'est transmis
-
-#### Mode "Fail Open"
-
-En cas d'erreur de l'API HIBP :
-- Le système laisse passer le mot de passe
-- Un avertissement est enregistré dans les logs
-- L'utilisateur n'est pas bloqué
-
-### Exemples de mots de passe
-
-#### Acceptés (conformes NIST)
-- `monmotdepasse` → Accepté (≥8 caractères, pas compromis)
-- `12345678` → Accepté SI pas dans HIBP
-- `Mon chat s'appelle Félix!` → Accepté (phrase de passe)
-- ` ` → Accepté si ≥8 espaces
-- `😀🎉🎈🎁🎂🍰🎊🎀` → Accepté (8 émojis)
-- `jean.dupont` → Accepté même si = username
-
-#### Refusés
-- `pass123` → Refusé (< 8 caractères)
-- `password` → Refusé (compromis dans HIBP)
-- `123456789` → Refusé (compromis dans HIBP)
-- Mot de passe > 64 caractères → Refusé
-
-### Force des mots de passe
-
-Le système privilégie la **LONGUEUR** sur la complexité (conforme NIST) :
-
-| Longueur | Force | Score |
-|----------|-------|-------|
-| < 8 car. | Trop court | 0-10 |
-| 8-11 car. | Acceptable | 20-40 |
-| 12-15 car. | Bon | 40-60 |
-| 16-19 car. | Fort | 60-80 |
-| ≥20 car. | Très fort | 80-100 |
-| Compromis | Compromis | ≤10 |
-
-### Génération automatique
-
-Pour la génération automatique, le système reste **strict** pour garantir des mots de passe forts :
-- Longueur : 12-16 caractères
-- Contient : majuscules + minuscules + chiffres + spéciaux
-- Vérifié contre HIBP (10 tentatives max)
-- Exemple : `Xk9#mP2$nL5!`
-
-### Gestion des comptes multiples par email
-
-Depuis janvier 2025, le système permet plusieurs comptes avec le même email :
-
-#### Fonction `lostPassword` adaptée
-- Recherche **TOUS** les comptes avec l'email fourni
-- Génère **UN SEUL** mot de passe pour tous ces comptes
-- Met à jour **TOUS** les comptes en une requête
-- Envoie **UN SEUL** email avec la liste des usernames concernés
-
-#### Exemple de comportement
-Si 3 comptes partagent l'email `contact@amicale.fr` :
-- `jean.dupont`
-- `marie.martin`
-- `paul.durand`
-
-L'email contiendra :
-```
-Bonjour,
-Voici votre nouveau mot de passe pour les comptes : jean.dupont, marie.martin, paul.durand
-Mot de passe : XyZ123!@#
-```
-
-### Endpoints API dédiés aux mots de passe
-
-#### Vérification de force (public)
-```http
-POST /api/password/check
-Content-Type: application/json
-
-{
- "password": "monmotdepasse",
- "check_compromised": true
-}
-```
-
-**Réponse :**
+**Réponse login :**
```json
{
- "status": "success",
- "valid": false,
- "errors": [
- "Ce mot de passe a été trouvé 23 547 fois dans des fuites de données."
- ],
- "warnings": [
- "Suggestion : Évitez les séquences communes pour plus de sécurité"
- ],
- "strength": {
- "score": 20,
- "strength": "Faible",
- "feedback": ["Ce mot de passe a été compromis"],
- "length": 13,
- "diversity": 1
- },
- "compromised": {
- "compromised": true,
- "occurrences": 23547,
- "message": "Ce mot de passe a été trouvé 23 547 fois dans des fuites de données"
- }
-}
-```
-
-#### Vérification de compromission uniquement (public)
-```http
-POST /api/password/compromised
-Content-Type: application/json
-
-{
- "password": "monmotdepasse"
-}
-```
-
-#### Génération automatique (authentifié)
-```http
-GET /api/password/generate?length=14
-Authorization: Bearer {session_id}
-```
-
-**Réponse :**
-```json
-{
- "status": "success",
- "password": "Xk9#mP2$nL5!qR",
- "length": 14,
- "strength": {
- "score": 85,
- "strength": "Très fort",
- "feedback": []
- }
-}
-```
-
-### Configuration et sécurité
-
-#### Paramètres de sécurité
-- **Timeout API HIBP** : 5 secondes
-- **Cache** : 15 minutes pour les vérifications répétées
-- **Logging** : Aucun mot de passe en clair dans les logs
-- **K-anonymity** : Seuls 5 caractères du hash SHA-1 envoyés
-
-#### Points d'intégration
-- `LoginController::register` : Validation lors de l'inscription
-- `LoginController::lostPassword` : Génération sécurisée
-- `UserController::createUser` : Validation si mot de passe manuel
-- `UserController::updateUser` : Validation lors du changement
-- `ApiService::generateSecurePassword` : Génération avec vérification HIBP
-
-### Résumé
-
-✅ **100% CONFORME NIST** pour les aspects techniques
-✅ **Adapté aux besoins du client** (emails multiples, mdp=username)
-✅ **Sécurité maximale** avec vérification HIBP
-✅ **Expérience utilisateur optimale** (souple mais sécurisé)
-
-## Endpoints API
-
-### Routes Publiques vs Privées
-
-L'API distingue deux types de routes :
-
-#### Routes Publiques
-
-- POST /api/login
-- POST /api/register
-- GET /api/health
-
-#### Routes Privées (Nécessitent une session authentifiée)
-
-- Toutes les autres routes
-
-### Authentification
-
-L'authentification utilise le système de session PHP natif.
-
-#### Login
-
-```http
-POST /api/login
-Content-Type: application/json
-
-{
- "email": "user@example.com",
- "password": "SecurePassword123"
-}
-```
-
-**Réponse réussie :**
-
-```json
-{
- "message": "Connecté avec succès",
- "user": {
- "id": 123,
- "email": "user@example.com"
- }
-}
-```
-
-**Notes importantes :**
-
-- Un cookie de session PHP sécurisé est automatiquement créé
-- Le cookie est httpOnly, secure et SameSite=Strict
-- L'ID de session est régénéré à chaque login réussi
-
-#### Logout
-
-```http
-POST /api/logout
-```
-
-**Réponse réussie :**
-
-```json
-{
- "message": "Déconnecté avec succès"
-}
-```
-
-### Sécurité des Sessions
-
-La configuration des sessions inclut :
-
-- Sessions PHP natives sécurisées
-- Protection contre la fixation de session
-- Cookies httpOnly (protection XSS)
-- Mode strict pour les cookies
-- Validation côté serveur à chaque requête
-
-- use_strict_mode = 1
-- cookie_httponly = 1
-- cookie_secure = 1
-- cookie_samesite = Strict
-- Régénération de l'ID de session après login
-- Destruction complète de la session au logout
-
-### Users
-
-#### Création d'utilisateur
-
-La création d'utilisateur s'adapte aux paramètres de l'entité (amicale) :
-
-```http
-POST /api/users
-Content-Type: application/json
-Authorization: Bearer {session_id}
-
-{
- "email": "john@example.com",
- "name": "John Doe",
- "first_name": "John",
- "role": 1,
- "fk_entite": 5,
- "username": "j.doe38", // Requis si chk_username_manuel=1 pour l'entité
- "password": "SecurePass123", // Requis si chk_mdp_manuel=1 pour l'entité
- "phone": "0476123456",
- "mobile": "0612345678",
- "sect_name": "Secteur A",
- "date_naissance": "1990-01-15",
- "date_embauche": "2020-03-01"
-}
-```
-
-**Comportement selon les paramètres de l'entité :**
-
-| chk_username_manuel | chk_mdp_manuel | Comportement |
-|---------------------|----------------|--------------|
-| 0 | 0 | Username et password générés automatiquement |
-| 0 | 1 | Username généré, password requis dans le payload |
-| 1 | 0 | Username requis dans le payload, password généré |
-| 1 | 1 | Username et password requis dans le payload |
-
-**Validation du username (si manuel) :**
-- Format : 10-30 caractères
-- Commence par une lettre
-- Caractères autorisés : a-z, 0-9, ., -, _
-- Doit être unique dans toute la base
-
-**Réponse réussie :**
-
-```json
-{
- "status": "success",
- "message": "Utilisateur créé avec succès",
- "id": 123,
- "username": "j.doe38", // Toujours retourné
- "password": "xY7#mK9@pL2" // Retourné seulement si généré automatiquement
-}
-```
-
-**Envoi d'emails :**
-- **Email 1** : Identifiant de connexion (toujours envoyé)
-- **Email 2** : Mot de passe (toujours envoyé, 1 seconde après le premier)
-
-**Codes de statut :**
-
-- 201: Création réussie
-- 400: Données invalides ou username/password manquant si requis
-- 401: Non authentifié
-- 403: Accès non autorisé (rôle insuffisant)
-- 409: Email ou username déjà utilisé
-- 500: Erreur serveur
-
-#### Vérification de disponibilité du username
-
-```http
-POST /api/users/check-username
-Content-Type: application/json
-Authorization: Bearer {session_id}
-
-{
- "username": "j.doe38"
-}
-```
-
-**Réponse si disponible :**
-
-```json
-{
- "status": "success",
- "available": true,
- "message": "Nom d'utilisateur disponible",
- "username": "j.doe38"
-}
-```
-
-**Réponse si déjà pris :**
-
-```json
-{
- "status": "success",
- "available": false,
- "message": "Ce nom d'utilisateur est déjà utilisé",
- "suggestions": ["j.doe38_42", "j.doe381234", "j.doe3825"]
-}
-```
-
-#### Autres endpoints
-
-- GET /api/users
-- GET /api/users/{id}
-- PUT /api/users/{id}
-- DELETE /api/users/{id}
-- POST /api/users/{id}/reset-password
-
-### Entités (Amicales)
-
-#### Upload du logo d'une entité
-
-```http
-POST /api/entites/{id}/logo
-Content-Type: multipart/form-data
-Authorization: Bearer {session_id}
-
-Body:
- logo: File (image/png, image/jpeg, image/jpg)
-```
-
-**Restrictions :**
-- Réservé aux administrateurs d'amicale (fk_role == 2)
-- L'admin ne peut uploader que le logo de sa propre amicale
-- Un seul logo actif par entité (le nouveau remplace l'ancien)
-
-**Traitement de l'image :**
-- Formats acceptés : PNG, JPG, JPEG
-- Redimensionnement automatique : 250x250px maximum (ratio conservé)
-- Résolution : 72 DPI (standard web)
-- Préservation de la transparence pour les PNG
-
-**Stockage :**
-- Chemin : `/uploads/entites/{id}/logo/logo_{id}_{timestamp}.{ext}`
-- Enregistrement dans la table `medias`
-- Suppression automatique de l'ancien logo
-
-**Réponse réussie :**
-
-```json
-{
- "status": "success",
- "message": "Logo uploadé avec succès",
- "media_id": 42,
- "file_name": "logo_5_1234567890.jpg",
- "file_path": "/entites/5/logo/logo_5_1234567890.jpg",
- "dimensions": {
- "width": 250,
- "height": 180
- }
-}
-```
-
-#### Récupération du logo d'une entité
-
-```http
-GET /api/entites/{id}/logo
-Authorization: Bearer {session_id}
-```
-
-**Réponse :**
-
-```json
-{
- "status": "success",
- "logo": {
- "id": 42,
- "data_url": "...",
- "file_name": "logo_5_1234567890.png",
- "mime_type": "image/png",
- "width": 250,
- "height": 180,
- "size": 15234
- }
-}
-```
-
-**Note :** Le logo est également inclus automatiquement dans la réponse du login si disponible.
-
-#### Mise à jour d'une entité
-
-```http
-PUT /api/entites/{id}
-Content-Type: application/json
-Authorization: Bearer {session_id}
-
-{
- "name": "Amicale de Grenoble",
- "adresse1": "123 rue de la Caserne",
- "adresse2": "",
- "code_postal": "38000",
- "ville": "Grenoble",
- "phone": "0476123456",
- "mobile": "0612345678",
- "email": "contact@amicale38.fr",
- "chk_stripe": true, // Activation paiement Stripe
- "chk_mdp_manuel": false, // Génération auto des mots de passe
- "chk_username_manuel": false, // Génération auto des usernames
- "chk_copie_mail_recu": true,
- "chk_accept_sms": false
-}
-```
-
-**Paramètres de gestion des membres :**
-
-| Paramètre | Type | Description |
-|-----------|------|-------------|
-| chk_mdp_manuel | boolean | `true`: L'admin saisit les mots de passe
`false`: Génération automatique |
-| chk_username_manuel | boolean | `true`: L'admin saisit les identifiants
`false`: Génération automatique |
-| chk_stripe | boolean | Active/désactive les paiements Stripe |
-
-**Note :** Ces paramètres sont modifiables uniquement par les administrateurs (fk_role > 1).
-
-#### Réponse du login avec paramètres entité
-
-Lors du login, les paramètres de l'entité sont retournés dans le groupe `amicale` :
-
-```json
-{
- "status": "success",
- "session_id": "abc123...",
- "session_expiry": "2025-01-09T15:30:00+00:00",
- "user": {
- "id": 9999980,
- "fk_entite": 5,
- "fk_role": 2,
- "fk_titre": null,
- "first_name": "Pierre",
- "sect_name": "",
- "date_naissance": "1990-01-15", // Maintenant correctement récupéré
- "date_embauche": "2020-03-01", // Maintenant correctement récupéré
- "username": "pv_admin",
- "name": "VALERY ADM",
- "phone": "0476123456", // Maintenant correctement récupéré
- "mobile": "0612345678", // Maintenant correctement récupéré
- "email": "contact@resalice.com"
- },
- "amicale": {
- "id": 5,
- "name": "Amicale de Grenoble",
- "chk_mdp_manuel": 0,
- "chk_username_manuel": 0,
- "chk_stripe": 1,
- "logo": { // Logo de l'entité (si disponible)
- "id": 42,
- "data_url": "...",
- "file_name": "logo_5_1234567890.png",
- "mime_type": "image/png",
- "width": 250,
- "height": 180
+ "users_sectors": [
+ {
+ "id": 123, // users.id (gestion membres)
+ "ope_user_id": 50, // ope_users.id (passages/secteurs)
+ "name": "John Doe",
+ "fk_sector": 456
}
- // ... autres champs
- }
+ ]
}
```
-Ces paramètres permettent à l'application Flutter d'adapter dynamiquement le formulaire de création de membre.
-
-## Paiements Stripe Connect
-
-### Vue d'ensemble
-
-L'API intègre un système complet de paiements via Stripe Connect, permettant aux amicales de recevoir des paiements pour leurs calendriers via deux méthodes :
-- **Paiements Web** : Interface de paiement dans un navigateur
-- **Tap to Pay** : Paiements NFC via l'application mobile Flutter
-
-### Architecture Stripe Connect
-
-#### Tables de base de données
-
-**Table `stripe_accounts` :**
-```sql
-CREATE TABLE `stripe_accounts` (
- `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
- `fk_entite` int(10) unsigned NOT NULL,
- `stripe_account_id` varchar(50) NOT NULL,
- `account_type` enum('express','standard','custom') DEFAULT 'express',
- `charges_enabled` tinyint(1) DEFAULT 0,
- `payouts_enabled` tinyint(1) DEFAULT 0,
- `details_submitted` tinyint(1) DEFAULT 0,
- `country` varchar(2) DEFAULT 'FR',
- `default_currency` varchar(3) DEFAULT 'eur',
- `business_name` varchar(255) DEFAULT NULL,
- `support_email` varchar(255) DEFAULT NULL,
- `onboarding_completed_at` timestamp NULL DEFAULT NULL,
- `created_at` timestamp NOT NULL DEFAULT current_timestamp(),
- `updated_at` timestamp NULL DEFAULT NULL ON UPDATE current_timestamp(),
- PRIMARY KEY (`id`),
- UNIQUE KEY `stripe_account_id` (`stripe_account_id`),
- KEY `fk_entite` (`fk_entite`),
- CONSTRAINT `stripe_accounts_ibfk_1` FOREIGN KEY (`fk_entite`) REFERENCES `entites` (`id`)
-);
-```
-
-**Ajout du champ `stripe_payment_id` dans `ope_pass` :**
-```sql
-ALTER TABLE `ope_pass` ADD COLUMN `stripe_payment_id` VARCHAR(50) DEFAULT NULL COMMENT 'ID du PaymentIntent Stripe (pi_xxx)';
-ALTER TABLE `ope_pass` ADD INDEX `idx_stripe_payment` (`stripe_payment_id`);
-```
-
-#### Services principaux
-
-**StripeService** (`src/Services/StripeService.php`) :
-- Gestion des PaymentIntents
-- Communication avec l'API Stripe
-- Gestion des comptes Stripe Connect
-
-**StripeController** (`src/Controllers/StripeController.php`) :
-- Endpoints pour la création de PaymentIntents
-- Gestion des webhooks Stripe
-- API pour les comptes Connect
-
-### Flow de paiement
-
-#### 1. Création du compte Stripe Connect (Onboarding)
-
-```http
-POST /api/stripe/accounts/create
-Authorization: Bearer {session_id}
-Content-Type: application/json
-
-{
- "amicale_id": 45,
- "type": "express",
- "country": "FR",
- "email": "contact@amicale-pompiers.fr",
- "business_profile": {
- "name": "Amicale des Pompiers",
- "product_description": "Vente de calendriers des pompiers",
- "mcc": "8398"
- }
-}
-```
-
-**Réponse :**
+**Requêtes API depuis Flutter :**
```json
-{
- "success": true,
- "stripe_account_id": "acct_1O3ABC456DEF789",
- "onboarding_url": "https://connect.stripe.com/express/oauth/authorize?...",
- "status": "pending"
-}
+// ✅ Création secteur : envoyer ope_users.id
+POST /api/sectors { "users": [50, 51] } // ope_users.id
+
+// ✅ Création passage : fk_user = ope_users.id
+POST /api/passages { "fk_user": 50 } // ope_users.id
+
+// ✅ Modification membre : utiliser users.id
+PUT /api/users/123 // users.id
```
-#### 2. Création d'un PaymentIntent (Tap to Pay)
-
-**Flow actuel (v2) :**
-1. L'application crée/modifie d'abord le passage pour obtenir un ID réel
-2. Puis crée le PaymentIntent avec cet ID
-
-```http
-POST /api/stripe/payments/create-intent
-Authorization: Bearer {session_id}
-Content-Type: application/json
-
-{
- "amount": 2500, // 25€ en centimes
- "passage_id": 456, // ID RÉEL du passage (jamais 0)
- "payment_method_types": ["card_present"], // Tap to Pay
- "location_id": "tml_xxx",
- "amicale_id": 45,
- "member_id": 67,
- "stripe_account": "acct_1234"
-}
-```
-
-**Réponse :**
-```json
-{
- "success": true,
- "client_secret": "pi_3QaXYZ_secret_xyz",
- "payment_intent_id": "pi_3QaXYZ123ABC456",
- "amount": 2500,
- "currency": "eur",
- "passage_id": 456,
- "type": "tap_to_pay"
-}
-```
-
-#### 3. Traitement du paiement
-
-**Côté application Flutter :**
-- Utilisation du SDK Stripe Terminal
-- Collecte NFC avec le `client_secret`
-- Traitement automatique du paiement
-
-**Mise à jour automatique :**
-- Le `stripe_payment_id` est automatiquement ajouté au passage lors de la création du PaymentIntent
-- Lien bidirectionnel entre le passage et le paiement Stripe
-
-### Endpoints Stripe
-
-#### Gestion des comptes
-
-- `POST /api/stripe/accounts/create` : Création d'un compte Connect
-- `GET /api/stripe/accounts/{id}` : Statut d'un compte
-- `PUT /api/stripe/accounts/{id}` : Mise à jour d'un compte
-
-#### Gestion des paiements
-
-- `POST /api/stripe/payments/create-intent` : Création d'un PaymentIntent
-- `GET /api/stripe/payments/{id}` : Statut d'un paiement
-- `POST /api/stripe/payments/confirm` : Confirmation d'un paiement
-
-#### Gestion des devices Tap to Pay
-
-- `GET /api/stripe/devices/certified-android` : Liste des appareils Android certifiés
-- `POST /api/stripe/devices/check-tap-to-pay` : Vérification de compatibilité d'un appareil
-- `GET /api/stripe/config` : Configuration publique Stripe
-- `GET /api/stripe/stats` : Statistiques de paiement
-
-#### Webhooks
-
-- `POST /api/stripe/webhooks` : Réception des événements Stripe
- - `account.updated` : Mise à jour du statut d'un compte
- - `payment_intent.succeeded` : Confirmation d'un paiement réussi
- - `payment_intent.payment_failed` : Échec d'un paiement
-
-### Sécurité et validation
-
-#### Prérequis pour les paiements :
-- ✅ Compte Stripe Connect activé (`charges_enabled = 1`)
-- ✅ Virements activés (`payouts_enabled = 1`)
-- ✅ Onboarding terminé (`details_submitted = 1`)
-- ✅ Passage existant avec montant correspondant
-- ✅ Utilisateur authentifié et autorisé
-
-#### Validation des montants :
-- Minimum : 1€ (100 centimes)
-- Maximum : 999€ (99 900 centimes)
-- Vérification de correspondance avec le passage
-
-#### Sécurité des transactions :
-- Headers CORS configurés
-- Validation côté serveur obligatoire
-- Logs de toutes les transactions
-- Gestion des erreurs robuste
-
-### États et statuts
-
-#### États des comptes Stripe :
-- `pending` : Onboarding en cours
-- `restricted` : Informations manquantes
-- `active` : Opérationnel pour les paiements
-- `rejected` : Refusé par Stripe
-
-#### États des paiements :
-- `requires_payment_method` : En attente de paiement
-- `processing` : Traitement en cours
-- `succeeded` : Paiement réussi
-- `canceled` : Paiement annulé
-- `requires_action` : Action utilisateur requise
-
-### Intégration avec l'application
-
-#### Flutter (Tap to Pay) :
-- SDK Stripe Terminal pour iOS/Android
-- Interface NFC native
-- Gestion des états du terminal
-- Validation en temps réel
-
-#### Web (Paiements navigateur) :
-- Stripe.js pour l'interface
-- Formulaire de carte sécurisé
-- Confirmation 3D Secure automatique
-
-### Monitoring et logs
-
-#### Logs importants :
-- Création/mise à jour des comptes Connect
-- Succès/échecs des paiements
-- Erreurs webhook Stripe
-- Tentatives de paiement frauduleuses
-
-#### Métriques de suivi :
-- Taux de succès des paiements par amicale
-- Montants moyens des transactions
-- Temps de traitement des paiements
-- Erreurs par type d'appareil
-
-### Configuration environnement
-
-#### Architecture des clés Stripe
-
-Depuis janvier 2025, les clés Stripe sont **séparées par environnement** dans `src/Config/AppConfig.php` :
-
-| Environnement | URL | Mode | Clés utilisées | Status |
-|---------------|-----|------|----------------|--------|
-| **DEV** | https://dapp.geosector.fr | `test` | Clés TEST Pierre (dev plateforme) | ✅ Opérationnel |
-| **RECETTE** | https://rapp.geosector.fr | `test` | Clés TEST du client | ⏳ À configurer |
-| **PRODUCTION** | https://app.geosector.fr | `live` | Clés LIVE du client | ⏳ À configurer |
-
-**Emplacement dans le code :**
-- **DEV** : `AppConfig.php` lignes 175-187 (section `dapp.geosector.fr`)
-- **RECETTE** : `AppConfig.php` lignes 150-162 (section `rapp.geosector.fr`)
-- **PRODUCTION** : `AppConfig.php` lignes 126-138 (section `app.geosector.fr`)
-
-#### Configuration des clés client
-
-Pour configurer les clés Stripe du client :
-
-1. **Récupérer les clés depuis le Dashboard Stripe du client**
- - Se connecter sur https://dashboard.stripe.com
- - Aller dans **Développeurs → Clés API**
- - Pour les clés TEST : Mode Test activé
- - Pour les clés LIVE : Mode Live activé
-
-2. **Remplacer les placeholders dans AppConfig.php**
- - **RECETTE** (ligne 152-153) : Remplacer `CLIENT_PK_TEST_A_REMPLACER` et `CLIENT_SK_TEST_A_REMPLACER`
- - **PRODUCTION** (ligne 130-131) : Remplacer `CLIENT_PK_LIVE_A_REMPLACER` et `CLIENT_SK_LIVE_A_REMPLACER`
-
-3. **Déployer selon l'environnement**
- ```bash
- # Déployer en RECETTE
- ./deploy-api.sh rca
-
- # Déployer en PRODUCTION
- ./deploy-api.sh pra
- ```
-
-**⚠️ Sécurité :** Voir `TODO-API.md` section "Sécurisation des clés Stripe" pour étudier une approche plus sécurisée (variables d'environnement, fichiers séparés).
-
-#### Comptes Connect
-- Type : Express (simplifié pour les associations)
-- Pays : France (FR)
-- Devise : Euro (EUR)
-- Frais : Standard Stripe Connect
-- Pas de commission plateforme (100% pour l'amicale)
-
-### Gestion des appareils certifiés Tap to Pay
-
-#### Table `stripe_android_certified_devices`
-
-Stocke la liste des appareils Android certifiés pour Tap to Pay en France :
-- **95+ appareils** pré-chargés lors de l'installation
-- **Mise à jour automatique** hebdomadaire via CRON
-- **Vérification de compatibilité** via endpoints dédiés
-
-#### Endpoint de vérification de compatibilité
-
-```http
-POST /api/stripe/devices/check-tap-to-pay
-Content-Type: application/json
-
-{
- "platform": "ios" | "android",
- "manufacturer": "Samsung", // Requis pour Android
- "model": "SM-S921B" // Requis pour Android
-}
-```
-
-**Réponse Android compatible :**
-```json
-{
- "status": "success",
- "tap_to_pay_supported": true,
- "message": "Tap to Pay disponible sur cet appareil",
- "min_android_version": 14
-}
-```
-
-**Réponse iOS :**
-```json
-{
- "status": "success",
- "message": "Vérification iOS à faire côté client",
- "requirements": "iPhone XS ou plus récent avec iOS 16.4+",
- "details": "iOS 16.4 minimum requis pour le support PIN complet"
-}
-```
-
-#### Requirements Tap to Pay
-
-| Plateforme | Appareil minimum | OS minimum | Notes |
-|------------|------------------|------------|-------|
-| **iOS** | iPhone XS (2018+) | iOS 16.4+ | Support PIN complet |
-| **Android** | Variable | Android 11+ | NFC obligatoire, non rooté |
-
-### Documentation technique complète
-
-Pour le flow détaillé complet, voir :
-- **`docs/STRIPE-TAP-TO-PAY-FLOW.md`** : Documentation technique complète du flow de paiement
-- **`docs/PLANNING-STRIPE-API.md`** : Planification et architecture Stripe
-- **`docs/STRIPE-TAP-TO-PAY-REQUIREMENTS.md`** : Requirements officiels et liste complète des devices certifiés
-
-## Intégration Frontend
-
-### Configuration des Requêtes
-
-Toutes les requêtes API depuis le frontend doivent inclure :
-
-```javascript
-fetch('/api/endpoint', {
- credentials: 'include', // Important pour les cookies de session
- // ... autres options
-});
-```
-
-### Gestion des Sessions
-
-- Les cookies de session sont automatiquement gérés par le navigateur
-- Pas besoin de stocker ou gérer des tokens manuellement
-- Redirection vers /login si session expirée (401)
-
-## Système de tâches CRON
-
-### Vue d'ensemble
-
-L'API utilise des scripts CRON pour automatiser les tâches de maintenance et de traitement. Les scripts sont situés dans `/scripts/cron/` et s'exécutent dans les containers Incus Alpine.
-
-### Tâches CRON configurées
-
-| Script | Fréquence | Fonction | Container |
-|--------|-----------|----------|-----------|
-| `process_email_queue.php` | */5 * * * * | Traite la queue d'emails (reçus, notifications) | DVA, RCA |
-| `cleanup_security_data.php` | 0 2 * * * | Nettoie les données de sécurité obsolètes | DVA, RCA |
-| `update_stripe_devices.php` | 0 3 * * 0 | Met à jour la liste des devices certifiés Tap to Pay | DVA, RCA |
-
-### Configuration des CRONs
-
-Sur les containers Alpine (dva-geo, rca-geo, pra-geo) :
+### Tables principales
+
+- **`entites`** : Amicales (chiffrement AES-256-CBC sur nom, email, IBAN)
+- **`users`** : Table centrale utilisateurs (conservée même si opération supprimée)
+- **`operations`** : Campagnes liées à une entité
+- **`ope_users`** : **PIVOT** users ↔ opérations (ON DELETE CASCADE depuis operations)
+- **`ope_sectors`** : Secteurs géographiques (polygones)
+- **`ope_users_sectors`** : Affectation users ↔ secteurs (FK → `ope_users.id`)
+- **`ope_pass`** : Passages (FK → `ope_users.id`, ON DELETE CASCADE)
+- **`medias`** : Fichiers (logos, exports, reçus) - Stockage : `/uploads/`
+
+## 🔒 Sécurité
+
+- **Auth** : Sessions PHP (httpOnly, secure, SameSite=Strict)
+- **Mots de passe** : NIST SP 800-63B, bcrypt, HIBP check (k-anonymity)
+- **Chiffrement** : AES-256-CBC (noms, emails, téléphones, IBAN)
+- **Protection** : Brute force (8 tentatives/5min), IP blocking, PDO prepared statements
+- **Monitoring** : `SecurityMonitor`, `AlertService`, `IPBlocker`
+
+## 💳 Stripe Connect
+
+- **DEV** : Clés TEST Pierre
+- **REC** : Clés TEST client + webhook `webhook-rca`
+- **PROD** : Clés LIVE client + webhook `webhook-pra`
+- **API** : `2025-08-27.basil`
+- **Tap to Pay** : iOS 16.4+ (iPhone XS+) | Android 11+ (95+ devices certifiés)
+- **Flow** : Passage créé → PaymentIntent → Tap to Pay → Mise à jour `stripe_payment_id`
+
+## 📦 Fonctionnalités
+
+1. **Reçus fiscaux** : PDF auto (<5KB) pour dons, envoi email queue
+2. **Logos entités** : Upload PNG/JPG, redimensionnement 250x250px, base64
+3. **Migration** : Endpoints REST par entité (9 phases)
+4. **CRONs** : Email queue (*/5), cleanup sécurité (2h), Stripe devices (dim 3h)
+
+## 🚀 Déploiement
```bash
-# Vérifier les crons actifs
-crontab -l
-
-# Éditer les crons
-crontab -e
-
-# Format des lignes cron
-*/5 * * * * /usr/bin/php /var/www/geosector/api/scripts/cron/process_email_queue.php >> /var/www/geosector/api/logs/email_queue.log 2>&1
-0 2 * * * /usr/bin/php /var/www/geosector/api/scripts/cron/cleanup_security_data.php >> /var/www/geosector/api/logs/cleanup_security.log 2>&1
-0 3 * * 0 /usr/bin/php /var/www/geosector/api/scripts/cron/update_stripe_devices.php >> /var/www/geosector/api/logs/stripe_devices.log 2>&1
+./deploy-api.sh # Local → dva-geo (DEV)
+./deploy-api.sh rca # dva-geo → rca-geo (REC)
+./deploy-api.sh pra # rca-geo → pra-geo (PROD)
```
-### Script `process_email_queue.php`
+- Backup auto (10 versions)
+- Préservation `/logs/` et `/uploads/`
+- Permissions : `nginx:nginx` (code), `nginx:nginx` (logs/uploads)
+- Composer install avec `--optimize-autoloader`
-- **Fonction** : Envoie les emails en attente dans la table `email_queue`
-- **Batch** : 50 emails maximum par exécution
-- **Lock file** : `/tmp/process_email_queue.lock` (évite l'exécution simultanée)
-- **Gestion d'erreur** : 3 tentatives max par email
+## ⚠️ Points d'attention API ↔ Flutter
-### Script `cleanup_security_data.php`
+### 1. Isolation opérations (depuis Oct 2025)
-- **Fonction** : Purge les données de sécurité selon la politique de rétention
-- **Rétention** :
- - Métriques de performance : 30 jours
- - Tentatives de login échouées : 7 jours
- - Alertes résolues : 90 jours
- - IPs expirées : Déblocage immédiat
+**Avant** : `ope_pass.fk_user` → `users.id` (table centrale)
+**Après** : `ope_pass.fk_user` → `ope_users.id` (pivot opération)
-### Script `update_stripe_devices.php`
+**Impact Flutter** :
+- Login retourne **2 IDs** : `id` (users.id) + `ope_user_id` (ope_users.id)
+- Création secteur/passage : envoyer `ope_user_id`
+- Affichage passages : mapper avec `ope_user_id`
-- **Fonction** : Maintient à jour la liste des appareils certifiés Tap to Pay
-- **Source** : Liste de 95+ devices intégrée + fichier JSON optionnel
-- **Actions** :
- - Ajoute les nouveaux appareils certifiés
- - Met à jour les versions Android minimales
- - Désactive les appareils obsolètes
- - Envoie une notification email si changements importants
-- **Personnalisation** : Possibilité d'ajouter des devices via `/data/stripe_certified_devices.json`
+### 2. Endpoints critiques modifiés
-### Monitoring des CRONs
+| Endpoint | Body envoyé | Mapping |
+|----------|-------------|---------|
+| `POST /api/sectors` | `users: [50, 51]` | ope_users.id |
+| `PUT /api/sectors/{id}` | `users: [50, 51]` | ope_users.id |
+| `POST /api/passages` | `fk_user: 50` | ope_users.id |
+| `PUT /api/passages/{id}` | `fk_user: 50` | ope_users.id |
+| `POST /api/users` | - | Retourne `ope_user_id` |
-Les logs sont stockés dans `/var/www/geosector/api/logs/` :
-- `email_queue.log` : Logs du traitement des emails
-- `cleanup_security.log` : Logs du nettoyage sécurité
-- `stripe_devices.log` : Logs de mise à jour des devices
+### 3. Requête SQL typique
-## Maintenance et Déploiement
+```sql
+-- ❌ AVANT (CASSÉ)
+SELECT op.*, u.encrypted_name
+FROM ope_pass op
+JOIN users u ON op.fk_user = u.id
-### Logs
-
-- Logs d'accès NGINX : /var/log/nginx/api-access.log
-- Logs d'erreur NGINX : /var/log/nginx/api-error.log
-- Logs PHP : /var/log/php/php-error.log
-
-### Déploiement
-
-Le script `deploy-api.sh` gère le déploiement sur les 3 environnements :
-
-```bash
-# Déploiement DEV : code local → container dva-geo sur IN3
-./deploy-api.sh
-
-# Déploiement RECETTE : container dva-geo → container rca-geo sur IN3
-./deploy-api.sh rca
-
-# Déploiement PRODUCTION : container rca-geo (IN3) → container pra-geo (IN4)
-./deploy-api.sh pra
+-- ✅ APRÈS (CORRECT)
+SELECT op.*, u.encrypted_name
+FROM ope_pass op
+JOIN ope_users ou ON op.fk_user = ou.id
+JOIN users u ON ou.fk_user = u.id
```
-Flux de déploiement :
-1. **DEV** : Archive du code local, déploiement sur container `dva-geo` sur IN3 (195.154.80.116)
- - URL publique : https://dapp.geosector.fr/api/
- - IP interne : http://13.23.33.43/api/
-2. **RECETTE** : Archive depuis container `dva-geo`, déploiement sur `rca-geo` sur IN3
- - URL publique : https://rapp.geosector.fr/api/
-3. **PRODUCTION** : Archive depuis `rca-geo` (IN3), déploiement sur `pra-geo` (51.159.7.190)
- - URL publique : https://app.geosector.fr/api/
+### 4. Suppression en cascade
-Caractéristiques :
-- Sauvegarde automatique avec rotation (garde les 10 dernières)
-- Préservation des dossiers `logs/` et `uploads/`
-- Gestion des permissions :
- - Code API : `nginx:nginx` (755/644)
- - Logs et uploads : `nobody:nginx` (755/644)
-- Installation des dépendances Composer (pas de mise à jour)
-- Journalisation dans `~/.geo_deploy_history`
+```sql
+DELETE FROM operations WHERE id = 850;
+-- Supprime automatiquement (CASCADE) :
+-- - ope_users
+-- - ope_users_sectors
+-- - ope_pass
+-- - ope_sectors
+-- ✅ users conservé (table centrale)
+```
-### Surveillance
+## 📝 Changelog critique
-- Monitoring des processus PHP-FPM
-- Surveillance de la base de données
-- Monitoring des performances
-- Alertes sur erreurs critiques
+**Version 3.3.7 (26 Oct 2025)** :
+- 🔧 Correction bug `SectorController::update()` : Recherche users par `ope_users.id` au lieu de `users.id`
+- 🔧 Permissions logs corrigées : `nginx:nginx` + `750/640`
+- 🔧 PHP `display_errors = Off` (warnings loggés dans `/var/log/php83/error.log`)
-## Changements récents
+**Version 3.3.6 (21 Oct 2025)** :
+- Validation inscription : Code postal + ville (doublon)
-### Version 3.2.5 (29 Septembre 2025)
+**Version 3.2.7 (16 Oct 2025)** :
+- Migration RCA-GEO vers maria3 complétée
+- URL PROD : `app3.geosector.fr`
-#### 1. Système de gestion automatique des devices Tap to Pay
+**Version 3.2.4-3.2.6 (Sep 2025)** :
+- Stripe Connect complet (Tap to Pay, webhooks multi-env)
-**Nouveaux endpoints ajoutés :**
-- `GET /api/stripe/devices/certified-android` : Récupération de la liste complète des appareils certifiés
-- `POST /api/stripe/devices/check-tap-to-pay` : Vérification de compatibilité d'un appareil spécifique
-- Endpoints publics (pas d'authentification requise) pour vérification côté app
+---
-**Script CRON de mise à jour automatique :**
-- **Script** : `/scripts/cron/update_stripe_devices.php`
-- **Fréquence** : Hebdomadaire (dimanche 3h)
-- **Fonction** : Maintient à jour la liste de 95+ appareils Android certifiés
-- **Base de données** : Table `stripe_android_certified_devices` avec 77 appareils actifs
-
-**Corrections des requirements iOS :**
-- Mise à jour : iOS 16.4+ minimum (au lieu de 15.4/16.0)
-- Raison : Support PIN complet obligatoire pour les paiements > 50€
-
-**Documentation ajoutée :**
-- `docs/STRIPE-TAP-TO-PAY-REQUIREMENTS.md` : Requirements officiels complets
-- Liste exhaustive des appareils certifiés par fabricant
-- Configuration SDK pour toutes les plateformes
-
-#### 2. Configuration des tâches CRON sur les containers
-
-**Environnements configurés :**
-- **DVA-GEO (DEV)** : 3 CRONs actifs
-- **RCA-GEO (RECETTE)** : 3 CRONs actifs (ajoutés le 29/09)
-- **PRA-GEO (PROD)** : À configurer
-
-**Tâches automatisées :**
-1. Queue d'emails : Toutes les 5 minutes
-2. Nettoyage sécurité : Quotidien à 2h
-3. Mise à jour devices Stripe : Hebdomadaire dimanche 3h
-
-### Version 3.2.4 (Septembre 2025)
-
-#### 1. Implémentation complète de Stripe Connect V1
-
-**Paiements Stripe intégrés pour les amicales :**
-- **Stripe Connect Express** : Onboarding simplifié pour les associations
-- **Tap to Pay** : Paiements NFC via l'application mobile Flutter
-- **Paiements Web** : Interface de paiement navigateur avec Stripe.js
-- **Webhooks** : Gestion automatique des événements Stripe
-
-**Nouvelles tables de base de données :**
-- `stripe_accounts` : Gestion des comptes Connect par amicale
-- `stripe_payment_history` : Historique des transactions Stripe
-- `stripe_refunds` : Gestion des remboursements
-- Ajout de `stripe_payment_id` dans `ope_pass` pour liaison bidirectionnelle
-
-**Nouveaux services :**
-- **StripeService** : Communication avec l'API Stripe, gestion des PaymentIntents
-- **StripeController** : Endpoints API pour création de comptes, paiements et webhooks
-
-**Flow de paiement optimisé (v2) :**
-1. Passage créé/modifié EN PREMIER pour obtenir un ID réel
-2. Création PaymentIntent avec `passage_id` réel (jamais 0)
-3. Traitement Tap to Pay via SDK Stripe Terminal
-4. Mise à jour automatique du passage avec `stripe_payment_id`
-
-**Endpoints ajoutés :**
-- `POST /api/stripe/accounts/create` : Création compte Connect
-- `POST /api/stripe/payments/create-intent` : Création PaymentIntent
-- `GET /api/stripe/payments/{id}` : Statut d'un paiement
-- `POST /api/stripe/webhooks` : Réception événements Stripe
-
-**Sécurité et validation :**
-- Validation stricte des montants (1€ à 999€)
-- Vérification correspondance passage/montant
-- Gestion des permissions par amicale
-- Logs complets des transactions
-
-**Configuration multi-environnements :**
-- DEV/RECETTE : Clés de test Stripe
-- PRODUCTION : Clés live avec webhooks sécurisés
-- Migration base de données via `migrate_stripe_payment_id.sql`
-
-**Documentation technique :**
-- `docs/STRIPE-TAP-TO-PAY-FLOW.md` : Flow complet de paiement
-- `docs/PLANNING-STRIPE-API.md` : Architecture et planification
-
-### Version 3.0.7 (Août 2025)
-
-#### 1. Implémentation complète de la norme NIST SP 800-63B pour les mots de passe
-- **Nouveau service :** `PasswordSecurityService` pour la gestion sécurisée des mots de passe
-- **Vérification HIBP :** Intégration de l'API Have I Been Pwned avec k-anonymity
-- **Validation souple :** Suppression des obligations de composition (majuscules, chiffres, spéciaux)
-- **Support Unicode :** Acceptation de tous les caractères, incluant émojis et espaces
-- **Nouveaux endpoints :** `/api/password/check`, `/api/password/compromised`, `/api/password/generate`
-
-#### 2. Autorisation des emails multiples
-- **Suppression de l'unicité :** Un même email peut être utilisé pour plusieurs comptes
-- **Adaptation de `lostPassword` :** Mise à jour de tous les comptes partageant l'email
-- **Un seul mot de passe :** Tous les comptes avec le même email reçoivent le même nouveau mot de passe
-
-#### 3. Autorisation mot de passe = identifiant
-- **Choix client :** Permet d'avoir un mot de passe identique au nom d'utilisateur
-- **Pas de vérification contextuelle :** Aucune vérification nom/email dans le mot de passe
-
-### Version 3.0.6 (Août 2025)
-
-#### 1. Correction des rôles administrateurs
-- **Avant :** Les administrateurs d'amicale devaient avoir `fk_role > 2`
-- **Après :** Les administrateurs d'amicale ont `fk_role > 1` (donc rôle 2 et plus)
-- **Impact :** Les champs `chk_stripe`, `chk_mdp_manuel`, `chk_username_manuel` sont maintenant modifiables par les admins d'amicale (rôle 2)
-
-#### 2. Envoi systématique des deux emails lors de la création d'utilisateur
-- **Avant :** Le 2ème email (mot de passe) n'était envoyé que si le mot de passe était généré automatiquement
-- **Après :** Les deux emails sont toujours envoyés lors de la création d'un membre
- - Email 1 : Identifiant (username)
- - Email 2 : Mot de passe (1 seconde après)
-- **Raison :** Le nouveau membre a toujours besoin des deux informations pour se connecter
-
-#### 3. Ajout des champs manquants dans la réponse du login
-- **Champs ajoutés dans la requête SQL :**
- - `fk_titre`
- - `date_naissance`
- - `date_embauche`
- - `encrypted_phone`
- - `encrypted_mobile`
-- **Impact :** Ces données sont maintenant correctement retournées dans l'objet `user` lors du login
-
-#### 4. Système de gestion des logos d'entité
-- **Nouvelle fonctionnalité :** Upload et gestion des logos pour les amicales
-- **Routes ajoutées :**
- - `POST /api/entites/{id}/logo` : Upload d'un nouveau logo
- - `GET /api/entites/{id}/logo` : Récupération du logo
-- **Caractéristiques :**
- - Réservé aux administrateurs d'amicale (fk_role == 2)
- - Un seul logo actif par entité
- - Redimensionnement automatique (250x250px max)
- - Format base64 dans les réponses JSON (compatible Flutter)
- - Logo inclus automatiquement dans la réponse du login
-
-#### 5. Amélioration de l'intégration Flutter
-- **Format d'envoi des images :** Base64 data URL pour compatibilité multiplateforme
-- **Structure de réponse enrichie :** Le logo est inclus dans l'objet `amicale` lors du login
-- **Optimisation :** Pas de requête HTTP supplémentaire nécessaire pour afficher le logo
-
-### Version 3.0.8 (Janvier 2025)
-
-#### 1. Système de génération automatique de reçus fiscaux pour les dons
-- **Nouveau service :** `ReceiptService` pour la génération automatique de reçus PDF
-- **Déclencheurs automatiques :**
- - Création d'un passage avec `fk_type=1` (don) et email valide
- - Mise à jour d'un passage en don si `nom_recu` est vide/null
-- **Caractéristiques techniques :**
- - PDF ultra-légers (< 5KB) générés en format natif sans librairie externe
- - Support des caractères accentués avec conversion automatique
- - Stockage structuré : `/uploads/entites/{entite_id}/recus/{operation_id}/`
- - Enregistrement dans la table `medias` avec catégorie `recu`
-- **Queue d'envoi email :**
- - Envoi automatique par email avec pièce jointe PDF
- - Format MIME multipart pour compatibilité maximale
- - Gestion dans la table `email_queue` avec statut de suivi
-- **Nouvelle route API :**
- - `GET /api/passages/{id}/receipt` : Récupération du PDF d'un reçu
- - Retourne le PDF en base64 ou téléchargement direct selon Accept header
-- **Champs base de données utilisés :**
- - `nom_recu` : Nom du fichier PDF généré
- - `date_creat_recu` : Date de génération du reçu
- - `date_sent_recu` : Date d'envoi par email
- - `chk_email_sent` : Indicateur d'envoi réussi
+**Mis à jour : 26 Octobre 2025**
diff --git a/api/docs/geo_app.sql b/api/docs/geo_app.sql
deleted file mode 100755
index 559489fe..00000000
--- a/api/docs/geo_app.sql
+++ /dev/null
@@ -1,476 +0,0 @@
--- -------------------------------------------------------------
--- TablePlus 6.4.8(608)
---
--- https://tableplus.com/
---
--- Database: geo_app
--- Generation Time: 2025-06-09 18:03:43.5140
--- -------------------------------------------------------------
-
-
-/*!40101 SET @OLD_CHARACTER_SET_CLIENT=@@CHARACTER_SET_CLIENT */;
-/*!40101 SET @OLD_CHARACTER_SET_RESULTS=@@CHARACTER_SET_RESULTS */;
-/*!40101 SET @OLD_COLLATION_CONNECTION=@@COLLATION_CONNECTION */;
-/*!40101 SET NAMES utf8mb4 */;
-/*!40014 SET @OLD_UNIQUE_CHECKS=@@UNIQUE_CHECKS, UNIQUE_CHECKS=0 */;
-/*!40014 SET @OLD_FOREIGN_KEY_CHECKS=@@FOREIGN_KEY_CHECKS, FOREIGN_KEY_CHECKS=0 */;
-/*!40101 SET @OLD_SQL_MODE=@@SQL_MODE, SQL_MODE='NO_AUTO_VALUE_ON_ZERO' */;
-/*!40111 SET @OLD_SQL_NOTES=@@SQL_NOTES, SQL_NOTES=0 */;
-
-
--- Tables préfixées "chat_"
-CREATE TABLE chat_rooms (
- id VARCHAR(36) PRIMARY KEY,
- title VARCHAR(255),
- type ENUM('private', 'group', 'broadcast'),
- created_at TIMESTAMP,
- created_by INT
-);
-
-CREATE TABLE chat_messages (
- id VARCHAR(36) PRIMARY KEY,
- room_id VARCHAR(36),
- content TEXT,
- sender_id INT,
- sent_at TIMESTAMP,
- FOREIGN KEY (room_id) REFERENCES chat_rooms(id)
-);
-
-CREATE TABLE chat_participants (
- room_id VARCHAR(36),
- user_id INT,
- role INT,
- entite_id INT,
- joined_at TIMESTAMP,
- PRIMARY KEY (room_id, user_id)
-);
-
-CREATE TABLE chat_read_receipts (
- message_id VARCHAR(36),
- user_id INT,
- read_at TIMESTAMP,
- PRIMARY KEY (message_id, user_id)
-);
-
-CREATE TABLE `email_counter` (
- `id` int(10) unsigned NOT NULL DEFAULT 1,
- `hour_start` timestamp NULL DEFAULT NULL,
- `count` int(10) unsigned DEFAULT 0,
- PRIMARY KEY (`id`)
-) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
-
-CREATE TABLE `email_queue` (
- `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
- `fk_pass` int(10) unsigned NOT NULL DEFAULT 0,
- `to_email` varchar(255) DEFAULT NULL,
- `subject` varchar(255) DEFAULT NULL,
- `body` text DEFAULT NULL,
- `headers` text DEFAULT NULL,
- `created_at` timestamp NULL DEFAULT current_timestamp(),
- `status` enum('pending','sent','failed') DEFAULT 'pending',
- `sent_at` timestamp NULL DEFAULT NULL,
- `attempts` int(10) unsigned DEFAULT 0,
- `error_message` text DEFAULT NULL,
- PRIMARY KEY (`id`)
-) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
-
-CREATE TABLE `entites` (
- `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
- `encrypted_name` varchar(255) DEFAULT NULL,
- `adresse1` varchar(45) DEFAULT '',
- `adresse2` varchar(45) DEFAULT '',
- `code_postal` varchar(5) DEFAULT '',
- `ville` varchar(45) DEFAULT '',
- `fk_region` int(10) unsigned DEFAULT NULL,
- `fk_type` int(10) unsigned DEFAULT 1,
- `encrypted_phone` varchar(128) DEFAULT '',
- `encrypted_mobile` varchar(128) DEFAULT '',
- `encrypted_email` varchar(255) DEFAULT '',
- `gps_lat` varchar(20) NOT NULL DEFAULT '',
- `gps_lng` varchar(20) NOT NULL DEFAULT '',
- `chk_stripe` tinyint(1) unsigned DEFAULT 0,
- `encrypted_stripe_id` varchar(255) DEFAULT '',
- `encrypted_iban` varchar(255) DEFAULT '',
- `encrypted_bic` varchar(128) DEFAULT '',
- `chk_demo` tinyint(1) unsigned DEFAULT 1,
- `chk_mdp_manuel` tinyint(1) unsigned NOT NULL DEFAULT 0 COMMENT 'Gestion des mots de passe manuelle (1) ou automatique (0)',
- `chk_username_manuel` tinyint(1) unsigned NOT NULL DEFAULT 0 COMMENT 'Gestion des usernames manuelle (1) ou automatique (0)',
- `chk_copie_mail_recu` tinyint(1) unsigned NOT NULL DEFAULT 0,
- `chk_accept_sms` tinyint(1) unsigned NOT NULL DEFAULT 0,
- `created_at` timestamp NOT NULL DEFAULT current_timestamp() COMMENT 'Date de création',
- `fk_user_creat` int(10) unsigned DEFAULT NULL,
- `updated_at` timestamp NULL DEFAULT NULL ON UPDATE current_timestamp() COMMENT 'Date de modification',
- `fk_user_modif` int(10) unsigned DEFAULT NULL,
- `chk_active` tinyint(1) unsigned DEFAULT 1,
- PRIMARY KEY (`id`),
- KEY `entites_ibfk_1` (`fk_region`),
- KEY `entites_ibfk_2` (`fk_type`),
- CONSTRAINT `entites_ibfk_1` FOREIGN KEY (`fk_region`) REFERENCES `x_regions` (`id`) ON UPDATE CASCADE,
- CONSTRAINT `entites_ibfk_2` FOREIGN KEY (`fk_type`) REFERENCES `x_entites_types` (`id`) ON UPDATE CASCADE
-) ENGINE=InnoDB AUTO_INCREMENT=1230 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
-
-CREATE TABLE `medias` (
- `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
- `support` varchar(45) NOT NULL DEFAULT '' COMMENT 'Type de support (entite, user, operation, passage)',
- `support_id` int(10) unsigned NOT NULL DEFAULT 0 COMMENT 'ID de élément associé',
- `fichier` varchar(250) NOT NULL DEFAULT '' COMMENT 'Nom du fichier stocké',
- `file_type` varchar(50) DEFAULT NULL COMMENT 'Extension du fichier (pdf, jpg, xlsx, etc.)',
- `file_category` varchar(50) DEFAULT NULL COMMENT 'export, logo, carte, etc.',
- `file_size` int(10) unsigned DEFAULT NULL COMMENT 'Taille du fichier en octets',
- `mime_type` varchar(100) DEFAULT NULL COMMENT 'Type MIME du fichier',
- `original_name` varchar(255) DEFAULT NULL COMMENT 'Nom original du fichier uploadé',
- `fk_entite` int(10) unsigned DEFAULT NULL COMMENT 'ID de entité propriétaire',
- `fk_operation` int(10) unsigned DEFAULT NULL COMMENT 'ID de opération (pour passages)',
- `file_path` varchar(500) DEFAULT NULL COMMENT 'Chemin complet du fichier',
- `original_width` int(10) unsigned DEFAULT NULL COMMENT 'Largeur originale de image',
- `original_height` int(10) unsigned DEFAULT NULL COMMENT 'Hauteur originale de image',
- `processed_width` int(10) unsigned DEFAULT NULL COMMENT 'Largeur après traitement',
- `processed_height` int(10) unsigned DEFAULT NULL COMMENT 'Hauteur après traitement',
- `is_processed` tinyint(1) unsigned DEFAULT 0 COMMENT 'Image redimensionnée (1) ou originale (0)',
- `description` varchar(100) NOT NULL DEFAULT '' COMMENT 'Description du fichier',
- `created_at` timestamp NOT NULL DEFAULT current_timestamp(),
- `fk_user_creat` int(10) unsigned NOT NULL DEFAULT 0,
- `updated_at` timestamp NULL DEFAULT NULL ON UPDATE current_timestamp(),
- `fk_user_modif` int(10) unsigned NOT NULL DEFAULT 0,
- PRIMARY KEY (`id`),
- UNIQUE KEY `id_UNIQUE` (`id`),
- KEY `idx_entite` (`fk_entite`),
- KEY `idx_operation` (`fk_operation`),
- KEY `idx_support_type` (`support`, `support_id`),
- KEY `idx_file_type` (`file_type`),
- KEY `idx_file_category` (`file_category`),
- CONSTRAINT `fk_medias_entite` FOREIGN KEY (`fk_entite`) REFERENCES `entites` (`id`) ON UPDATE CASCADE ON DELETE CASCADE,
- CONSTRAINT `fk_medias_operation` FOREIGN KEY (`fk_operation`) REFERENCES `operations` (`id`) ON UPDATE CASCADE ON DELETE CASCADE
-) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-
-CREATE TABLE `ope_pass` (
- `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
- `fk_operation` int(10) unsigned NOT NULL DEFAULT 0,
- `fk_sector` int(10) unsigned DEFAULT 0,
- `fk_user` int(10) unsigned NOT NULL DEFAULT 0,
- `fk_adresse` varchar(25) DEFAULT '' COMMENT 'adresses.cp??.id',
- `passed_at` timestamp NULL DEFAULT NULL COMMENT 'Date du passage',
- `fk_type` int(10) unsigned DEFAULT 0,
- `numero` varchar(10) NOT NULL DEFAULT '',
- `rue` varchar(75) NOT NULL DEFAULT '',
- `rue_bis` varchar(1) NOT NULL DEFAULT '',
- `ville` varchar(75) NOT NULL DEFAULT '',
- `fk_habitat` int(10) unsigned DEFAULT 1,
- `appt` varchar(5) DEFAULT '',
- `niveau` varchar(5) DEFAULT '',
- `residence` varchar(75) DEFAULT '',
- `gps_lat` varchar(20) NOT NULL DEFAULT '',
- `gps_lng` varchar(20) NOT NULL DEFAULT '',
- `encrypted_name` varchar(255) NOT NULL DEFAULT '',
- `montant` decimal(7,2) NOT NULL DEFAULT 0.00,
- `fk_type_reglement` int(10) unsigned DEFAULT 1,
- `remarque` text DEFAULT '',
- `encrypted_email` varchar(255) DEFAULT '',
- `nom_recu` varchar(50) DEFAULT NULL,
- `date_recu` timestamp NULL DEFAULT NULL COMMENT 'Date de réception',
- `date_creat_recu` timestamp NULL DEFAULT NULL COMMENT 'Date de création du reçu',
- `date_sent_recu` timestamp NULL DEFAULT NULL COMMENT 'Date envoi du reçu',
- `email_erreur` varchar(30) DEFAULT '',
- `chk_email_sent` tinyint(1) unsigned NOT NULL DEFAULT 0,
- `encrypted_phone` varchar(128) NOT NULL DEFAULT '',
- `is_striped` tinyint(1) unsigned NOT NULL DEFAULT 0,
- `docremis` tinyint(1) unsigned DEFAULT 0,
- `date_repasser` timestamp NULL DEFAULT NULL COMMENT 'Date prévue pour repasser',
- `nb_passages` int(11) DEFAULT 1 COMMENT 'Nb passages pour les a repasser',
- `chk_gps_maj` tinyint(1) unsigned DEFAULT 0,
- `chk_map_create` tinyint(1) unsigned DEFAULT 0,
- `chk_mobile` tinyint(1) unsigned DEFAULT 0,
- `chk_synchro` tinyint(1) unsigned DEFAULT 1 COMMENT 'chk synchro entre web et appli',
- `chk_api_adresse` tinyint(1) unsigned DEFAULT 0,
- `chk_maj_adresse` tinyint(1) unsigned DEFAULT 0,
- `anomalie` tinyint(1) unsigned DEFAULT 0,
- `created_at` timestamp NOT NULL DEFAULT current_timestamp() COMMENT 'Date de création',
- `fk_user_creat` int(10) unsigned DEFAULT NULL,
- `updated_at` timestamp NULL DEFAULT NULL ON UPDATE current_timestamp() COMMENT 'Date de modification',
- `fk_user_modif` int(10) unsigned DEFAULT NULL,
- `chk_active` tinyint(1) unsigned NOT NULL DEFAULT 1,
- PRIMARY KEY (`id`),
- KEY `fk_operation` (`fk_operation`),
- KEY `fk_sector` (`fk_sector`),
- KEY `fk_user` (`fk_user`),
- KEY `fk_type` (`fk_type`),
- KEY `fk_type_reglement` (`fk_type_reglement`),
- KEY `email` (`encrypted_email`),
- CONSTRAINT `ope_pass_ibfk_1` FOREIGN KEY (`fk_operation`) REFERENCES `operations` (`id`) ON UPDATE CASCADE,
- CONSTRAINT `ope_pass_ibfk_2` FOREIGN KEY (`fk_sector`) REFERENCES `ope_sectors` (`id`) ON UPDATE CASCADE,
- CONSTRAINT `ope_pass_ibfk_3` FOREIGN KEY (`fk_user`) REFERENCES `users` (`id`) ON UPDATE CASCADE,
- CONSTRAINT `ope_pass_ibfk_4` FOREIGN KEY (`fk_type_reglement`) REFERENCES `x_types_reglements` (`id`) ON UPDATE CASCADE
-) ENGINE=InnoDB AUTO_INCREMENT=19499566 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
-
-CREATE TABLE `ope_pass_histo` (
- `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
- `fk_pass` int(10) unsigned NOT NULL DEFAULT 0,
- `date_histo` timestamp NOT NULL DEFAULT current_timestamp() COMMENT 'Date historique',
- `sujet` varchar(50) DEFAULT NULL,
- `remarque` varchar(250) NOT NULL DEFAULT '',
- PRIMARY KEY (`id`),
- KEY `ope_pass_histo_fk_pass_IDX` (`fk_pass`) USING BTREE,
- KEY `ope_pass_histo_date_histo_IDX` (`date_histo`) USING BTREE,
- CONSTRAINT `ope_pass_histo_ibfk_1` FOREIGN KEY (`fk_pass`) REFERENCES `ope_pass` (`id`) ON DELETE CASCADE ON UPDATE CASCADE
-) ENGINE=InnoDB AUTO_INCREMENT=6752 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
-
-CREATE TABLE `ope_sectors` (
- `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
- `fk_operation` int(10) unsigned NOT NULL DEFAULT 0,
- `fk_old_sector` int(10) unsigned DEFAULT NULL,
- `libelle` varchar(75) NOT NULL DEFAULT '',
- `sector` text NOT NULL DEFAULT '',
- `color` varchar(7) NOT NULL DEFAULT '#4B77BE',
- `created_at` timestamp NOT NULL DEFAULT current_timestamp() COMMENT 'Date de création',
- `fk_user_creat` int(10) unsigned NOT NULL DEFAULT 0,
- `updated_at` timestamp NULL DEFAULT NULL ON UPDATE current_timestamp() COMMENT 'Date de modification',
- `fk_user_modif` int(10) unsigned NOT NULL DEFAULT 0,
- `chk_active` tinyint(1) unsigned NOT NULL DEFAULT 1,
- PRIMARY KEY (`id`),
- UNIQUE KEY `id` (`id`),
- KEY `fk_operation` (`fk_operation`),
- CONSTRAINT `ope_sectors_ibfk_1` FOREIGN KEY (`fk_operation`) REFERENCES `operations` (`id`) ON UPDATE CASCADE
-) ENGINE=InnoDB AUTO_INCREMENT=27675 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
-
-CREATE TABLE `ope_users` (
- `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
- `fk_operation` int(10) unsigned NOT NULL DEFAULT 0,
- `fk_user` int(10) unsigned NOT NULL DEFAULT 0,
- `created_at` timestamp NOT NULL DEFAULT current_timestamp() COMMENT 'Date de création',
- `fk_user_creat` int(10) unsigned DEFAULT NULL,
- `updated_at` timestamp NULL DEFAULT NULL ON UPDATE current_timestamp() COMMENT 'Date de modification',
- `fk_user_modif` int(10) unsigned DEFAULT NULL,
- `chk_active` tinyint(1) unsigned NOT NULL DEFAULT 1,
- PRIMARY KEY (`id`),
- UNIQUE KEY `id_UNIQUE` (`id`),
- KEY `ope_users_ibfk_1` (`fk_operation`),
- KEY `ope_users_ibfk_2` (`fk_user`),
- CONSTRAINT `ope_users_ibfk_1` FOREIGN KEY (`fk_operation`) REFERENCES `operations` (`id`) ON UPDATE CASCADE,
- CONSTRAINT `ope_users_ibfk_2` FOREIGN KEY (`fk_user`) REFERENCES `users` (`id`) ON UPDATE CASCADE
-) ENGINE=InnoDB AUTO_INCREMENT=199006 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
-
-CREATE TABLE `ope_users_sectors` (
- `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
- `fk_operation` int(10) unsigned NOT NULL DEFAULT 0,
- `fk_user` int(10) unsigned NOT NULL DEFAULT 0,
- `fk_sector` int(10) unsigned NOT NULL DEFAULT 0,
- `created_at` timestamp NOT NULL DEFAULT current_timestamp() COMMENT 'Date de création',
- `fk_user_creat` int(10) unsigned NOT NULL DEFAULT 0,
- `updated_at` timestamp NULL DEFAULT NULL ON UPDATE current_timestamp() COMMENT 'Date de modification',
- `fk_user_modif` int(10) unsigned DEFAULT NULL,
- `chk_active` tinyint(1) unsigned DEFAULT 1,
- PRIMARY KEY (`id`),
- UNIQUE KEY `id` (`id`),
- KEY `fk_operation` (`fk_operation`),
- KEY `fk_user` (`fk_user`),
- KEY `fk_sector` (`fk_sector`),
- CONSTRAINT `ope_users_sectors_ibfk_1` FOREIGN KEY (`fk_operation`) REFERENCES `operations` (`id`) ON UPDATE CASCADE,
- CONSTRAINT `ope_users_sectors_ibfk_2` FOREIGN KEY (`fk_user`) REFERENCES `users` (`id`) ON UPDATE CASCADE,
- CONSTRAINT `ope_users_sectors_ibfk_3` FOREIGN KEY (`fk_sector`) REFERENCES `ope_sectors` (`id`) ON UPDATE CASCADE
-) ENGINE=InnoDB AUTO_INCREMENT=48082 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
-
-CREATE TABLE `operations` (
- `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
- `fk_entite` int(10) unsigned NOT NULL DEFAULT 1,
- `libelle` varchar(75) NOT NULL DEFAULT '',
- `date_deb` date NOT NULL DEFAULT '0000-00-00',
- `date_fin` date NOT NULL DEFAULT '0000-00-00',
- `chk_distinct_sectors` tinyint(1) unsigned NOT NULL DEFAULT 0,
- `created_at` timestamp NOT NULL DEFAULT current_timestamp() COMMENT 'Date de création',
- `fk_user_creat` int(10) unsigned NOT NULL DEFAULT 0,
- `updated_at` timestamp NULL DEFAULT NULL ON UPDATE current_timestamp() COMMENT 'Date de modification',
- `fk_user_modif` int(10) unsigned NOT NULL DEFAULT 0,
- `chk_active` tinyint(1) unsigned NOT NULL DEFAULT 1,
- PRIMARY KEY (`id`),
- KEY `fk_entite` (`fk_entite`),
- KEY `date_deb` (`date_deb`),
- CONSTRAINT `operations_ibfk_1` FOREIGN KEY (`fk_entite`) REFERENCES `entites` (`id`) ON UPDATE CASCADE
-) ENGINE=InnoDB AUTO_INCREMENT=3121 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
-
-CREATE TABLE `params` (
- `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
- `libelle` varchar(35) NOT NULL DEFAULT '',
- `valeur` varchar(255) NOT NULL DEFAULT '',
- `aide` varchar(150) NOT NULL DEFAULT '',
- PRIMARY KEY (`id`)
-) ENGINE=InnoDB AUTO_INCREMENT=2 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
-
-CREATE TABLE `sectors_adresses` (
- `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
- `fk_adresse` varchar(25) DEFAULT NULL COMMENT 'adresses.cp??.id',
- `osm_id` int(10) unsigned NOT NULL DEFAULT 0,
- `fk_sector` int(10) unsigned NOT NULL DEFAULT 0,
- `osm_name` varchar(50) NOT NULL DEFAULT '',
- `numero` varchar(5) NOT NULL DEFAULT '',
- `rue_bis` varchar(5) NOT NULL DEFAULT '',
- `rue` varchar(60) NOT NULL DEFAULT '',
- `cp` varchar(5) NOT NULL DEFAULT '',
- `ville` varchar(60) NOT NULL DEFAULT '',
- `gps_lat` varchar(20) NOT NULL DEFAULT '',
- `gps_lng` varchar(20) NOT NULL DEFAULT '',
- `osm_date_creat` timestamp NOT NULL DEFAULT '0000-00-00 00:00:00',
- `created_at` timestamp NOT NULL DEFAULT current_timestamp() COMMENT 'Date de création',
- `updated_at` timestamp NULL DEFAULT NULL ON UPDATE current_timestamp() COMMENT 'Date de modification',
- PRIMARY KEY (`id`),
- KEY `sectors_adresses_fk_sector_index` (`fk_sector`),
- KEY `sectors_adresses_numero_index` (`numero`),
- KEY `sectors_adresses_rue_index` (`rue`),
- KEY `sectors_adresses_ville_index` (`ville`),
- CONSTRAINT `sectors_adresses_ibfk_1` FOREIGN KEY (`fk_sector`) REFERENCES `ope_sectors` (`id`) ON UPDATE CASCADE
-) ENGINE=InnoDB AUTO_INCREMENT=1562946 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
-
-CREATE TABLE `users` (
- `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
- `fk_entite` int(10) unsigned DEFAULT 1,
- `fk_role` int(10) unsigned DEFAULT 1,
- `fk_titre` int(10) unsigned DEFAULT 1,
- `encrypted_name` varchar(255) DEFAULT NULL,
- `first_name` varchar(45) DEFAULT NULL,
- `sect_name` varchar(60) DEFAULT '',
- `encrypted_user_name` varchar(128) DEFAULT '',
- `user_pass_hash` varchar(60) DEFAULT NULL,
- `encrypted_phone` varchar(128) DEFAULT NULL,
- `encrypted_mobile` varchar(128) DEFAULT NULL,
- `encrypted_email` varchar(255) DEFAULT '',
- `chk_alert_email` tinyint(1) unsigned DEFAULT 1,
- `chk_suivi` tinyint(1) unsigned DEFAULT 0,
- `date_naissance` date DEFAULT NULL,
- `date_embauche` date DEFAULT NULL,
- `created_at` timestamp NOT NULL DEFAULT current_timestamp() COMMENT 'Date de création',
- `fk_user_creat` int(10) unsigned DEFAULT NULL,
- `updated_at` timestamp NULL DEFAULT NULL ON UPDATE current_timestamp() COMMENT 'Date de modification',
- `fk_user_modif` int(10) unsigned DEFAULT NULL,
- `chk_active` tinyint(1) unsigned DEFAULT 1,
- PRIMARY KEY (`id`),
- KEY `fk_entite` (`fk_entite`),
- KEY `username` (`encrypted_user_name`),
- KEY `users_ibfk_2` (`fk_role`),
- KEY `users_ibfk_3` (`fk_titre`),
- CONSTRAINT `users_ibfk_1` FOREIGN KEY (`fk_entite`) REFERENCES `entites` (`id`) ON UPDATE CASCADE,
- CONSTRAINT `users_ibfk_2` FOREIGN KEY (`fk_role`) REFERENCES `x_users_roles` (`id`) ON UPDATE CASCADE,
- CONSTRAINT `users_ibfk_3` FOREIGN KEY (`fk_titre`) REFERENCES `x_users_titres` (`id`) ON UPDATE CASCADE
-) ENGINE=InnoDB AUTO_INCREMENT=10027748 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
-
-CREATE TABLE `x_departements` (
- `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
- `code` varchar(3) DEFAULT NULL,
- `fk_region` int(10) unsigned DEFAULT 1,
- `libelle` varchar(45) DEFAULT NULL,
- `chk_active` tinyint(1) unsigned DEFAULT 1,
- PRIMARY KEY (`id`),
- UNIQUE KEY `id_UNIQUE` (`id`),
- KEY `x_departements_ibfk_1` (`fk_region`),
- CONSTRAINT `x_departements_ibfk_1` FOREIGN KEY (`fk_region`) REFERENCES `x_regions` (`id`) ON UPDATE CASCADE
-) ENGINE=InnoDB AUTO_INCREMENT=105 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
-
-CREATE TABLE `x_devises` (
- `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
- `code` varchar(3) DEFAULT NULL,
- `symbole` varchar(6) DEFAULT NULL,
- `libelle` varchar(45) DEFAULT NULL,
- `chk_active` tinyint(1) unsigned DEFAULT 1,
- PRIMARY KEY (`id`),
- UNIQUE KEY `id_UNIQUE` (`id`)
-) ENGINE=InnoDB AUTO_INCREMENT=2 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
-
-CREATE TABLE `x_entites_types` (
- `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
- `libelle` varchar(45) DEFAULT NULL,
- `chk_active` tinyint(1) unsigned DEFAULT NULL,
- PRIMARY KEY (`id`),
- UNIQUE KEY `id_UNIQUE` (`id`)
-) ENGINE=InnoDB AUTO_INCREMENT=3 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
-
-CREATE TABLE `x_pays` (
- `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
- `code` varchar(3) DEFAULT NULL,
- `fk_continent` int(10) unsigned DEFAULT NULL,
- `fk_devise` int(10) unsigned DEFAULT 1,
- `libelle` varchar(45) DEFAULT NULL,
- `chk_active` tinyint(1) unsigned DEFAULT 1,
- PRIMARY KEY (`id`),
- UNIQUE KEY `id_UNIQUE` (`id`),
- KEY `x_pays_ibfk_1` (`fk_devise`),
- CONSTRAINT `x_pays_ibfk_1` FOREIGN KEY (`fk_devise`) REFERENCES `x_devises` (`id`) ON UPDATE CASCADE
-) ENGINE=InnoDB AUTO_INCREMENT=10 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci COMMENT='Table des pays avec leurs codes' `PAGE_COMPRESSED`='ON';
-
-CREATE TABLE `x_regions` (
- `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
- `fk_pays` int(10) unsigned DEFAULT 1,
- `libelle` varchar(45) DEFAULT NULL,
- `libelle_long` varchar(45) DEFAULT NULL,
- `table_osm` varchar(45) DEFAULT NULL,
- `departements` varchar(45) DEFAULT NULL,
- `chk_active` tinyint(1) unsigned DEFAULT 1,
- PRIMARY KEY (`id`),
- UNIQUE KEY `id_UNIQUE` (`id`),
- KEY `x_regions_ibfk_1` (`fk_pays`),
- CONSTRAINT `x_regions_ibfk_1` FOREIGN KEY (`fk_pays`) REFERENCES `x_pays` (`id`) ON UPDATE CASCADE
-) ENGINE=InnoDB AUTO_INCREMENT=29 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
-
-CREATE TABLE `x_types_passages` (
- `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
- `libelle` varchar(10) DEFAULT NULL,
- `color_button` varchar(15) DEFAULT NULL,
- `color_mark` varchar(15) DEFAULT NULL,
- `color_table` varchar(15) DEFAULT NULL,
- `chk_active` tinyint(1) unsigned NOT NULL DEFAULT 1,
- PRIMARY KEY (`id`)
-) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
-
-CREATE TABLE `x_types_reglements` (
- `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
- `libelle` varchar(45) DEFAULT NULL,
- `chk_active` tinyint(1) unsigned DEFAULT 1,
- PRIMARY KEY (`id`)
-) ENGINE=InnoDB AUTO_INCREMENT=5 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
-
-CREATE TABLE `x_users_roles` (
- `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
- `libelle` varchar(45) DEFAULT NULL,
- `chk_active` tinyint(1) unsigned DEFAULT 1,
- PRIMARY KEY (`id`),
- UNIQUE KEY `id_UNIQUE` (`id`)
-) ENGINE=InnoDB AUTO_INCREMENT=10 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci COMMENT='Les différents rôles des utilisateurs' `PAGE_COMPRESSED`='ON';
-
-CREATE TABLE `x_users_titres` (
- `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
- `libelle` varchar(45) DEFAULT NULL,
- `chk_active` tinyint(1) unsigned DEFAULT 1,
- PRIMARY KEY (`id`),
- UNIQUE KEY `id_UNIQUE` (`id`)
-) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci COMMENT='Les différents titres des utilisateurs' `PAGE_COMPRESSED`='ON';
-
-CREATE TABLE `x_villes` (
- `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
- `fk_departement` int(10) unsigned DEFAULT 1,
- `libelle` varchar(65) DEFAULT NULL,
- `code_postal` varchar(5) DEFAULT NULL,
- `code_insee` varchar(5) DEFAULT NULL,
- `chk_active` tinyint(1) unsigned DEFAULT 1,
- PRIMARY KEY (`id`),
- UNIQUE KEY `id_UNIQUE` (`id`),
- KEY `x_villes_ibfk_1` (`fk_departement`),
- CONSTRAINT `x_villes_ibfk_1` FOREIGN KEY (`fk_departement`) REFERENCES `x_departements` (`id`) ON UPDATE CASCADE
-) ENGINE=InnoDB AUTO_INCREMENT=38950 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
-
-CREATE TABLE `z_sessions` (
- `sid` text NOT NULL,
- `fk_user` int(11) NOT NULL,
- `role` varchar(10) DEFAULT NULL,
- `date_modified` timestamp NOT NULL DEFAULT current_timestamp() ON UPDATE current_timestamp(),
- `ip` varchar(50) NOT NULL,
- `browser` varchar(150) NOT NULL,
- `data` mediumtext DEFAULT NULL
-) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
-
-CREATE ALGORITHM=UNDEFINED DEFINER=`root`@`%` SQL SECURITY DEFINER VIEW `chat_conversations_unread` AS select `r`.`id` AS `id`,`r`.`type` AS `type`,`r`.`title` AS `title`,`r`.`date_creation` AS `date_creation`,`r`.`fk_user` AS `fk_user`,`r`.`fk_entite` AS `fk_entite`,`r`.`statut` AS `statut`,`r`.`description` AS `description`,`r`.`reply_permission` AS `reply_permission`,`r`.`is_pinned` AS `is_pinned`,`r`.`expiry_date` AS `expiry_date`,`r`.`updated_at` AS `updated_at`,count(distinct `m`.`id`) AS `total_messages`,count(distinct `rm`.`id`) AS `read_messages`,count(distinct `m`.`id`) - count(distinct `rm`.`id`) AS `unread_messages`,(select `geo_app`.`chat_messages`.`date_sent` from `chat_messages` where `geo_app`.`chat_messages`.`fk_room` = `r`.`id` order by `geo_app`.`chat_messages`.`date_sent` desc limit 1) AS `last_message_date` from ((`chat_rooms` `r` left join `chat_messages` `m` on(`r`.`id` = `m`.`fk_room`)) left join `chat_read_messages` `rm` on(`m`.`id` = `rm`.`fk_message`)) group by `r`.`id`;
-
-
-/*!40101 SET SQL_MODE=@OLD_SQL_MODE */;
-/*!40014 SET FOREIGN_KEY_CHECKS=@OLD_FOREIGN_KEY_CHECKS */;
-/*!40014 SET UNIQUE_CHECKS=@OLD_UNIQUE_CHECKS */;
-/*!40101 SET CHARACTER_SET_CLIENT=@OLD_CHARACTER_SET_CLIENT */;
-/*!40101 SET CHARACTER_SET_RESULTS=@OLD_CHARACTER_SET_RESULTS */;
-/*!40101 SET COLLATION_CONNECTION=@OLD_COLLATION_CONNECTION */;
-/*!40111 SET SQL_NOTES=@OLD_SQL_NOTES */;
\ No newline at end of file
diff --git a/api/docs/geosector_app.sql b/api/docs/geosector_app.sql
deleted file mode 100755
index 3d248c3a..00000000
--- a/api/docs/geosector_app.sql
+++ /dev/null
@@ -1,621 +0,0 @@
--- Création de la base de données geo_app si elle n'existe pas
-DROP DATABASE IF EXISTS `geo_app`;
-CREATE DATABASE IF NOT EXISTS `geo_app` DEFAULT CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;
-
--- Création de l'utilisateur et attribution des droits
-CREATE USER IF NOT EXISTS 'geo_app_user'@'localhost' IDENTIFIED BY 'QO:96df*?k{4W6m';
-GRANT SELECT, INSERT, UPDATE, DELETE ON `geo_app`.* TO 'geo_app_user'@'localhost';
-FLUSH PRIVILEGES;
-
-USE geo_app;
-
---
--- Table structure for table `email_counter`
---
-
-DROP TABLE IF EXISTS `email_counter`;
-/*!40101 SET @saved_cs_client = @@character_set_client */;
-/*!50503 SET character_set_client = utf8mb4 */;
-CREATE TABLE `email_counter` (
- `id` int unsigned NOT NULL DEFAULT '1',
- `hour_start` timestamp NULL DEFAULT NULL,
- `count` int unsigned DEFAULT '0',
- PRIMARY KEY (`id`)
-) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-/*!40101 SET character_set_client = @saved_cs_client */;
-
-DROP TABLE IF EXISTS `x_devises`;
-/*!40101 SET @saved_cs_client = @@character_set_client */;
-/*!50503 SET character_set_client = utf8mb4 */;
-CREATE TABLE `x_devises` (
- `id` int unsigned NOT NULL AUTO_INCREMENT,
- `code` varchar(3) DEFAULT NULL,
- `symbole` varchar(6) DEFAULT NULL,
- `libelle` varchar(45) DEFAULT NULL,
- `chk_active` tinyint(1) unsigned DEFAULT '1',
- PRIMARY KEY (`id`),
- UNIQUE KEY `id_UNIQUE` (`id`)
-) ENGINE=InnoDB AUTO_INCREMENT=2 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-/*!40101 SET character_set_client = @saved_cs_client */;
-
---
--- Table structure for table `x_entites_types`
---
-
-DROP TABLE IF EXISTS `x_entites_types`;
-/*!40101 SET @saved_cs_client = @@character_set_client */;
-/*!50503 SET character_set_client = utf8mb4 */;
-CREATE TABLE `x_entites_types` (
- `id` int unsigned NOT NULL AUTO_INCREMENT,
- `libelle` varchar(45) DEFAULT NULL,
- `chk_active` tinyint(1) unsigned DEFAULT NULL,
- PRIMARY KEY (`id`),
- UNIQUE KEY `id_UNIQUE` (`id`)
-) ENGINE=InnoDB AUTO_INCREMENT=3 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-/*!40101 SET character_set_client = @saved_cs_client */;
-
---
--- Table structure for table `x_types_passages`
---
-
-DROP TABLE IF EXISTS `x_types_passages`;
-/*!40101 SET @saved_cs_client = @@character_set_client */;
-/*!50503 SET character_set_client = utf8mb4 */;
-CREATE TABLE `x_types_passages` (
- `id` int unsigned NOT NULL AUTO_INCREMENT,
- `libelle` varchar(10) CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci DEFAULT NULL,
- `color_button` varchar(15) CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci DEFAULT NULL,
- `color_mark` varchar(15) CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci DEFAULT NULL,
- `color_table` varchar(15) CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci DEFAULT NULL,
- `chk_active` tinyint(1) unsigned NOT NULL DEFAULT '1',
- PRIMARY KEY (`id`)
-) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-/*!40101 SET character_set_client = @saved_cs_client */;
-
---
--- Table structure for table `x_types_reglements`
---
-
-DROP TABLE IF EXISTS `x_types_reglements`;
-/*!40101 SET @saved_cs_client = @@character_set_client */;
-/*!50503 SET character_set_client = utf8mb4 */;
-CREATE TABLE `x_types_reglements` (
- `id` int unsigned NOT NULL AUTO_INCREMENT,
- `libelle` varchar(45) DEFAULT NULL,
- `chk_active` tinyint(1) unsigned DEFAULT '1',
- PRIMARY KEY (`id`)
-) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-/*!40101 SET character_set_client = @saved_cs_client */;
-
---
--- Table structure for table `x_users_roles`
---
-
-DROP TABLE IF EXISTS `x_users_roles`;
-
-CREATE TABLE `x_users_roles` (
- `id` int unsigned NOT NULL AUTO_INCREMENT,
- `libelle` varchar(45) DEFAULT NULL,
- `chk_active` tinyint(1) unsigned DEFAULT '1',
- PRIMARY KEY (`id`),
- UNIQUE KEY `id_UNIQUE` (`id`)
-) ENGINE=InnoDB AUTO_INCREMENT=10 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci COMMENT='Les différents rôles des utilisateurs';
-/*!40101 SET character_set_client = @saved_cs_client */;
-
-DROP TABLE IF EXISTS `x_users_titres`;
-/*!40101 SET @saved_cs_client = @@character_set_client */;
-/*!50503 SET character_set_client = utf8mb4 */;
-CREATE TABLE `x_users_titres` (
- `id` int unsigned NOT NULL AUTO_INCREMENT,
- `libelle` varchar(45) DEFAULT NULL,
- `chk_active` tinyint(1) unsigned DEFAULT '1',
- PRIMARY KEY (`id`),
- UNIQUE KEY `id_UNIQUE` (`id`)
-) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci COMMENT='Les différents titres des utilisateurs';
-
-DROP TABLE IF EXISTS `x_pays`;
-/*!40101 SET @saved_cs_client = @@character_set_client */;
-/*!50503 SET character_set_client = utf8mb4 */;
-CREATE TABLE `x_pays` (
- `id` int unsigned NOT NULL AUTO_INCREMENT,
- `code` varchar(3) DEFAULT NULL,
- `fk_continent` int unsigned DEFAULT NULL,
- `fk_devise` int unsigned DEFAULT '1',
- `libelle` varchar(45) DEFAULT NULL,
- `chk_active` tinyint(1) unsigned DEFAULT '1',
- PRIMARY KEY (`id`),
- UNIQUE KEY `id_UNIQUE` (`id`),
- CONSTRAINT `x_pays_ibfk_1` FOREIGN KEY (`fk_devise`) REFERENCES `x_devises` (`id`) ON DELETE RESTRICT ON UPDATE CASCADE
-) ENGINE=InnoDB AUTO_INCREMENT=10 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci COMMENT='Table des pays avec leurs codes';
-/*!40101 SET character_set_client = @saved_cs_client */;
-
-
-DROP TABLE IF EXISTS `x_regions`;
-/*!40101 SET @saved_cs_client = @@character_set_client */;
-/*!50503 SET character_set_client = utf8mb4 */;
-CREATE TABLE `x_regions` (
- `id` int unsigned NOT NULL AUTO_INCREMENT,
- `fk_pays` int unsigned DEFAULT '1',
- `libelle` varchar(45) DEFAULT NULL,
- `libelle_long` varchar(45) DEFAULT NULL,
- `table_osm` varchar(45) DEFAULT NULL,
- `departements` varchar(45) DEFAULT NULL,
- `chk_active` tinyint(1) unsigned DEFAULT '1',
- PRIMARY KEY (`id`),
- UNIQUE KEY `id_UNIQUE` (`id`),
- CONSTRAINT `x_regions_ibfk_1` FOREIGN KEY (`fk_pays`) REFERENCES `x_pays` (`id`) ON DELETE RESTRICT ON UPDATE CASCADE
-) ENGINE=InnoDB AUTO_INCREMENT=29 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-/*!40101 SET character_set_client = @saved_cs_client */;
-
-DROP TABLE IF EXISTS `x_departements`;
-/*!40101 SET @saved_cs_client = @@character_set_client */;
-/*!50503 SET character_set_client = utf8mb4 */;
-CREATE TABLE `x_departements` (
- `id` int unsigned NOT NULL AUTO_INCREMENT,
- `code` varchar(3) DEFAULT NULL,
- `fk_region` int unsigned DEFAULT '1',
- `libelle` varchar(45) DEFAULT NULL,
- `chk_active` tinyint(1) unsigned DEFAULT '1',
- PRIMARY KEY (`id`),
- UNIQUE KEY `id_UNIQUE` (`id`),
- CONSTRAINT `x_departements_ibfk_1` FOREIGN KEY (`fk_region`) REFERENCES `x_regions` (`id`) ON DELETE RESTRICT ON UPDATE CASCADE
-) ENGINE=InnoDB AUTO_INCREMENT=105 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-/*!40101 SET character_set_client = @saved_cs_client */;
-
-DROP TABLE IF EXISTS `entites`;
-/*!40101 SET @saved_cs_client = @@character_set_client */;
-/*!50503 SET character_set_client = utf8mb4 */;
-CREATE TABLE `entites` (
- `id` int unsigned NOT NULL AUTO_INCREMENT,
- `encrypted_name` varchar(255) DEFAULT NULL,
- `adresse1` varchar(45) DEFAULT '',
- `adresse2` varchar(45) DEFAULT '',
- `code_postal` varchar(5) DEFAULT '',
- `ville` varchar(45) DEFAULT '',
- `fk_region` int unsigned DEFAULT NULL,
- `fk_type` int unsigned DEFAULT '1',
- `encrypted_phone` varchar(128) DEFAULT '',
- `encrypted_mobile` varchar(128) DEFAULT '',
- `encrypted_email` varchar(255) DEFAULT '',
- `gps_lat` varchar(20) NOT NULL DEFAULT '',
- `gps_lng` varchar(20) NOT NULL DEFAULT '',
- `encrypted_stripe_id` varchar(255) DEFAULT '',
- `encrypted_iban` varchar(255) DEFAULT '',
- `encrypted_bic` varchar(128) DEFAULT '',
- `chk_demo` tinyint(1) unsigned DEFAULT '1',
- `chk_mdp_manuel` tinyint(1) unsigned NOT NULL DEFAULT '1' COMMENT 'Gestion des mots de passe manuelle O/N',
- `chk_copie_mail_recu` tinyint(1) unsigned NOT NULL DEFAULT '0',
- `chk_accept_sms` tinyint(1) unsigned NOT NULL DEFAULT '0',
- `created_at` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT 'Date de création',
- `fk_user_creat` int unsigned DEFAULT NULL,
- `updated_at` timestamp NULL DEFAULT NULL ON UPDATE CURRENT_TIMESTAMP COMMENT 'Date de modification',
- `fk_user_modif` int unsigned DEFAULT NULL,
- `chk_active` tinyint(1) unsigned DEFAULT '1',
- PRIMARY KEY (`id`),
- CONSTRAINT `entites_ibfk_1` FOREIGN KEY (`fk_region`) REFERENCES `x_regions` (`id`) ON DELETE RESTRICT ON UPDATE CASCADE,
- CONSTRAINT `entites_ibfk_2` FOREIGN KEY (`fk_type`) REFERENCES `x_entites_types` (`id`) ON DELETE RESTRICT ON UPDATE CASCADE
-) ENGINE=InnoDB AUTO_INCREMENT=0 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-/*!40101 SET character_set_client = @saved_cs_client */;
-
-DROP TABLE IF EXISTS `x_villes`;
-/*!40101 SET @saved_cs_client = @@character_set_client */;
-/*!50503 SET character_set_client = utf8mb4 */;
-CREATE TABLE `x_villes` (
- `id` int unsigned NOT NULL AUTO_INCREMENT,
- `fk_departement` int unsigned DEFAULT '1',
- `libelle` varchar(65) DEFAULT NULL,
- `cp` varchar(5) DEFAULT NULL,
- `code_insee` varchar(5) DEFAULT NULL,
- `departement` varchar(65) DEFAULT NULL,
- `chk_active` tinyint(1) unsigned DEFAULT '1',
- PRIMARY KEY (`id`),
- UNIQUE KEY `id_UNIQUE` (`id`),
- CONSTRAINT `x_villes_ibfk_1` FOREIGN KEY (`fk_departement`) REFERENCES `x_departements` (`id`) ON DELETE RESTRICT ON UPDATE CASCADE
-) ENGINE=InnoDB AUTO_INCREMENT=0 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-/*!40101 SET character_set_client = @saved_cs_client */;
-
-DROP TABLE IF EXISTS `users`;
-/*!40101 SET @saved_cs_client = @@character_set_client */;
-/*!50503 SET character_set_client = utf8mb4 */;
-CREATE TABLE `users` (
- `id` int unsigned NOT NULL AUTO_INCREMENT,
- `fk_entite` int unsigned DEFAULT '1',
- `fk_role` int unsigned DEFAULT '1',
- `fk_titre` int unsigned DEFAULT '1',
- `encrypted_name` varchar(255) DEFAULT NULL,
- `first_name` varchar(45) DEFAULT NULL,
- `sect_name` varchar(60) DEFAULT '',
- `encrypted_user_name` varchar(128) DEFAULT '',
- `user_pass_hash` varchar(60) DEFAULT NULL,
- `encrypted_phone` varchar(128) DEFAULT NULL,
- `encrypted_mobile` varchar(128) DEFAULT NULL,
- `encrypted_email` varchar(255) DEFAULT '',
- `chk_alert_email` tinyint(1) unsigned DEFAULT '1',
- `chk_suivi` tinyint(1) unsigned DEFAULT '0',
- `date_naissance` date DEFAULT NULL,
- `date_embauche` date DEFAULT NULL,
- `created_at` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT 'Date de création',
- `fk_user_creat` int unsigned DEFAULT NULL,
- `updated_at` timestamp NULL DEFAULT NULL ON UPDATE CURRENT_TIMESTAMP COMMENT 'Date de modification',
- `fk_user_modif` int unsigned DEFAULT NULL,
- `chk_active` tinyint(1) unsigned DEFAULT '1',
- PRIMARY KEY (`id`),
- KEY `fk_entite` (`fk_entite`),
- KEY `username` (`encrypted_user_name`),
- CONSTRAINT `users_ibfk_1` FOREIGN KEY (`fk_entite`) REFERENCES `entites` (`id`) ON DELETE RESTRICT ON UPDATE CASCADE,
- CONSTRAINT `users_ibfk_2` FOREIGN KEY (`fk_role`) REFERENCES `x_users_roles` (`id`) ON DELETE RESTRICT ON UPDATE CASCADE,
- CONSTRAINT `users_ibfk_3` FOREIGN KEY (`fk_titre`) REFERENCES `x_users_titres` (`id`) ON DELETE RESTRICT ON UPDATE CASCADE
-) ENGINE=InnoDB AUTO_INCREMENT=0 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-/*!40101 SET character_set_client = @saved_cs_client */;
-
-DROP TABLE IF EXISTS `operations`;
-/*!40101 SET @saved_cs_client = @@character_set_client */;
-/*!50503 SET character_set_client = utf8mb4 */;
-CREATE TABLE `operations` (
- `id` int unsigned NOT NULL AUTO_INCREMENT,
- `fk_entite` int unsigned NOT NULL DEFAULT '1',
- `libelle` varchar(75) NOT NULL DEFAULT '',
- `date_deb` date NOT NULL DEFAULT '0000-00-00',
- `date_fin` date NOT NULL DEFAULT '0000-00-00',
- `chk_distinct_sectors` tinyint(1) unsigned NOT NULL DEFAULT '0',
- `created_at` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT 'Date de création',
- `fk_user_creat` int unsigned NOT NULL DEFAULT '0',
- `updated_at` timestamp NULL DEFAULT NULL ON UPDATE CURRENT_TIMESTAMP COMMENT 'Date de modification',
- `fk_user_modif` int unsigned NOT NULL DEFAULT '0',
- `chk_active` tinyint(1) unsigned NOT NULL DEFAULT '1',
- PRIMARY KEY (`id`),
- KEY `fk_entite` (`fk_entite`),
- KEY `date_deb` (`date_deb`),
- CONSTRAINT `operations_ibfk_1` FOREIGN KEY (`fk_entite`) REFERENCES `entites` (`id`) ON DELETE RESTRICT ON UPDATE CASCADE
-) ENGINE=InnoDB AUTO_INCREMENT=0 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-/*!40101 SET character_set_client = @saved_cs_client */;
-
-
-DROP TABLE IF EXISTS `ope_sectors`;
-/*!40101 SET @saved_cs_client = @@character_set_client */;
-/*!50503 SET character_set_client = utf8mb4 */;
-CREATE TABLE `ope_sectors` (
- `id` int unsigned NOT NULL AUTO_INCREMENT,
- `fk_operation` int unsigned NOT NULL DEFAULT '0',
- `fk_old_sector` int unsigned NOT NULL DEFAULT '0',
- `libelle` varchar(75) NOT NULL DEFAULT '',
- `sector` text NOT NULL DEFAULT '',
- `color` varchar(7) NOT NULL DEFAULT '#4B77BE',
- `created_at` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT 'Date de création',
- `fk_user_creat` int unsigned NOT NULL DEFAULT '0',
- `updated_at` timestamp NULL DEFAULT NULL ON UPDATE CURRENT_TIMESTAMP COMMENT 'Date de modification',
- `fk_user_modif` int unsigned NOT NULL DEFAULT '0',
- `chk_active` tinyint(1) unsigned NOT NULL DEFAULT '1',
- PRIMARY KEY (`id`),
- UNIQUE KEY `id` (`id`),
- KEY `fk_operation` (`fk_operation`),
- CONSTRAINT `ope_sectors_ibfk_1` FOREIGN KEY (`fk_operation`) REFERENCES `operations` (`id`) ON DELETE RESTRICT ON UPDATE CASCADE
-) ENGINE=InnoDB AUTO_INCREMENT=0 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-
-
-DROP TABLE IF EXISTS `ope_users`;
-/*!40101 SET @saved_cs_client = @@character_set_client */;
-/*!50503 SET character_set_client = utf8mb4 */;
-CREATE TABLE `ope_users` (
- `id` int unsigned NOT NULL AUTO_INCREMENT,
- `fk_operation` int unsigned NOT NULL DEFAULT '0',
- `fk_user` int unsigned NOT NULL DEFAULT '0',
- `created_at` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT 'Date de création',
- `fk_user_creat` int unsigned DEFAULT NULL,
- `updated_at` timestamp NULL DEFAULT NULL ON UPDATE CURRENT_TIMESTAMP COMMENT 'Date de modification',
- `fk_user_modif` int unsigned DEFAULT NULL,
- `chk_active` tinyint(1) unsigned NOT NULL DEFAULT '1',
- PRIMARY KEY (`id`),
- UNIQUE KEY `id_UNIQUE` (`id`),
- CONSTRAINT `ope_users_ibfk_1` FOREIGN KEY (`fk_operation`) REFERENCES `operations` (`id`) ON DELETE RESTRICT ON UPDATE CASCADE,
- CONSTRAINT `ope_users_ibfk_2` FOREIGN KEY (`fk_user`) REFERENCES `users` (`id`) ON DELETE RESTRICT ON UPDATE CASCADE
-) ENGINE=InnoDB AUTO_INCREMENT=0 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-/*!40101 SET character_set_client = @saved_cs_client */;
-
-DROP TABLE IF EXISTS `email_queue`;
-/*!40101 SET @saved_cs_client = @@character_set_client */;
-/*!50503 SET character_set_client = utf8mb4 */;
-CREATE TABLE `email_queue` (
- `id` int unsigned NOT NULL AUTO_INCREMENT,
- `fk_pass` int unsigned NOT NULL DEFAULT '0',
- `to_email` varchar(255) COLLATE utf8mb4_unicode_ci DEFAULT NULL,
- `subject` varchar(255) COLLATE utf8mb4_unicode_ci DEFAULT NULL,
- `body` text COLLATE utf8mb4_unicode_ci,
- `headers` text COLLATE utf8mb4_unicode_ci,
- `created_at` timestamp NULL DEFAULT CURRENT_TIMESTAMP,
- `status` enum('pending','sent','failed') COLLATE utf8mb4_unicode_ci DEFAULT 'pending',
- `sent_at` timestamp NULL DEFAULT NULL,
- `attempts` int unsigned DEFAULT '0',
- `error_message` text COLLATE utf8mb4_unicode_ci DEFAULT NULL,
- PRIMARY KEY (`id`)
-) ENGINE=InnoDB AUTO_INCREMENT=0 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-/*!40101 SET character_set_client = @saved_cs_client */;
-
-DROP TABLE IF EXISTS `ope_users_sectors`;
-/*!40101 SET @saved_cs_client = @@character_set_client */;
-/*!50503 SET character_set_client = utf8mb4 */;
-CREATE TABLE `ope_users_sectors` (
- `id` int unsigned NOT NULL AUTO_INCREMENT,
- `fk_operation` int unsigned NOT NULL DEFAULT '0',
- `fk_user` int unsigned NOT NULL DEFAULT '0',
- `fk_sector` int unsigned NOT NULL DEFAULT '0',
- `created_at` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT 'Date de création',
- `fk_user_creat` int unsigned NOT NULL DEFAULT '0',
- `updated_at` timestamp NULL DEFAULT NULL ON UPDATE CURRENT_TIMESTAMP COMMENT 'Date de modification',
- `fk_user_modif` int unsigned DEFAULT NULL,
- `chk_active` tinyint(1) unsigned DEFAULT '1',
- PRIMARY KEY (`id`),
- UNIQUE KEY `id` (`id`),
- KEY `fk_operation` (`fk_operation`),
- KEY `fk_user` (`fk_user`),
- KEY `fk_sector` (`fk_sector`),
- CONSTRAINT `ope_users_sectors_ibfk_1` FOREIGN KEY (`fk_operation`) REFERENCES `operations` (`id`) ON DELETE RESTRICT ON UPDATE CASCADE,
- CONSTRAINT `ope_users_sectors_ibfk_2` FOREIGN KEY (`fk_user`) REFERENCES `users` (`id`) ON DELETE RESTRICT ON UPDATE CASCADE,
- CONSTRAINT `ope_users_sectors_ibfk_3` FOREIGN KEY (`fk_sector`) REFERENCES `ope_sectors` (`id`) ON DELETE RESTRICT ON UPDATE CASCADE
-) ENGINE=InnoDB AUTO_INCREMENT=0 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-/*!40101 SET character_set_client = @saved_cs_client */;
-
-DROP TABLE IF EXISTS `ope_users_suivis`;
-/*!40101 SET @saved_cs_client = @@character_set_client */;
-/*!50503 SET character_set_client = utf8mb4 */;
-CREATE TABLE `ope_users_suivis` (
- `id` int unsigned NOT NULL AUTO_INCREMENT,
- `fk_operation` int unsigned NOT NULL DEFAULT '0',
- `fk_user` int unsigned NOT NULL DEFAULT '0',
- `date_suivi` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT 'Date du suivi',
- `gps_lat` varchar(20) NOT NULL DEFAULT '',
- `gps_lng` varchar(20) NOT NULL DEFAULT '',
- `vitesse` varchar(20) NOT NULL DEFAULT '',
- `created_at` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT 'Date de création',
- `fk_user_creat` int unsigned NOT NULL DEFAULT '0',
- `updated_at` timestamp NULL DEFAULT NULL ON UPDATE CURRENT_TIMESTAMP COMMENT 'Date de modification',
- `fk_user_modif` int unsigned NOT NULL DEFAULT '0',
- PRIMARY KEY (`id`),
- UNIQUE KEY `id_UNIQUE` (`id`)
-) ENGINE=InnoDB AUTO_INCREMENT=0 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-/*!40101 SET character_set_client = @saved_cs_client */;
-
-DROP TABLE IF EXISTS `sectors_adresses`;
-/*!40101 SET @saved_cs_client = @@character_set_client */;
-/*!50503 SET character_set_client = utf8mb4 */;
-CREATE TABLE `sectors_adresses` (
- `id` int unsigned NOT NULL AUTO_INCREMENT,
- `fk_adresse` varchar(25) CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci DEFAULT NULL COMMENT 'adresses.cp??.id',
- `osm_id` int unsigned NOT NULL DEFAULT '0',
- `fk_sector` int unsigned NOT NULL DEFAULT '0',
- `osm_name` varchar(50) NOT NULL DEFAULT '',
- `numero` varchar(5) CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci NOT NULL DEFAULT '',
- `rue_bis` varchar(5) CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci NOT NULL DEFAULT '',
- `rue` varchar(60) CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci NOT NULL DEFAULT '',
- `cp` varchar(5) CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci NOT NULL DEFAULT '',
- `ville` varchar(60) CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci NOT NULL DEFAULT '',
- `gps_lat` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci NOT NULL DEFAULT '',
- `gps_lng` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci NOT NULL DEFAULT '',
- `osm_date_creat` timestamp NOT NULL DEFAULT '0000-00-00 00:00:00',
- `created_at` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT 'Date de création',
- `updated_at` timestamp NULL DEFAULT NULL ON UPDATE CURRENT_TIMESTAMP COMMENT 'Date de modification',
- PRIMARY KEY (`id`),
- KEY `sectors_adresses_fk_sector_index` (`fk_sector`),
- KEY `sectors_adresses_numero_index` (`numero`),
- KEY `sectors_adresses_rue_index` (`rue`),
- KEY `sectors_adresses_ville_index` (`ville`),
- CONSTRAINT `sectors_adresses_ibfk_1` FOREIGN KEY (`fk_sector`) REFERENCES `ope_sectors` (`id`) ON DELETE RESTRICT ON UPDATE CASCADE
-) ENGINE=InnoDB AUTO_INCREMENT=0 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-
-DROP TABLE IF EXISTS `ope_pass`;
-/*!40101 SET @saved_cs_client = @@character_set_client */;
-/*!50503 SET character_set_client = utf8mb4 */;
-CREATE TABLE `ope_pass` (
- `id` int unsigned NOT NULL AUTO_INCREMENT,
- `fk_operation` int unsigned NOT NULL DEFAULT '0',
- `fk_sector` int unsigned DEFAULT '0',
- `fk_user` int unsigned NOT NULL DEFAULT '0',
- `fk_adresse` varchar(25) DEFAULT '' COMMENT 'adresses.cp??.id',
- `passed_at` timestamp NULL DEFAULT NULL COMMENT 'Date du passage',
- `fk_type` int unsigned DEFAULT '0',
- `numero` varchar(10) NOT NULL DEFAULT '',
- `rue` varchar(75) NOT NULL DEFAULT '',
- `rue_bis` varchar(1) NOT NULL DEFAULT '',
- `ville` varchar(75) NOT NULL DEFAULT '',
- `fk_habitat` int unsigned DEFAULT '1',
- `appt` varchar(5) DEFAULT '',
- `niveau` varchar(5) DEFAULT '',
- `residence` varchar(75) DEFAULT '',
- `gps_lat` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci NOT NULL DEFAULT '',
- `gps_lng` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci NOT NULL DEFAULT '',
- `encrypted_name` varchar(255) NOT NULL DEFAULT '',
- `montant` decimal(7,2) NOT NULL DEFAULT '0.00',
- `fk_type_reglement` int unsigned DEFAULT '1',
- `remarque` text DEFAULT '',
- `encrypted_email` varchar(255) DEFAULT '',
- `nom_recu` varchar(50) COLLATE utf8mb4_unicode_ci DEFAULT NULL,
- `date_recu` timestamp NULL DEFAULT NULL COMMENT 'Date de réception',
- `date_creat_recu` timestamp NULL DEFAULT NULL COMMENT 'Date de création du reçu',
- `date_sent_recu` timestamp NULL DEFAULT NULL COMMENT 'Date envoi du reçu',
- `email_erreur` varchar(30) DEFAULT '',
- `chk_email_sent` tinyint(1) unsigned NOT NULL DEFAULT '0',
- `encrypted_phone` varchar(128) NOT NULL DEFAULT '',
- `chk_striped` tinyint(1) unsigned DEFAULT '0',
- `docremis` tinyint(1) unsigned DEFAULT '0',
- `date_repasser` timestamp NULL DEFAULT NULL COMMENT 'Date prévue pour repasser',
- `nb_passages` int DEFAULT '1' COMMENT 'Nb passages pour les a repasser',
- `chk_gps_maj` tinyint(1) unsigned DEFAULT '0',
- `chk_map_create` tinyint(1) unsigned DEFAULT '0',
- `chk_mobile` tinyint(1) unsigned DEFAULT '0',
- `chk_synchro` tinyint(1) unsigned DEFAULT '1' COMMENT 'chk synchro entre web et appli',
- `chk_api_adresse` tinyint(1) unsigned DEFAULT '0',
- `chk_maj_adresse` tinyint(1) unsigned DEFAULT '0',
- `anomalie` tinyint(1) unsigned DEFAULT '0',
- `created_at` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT 'Date de création',
- `fk_user_creat` int unsigned DEFAULT NULL,
- `updated_at` timestamp NULL DEFAULT NULL ON UPDATE CURRENT_TIMESTAMP COMMENT 'Date de modification',
- `fk_user_modif` int unsigned DEFAULT NULL,
- `chk_active` tinyint(1) unsigned NOT NULL DEFAULT '1',
- PRIMARY KEY (`id`),
- KEY `fk_operation` (`fk_operation`),
- KEY `fk_sector` (`fk_sector`),
- KEY `fk_user` (`fk_user`),
- KEY `fk_type` (`fk_type`),
- KEY `fk_type_reglement` (`fk_type_reglement`),
- KEY `email` (`email`),
- CONSTRAINT `ope_pass_ibfk_1` FOREIGN KEY (`fk_operation`) REFERENCES `operations` (`id`) ON DELETE RESTRICT ON UPDATE CASCADE,
- CONSTRAINT `ope_pass_ibfk_2` FOREIGN KEY (`fk_sector`) REFERENCES `ope_sectors` (`id`) ON DELETE RESTRICT ON UPDATE CASCADE,
- CONSTRAINT `ope_pass_ibfk_3` FOREIGN KEY (`fk_user`) REFERENCES `users` (`id`) ON DELETE RESTRICT ON UPDATE CASCADE,
- CONSTRAINT `ope_pass_ibfk_4` FOREIGN KEY (`fk_type_reglement`) REFERENCES `x_types_reglements` (`id`) ON DELETE RESTRICT ON UPDATE CASCADE
-) ENGINE=InnoDB AUTO_INCREMENT=0 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-/*!40101 SET character_set_client = @saved_cs_client */;
-
-DROP TABLE IF EXISTS `ope_pass_histo`;
-/*!40101 SET @saved_cs_client = @@character_set_client */;
-/*!50503 SET character_set_client = utf8mb4 */;
-CREATE TABLE `ope_pass_histo` (
- `id` int unsigned NOT NULL AUTO_INCREMENT,
- `fk_pass` int unsigned NOT NULL DEFAULT '0',
- `date_histo` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT 'Date historique',
- `sujet` varchar(50) CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci DEFAULT NULL,
- `remarque` varchar(250) NOT NULL DEFAULT '',
- PRIMARY KEY (`id`),
- KEY `ope_pass_histo_fk_pass_IDX` (`fk_pass`) USING BTREE,
- KEY `ope_pass_histo_date_histo_IDX` (`date_histo`) USING BTREE,
- CONSTRAINT `ope_pass_histo_ibfk_1` FOREIGN KEY (`fk_pass`) REFERENCES `ope_pass` (`id`) ON DELETE CASCADE ON UPDATE CASCADE
-) ENGINE=InnoDB AUTO_INCREMENT=0 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-/*!40101 SET character_set_client = @saved_cs_client */;
-
-DROP TABLE IF EXISTS `medias`;
-/*!40101 SET @saved_cs_client = @@character_set_client */;
-/*!50503 SET character_set_client = utf8mb4 */;
-CREATE TABLE `medias` (
- `id` int unsigned NOT NULL AUTO_INCREMENT,
- `support` varchar(45) NOT NULL DEFAULT '',
- `support_id` int unsigned NOT NULL DEFAULT '0',
- `fichier` varchar(250) NOT NULL DEFAULT '',
- `description` varchar(100) NOT NULL DEFAULT '',
- `created_at` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP,
- `fk_user_creat` int unsigned NOT NULL DEFAULT '0',
- `updated_at` timestamp NULL DEFAULT NULL ON UPDATE CURRENT_TIMESTAMP,
- `fk_user_modif` int unsigned NOT NULL DEFAULT '0',
- PRIMARY KEY (`id`),
- UNIQUE KEY `id_UNIQUE` (`id`)
-) ENGINE=InnoDB AUTO_INCREMENT=0 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-/*!40101 SET character_set_client = @saved_cs_client */;
-
--- Création des tables pour le système de chat
-DROP TABLE IF EXISTS `chat_rooms`;
--- Table des salles de discussion
-CREATE TABLE chat_rooms (
- id INT UNSIGNED AUTO_INCREMENT PRIMARY KEY,
- name VARCHAR(100) NOT NULL,
- type ENUM('privee', 'groupe', 'liste_diffusion') NOT NULL,
- date_creation timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT 'Date de création',
- fk_user INT UNSIGNED NOT NULL,
- fk_entite INT UNSIGNED,
- statut ENUM('active', 'archive') NOT NULL DEFAULT 'active',
- description TEXT,
- INDEX idx_user (fk_user),
- INDEX idx_entite (fk_entite)
-) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-
-
-DROP TABLE IF EXISTS `chat_participants`;
--- Table des participants aux salles de discussion
-CREATE TABLE chat_participants (
- id INT UNSIGNED AUTO_INCREMENT PRIMARY KEY,
- id_room INT UNSIGNED NOT NULL,
- id_user INT UNSIGNED NOT NULL,
- role ENUM('administrateur', 'participant', 'en_lecture_seule') NOT NULL DEFAULT 'participant',
- date_ajout timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT 'Date ajout',
- notification_activee BOOLEAN NOT NULL DEFAULT TRUE,
- INDEX idx_room (id_room),
- INDEX idx_user (id_user),
- CONSTRAINT uc_room_user UNIQUE (id_room, id_user),
- FOREIGN KEY (id_room) REFERENCES chat_rooms(id) ON DELETE CASCADE
-) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-
-DROP TABLE IF EXISTS `chat_messages`;
--- Table des messages
-CREATE TABLE chat_messages (
- id INT UNSIGNED AUTO_INCREMENT PRIMARY KEY,
- fk_room INT UNSIGNED NOT NULL,
- fk_user INT UNSIGNED NOT NULL,
- content TEXT,
- date_sent timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT 'Date envoi',
- type ENUM('texte', 'media', 'systeme') NOT NULL DEFAULT 'texte',
- statut ENUM('envoye', 'livre', 'lu') NOT NULL DEFAULT 'envoye',
- INDEX idx_room (fk_room),
- INDEX idx_user (fk_user),
- INDEX idx_date (date_sent),
- FOREIGN KEY (fk_room) REFERENCES chat_rooms(id) ON DELETE CASCADE
-) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-
-DROP TABLE IF EXISTS `chat_listes_diffusion`;
--- Table des listes de diffusion
-CREATE TABLE chat_listes_diffusion (
- id INT UNSIGNED AUTO_INCREMENT PRIMARY KEY,
- fk_room INT UNSIGNED NOT NULL,
- name VARCHAR(100) NOT NULL,
- description TEXT,
- fk_user INT UNSIGNED NOT NULL,
- date_creation timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT 'Date de création',
- INDEX idx_room (fk_room),
- INDEX idx_user (fk_user),
- FOREIGN KEY (fk_room) REFERENCES chat_rooms(id) ON DELETE CASCADE
-) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-
-DROP TABLE IF EXISTS `chat_read_messages`;
--- Table pour suivre la lecture des messages
-CREATE TABLE chat_read_messages (
- id BIGINT UNSIGNED AUTO_INCREMENT PRIMARY KEY,
- fk_message INT UNSIGNED NOT NULL,
- fk_user INT UNSIGNED NOT NULL,
- date_read timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT 'Date de lecture',
- INDEX idx_message (fk_message),
- INDEX idx_user (fk_user),
- CONSTRAINT uc_message_user UNIQUE (fk_message, fk_user),
- FOREIGN KEY (fk_message) REFERENCES chat_messages(id) ON DELETE CASCADE
-) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-
-DROP TABLE IF EXISTS `chat_notifications`;
--- Table des notifications
-CREATE TABLE chat_notifications (
- id BIGINT UNSIGNED AUTO_INCREMENT PRIMARY KEY,
- fk_user INT UNSIGNED NOT NULL,
- fk_message INT UNSIGNED,
- fk_room INT UNSIGNED,
- type VARCHAR(50) NOT NULL,
- contenu TEXT,
- date_creation timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT 'Date de création',
- date_lecture timestamp NULL DEFAULT NULL COMMENT 'Date de lecture',
- statut ENUM('non_lue', 'lue') NOT NULL DEFAULT 'non_lue',
- INDEX idx_user (fk_user),
- INDEX idx_message (fk_message),
- INDEX idx_room (fk_room),
- FOREIGN KEY (fk_message) REFERENCES chat_messages(id) ON DELETE SET NULL,
- FOREIGN KEY (fk_room) REFERENCES chat_rooms(id) ON DELETE SET NULL
-) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-
-DROP TABLE IF EXISTS `z_params`;
-/*!40101 SET @saved_cs_client = @@character_set_client */;
-/*!50503 SET character_set_client = utf8mb4 */;
-CREATE TABLE `params` (
- `id` int unsigned NOT NULL AUTO_INCREMENT,
- `libelle` varchar(35) NOT NULL DEFAULT '',
- `valeur` varchar(255) NOT NULL DEFAULT '',
- `aide` varchar(150) NOT NULL DEFAULT '',
- PRIMARY KEY (`id`)
-) ENGINE=InnoDB AUTO_INCREMENT=2 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-/*!40101 SET character_set_client = @saved_cs_client */;
-
-
-DROP TABLE IF EXISTS `z_sessions`;
-/*!40101 SET @saved_cs_client = @@character_set_client */;
-/*!50503 SET character_set_client = utf8mb4 */;
-CREATE TABLE `z_sessions` (
- `sid` text NOT NULL,
- `fk_user` int NOT NULL,
- `role` varchar(10) DEFAULT NULL,
- `date_modified` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
- `ip` varchar(50) NOT NULL,
- `browser` varchar(150) NOT NULL,
- `data` mediumtext
-) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-/*!40101 SET character_set_client = @saved_cs_client */;
diff --git a/api/docs/recu_13718.pdf b/api/docs/recu_13718.pdf
new file mode 100644
index 00000000..f912f84e
Binary files /dev/null and b/api/docs/recu_13718.pdf differ
diff --git a/api/docs/traite_batiments.sql b/api/docs/traite_batiments.sql
new file mode 100644
index 00000000..f56ab07f
--- /dev/null
+++ b/api/docs/traite_batiments.sql
@@ -0,0 +1,193 @@
+USE batiments;
+
+-- Table temp pour FFO (nb_niveau, nb_log)
+DROP TABLE IF EXISTS tmp_ffo_999;
+CREATE TABLE tmp_ffo_999 (
+ batiment_groupe_id VARCHAR(50),
+ code_departement_insee VARCHAR(5),
+ nb_niveau INT,
+ annee_construction INT,
+ usage_niveau_1_txt VARCHAR(100),
+ mat_mur_txt VARCHAR(100),
+ mat_toit_txt VARCHAR(100),
+ nb_log INT,
+ KEY (batiment_groupe_id)
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
+
+LOAD DATA LOCAL INFILE '/var/osm/csv/batiment_groupe_ffo_bat.csv'
+INTO TABLE tmp_ffo_999
+CHARACTER SET 'UTF8mb4'
+FIELDS TERMINATED BY ','
+OPTIONALLY ENCLOSED BY '"'
+IGNORE 1 LINES;
+
+-- Table temp pour Adresse (lien BAN)
+DROP TABLE IF EXISTS tmp_adr_999;
+CREATE TABLE tmp_adr_999 (
+ wkt TEXT,
+ batiment_groupe_id VARCHAR(50),
+ cle_interop_adr VARCHAR(50),
+ code_departement_insee VARCHAR(5),
+ classe VARCHAR(50),
+ lien_valide TINYINT,
+ origine VARCHAR(50),
+ KEY (batiment_groupe_id),
+ KEY (cle_interop_adr)
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
+
+LOAD DATA LOCAL INFILE '/var/osm/csv/rel_batiment_groupe_adresse.csv'
+INTO TABLE tmp_adr_999
+CHARACTER SET 'UTF8mb4'
+FIELDS TERMINATED BY ','
+OPTIONALLY ENCLOSED BY '"'
+IGNORE 1 LINES;
+
+-- Table temp pour RNC (copropriétés)
+DROP TABLE IF EXISTS tmp_rnc_999;
+CREATE TABLE tmp_rnc_999 (
+ batiment_groupe_id VARCHAR(50),
+ code_departement_insee VARCHAR(5),
+ numero_immat_principal VARCHAR(50),
+ periode_construction_max VARCHAR(50),
+ l_annee_construction VARCHAR(100),
+ nb_lot_garpark INT,
+ nb_lot_tot INT,
+ nb_log INT,
+ nb_lot_tertiaire INT,
+ l_nom_copro VARCHAR(200),
+ l_siret VARCHAR(50),
+ copro_dans_pvd TINYINT,
+ KEY (batiment_groupe_id)
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
+
+LOAD DATA LOCAL INFILE '/var/osm/csv/batiment_groupe_rnc.csv'
+INTO TABLE tmp_rnc_999
+CHARACTER SET 'UTF8mb4'
+FIELDS TERMINATED BY ','
+OPTIONALLY ENCLOSED BY '"'
+IGNORE 1 LINES;
+
+-- Table temp pour BDTOPO (altitude)
+DROP TABLE IF EXISTS tmp_topo_999;
+CREATE TABLE tmp_topo_999 (
+ batiment_groupe_id VARCHAR(50),
+ code_departement_insee VARCHAR(5),
+ l_nature VARCHAR(200),
+ l_usage_1 VARCHAR(200),
+ l_usage_2 VARCHAR(200),
+ l_etat VARCHAR(100),
+ hauteur_mean DECIMAL(10,2),
+ max_hauteur DECIMAL(10,2),
+ altitude_sol_mean DECIMAL(10,2),
+ KEY (batiment_groupe_id)
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
+
+LOAD DATA LOCAL INFILE '/var/osm/csv/batiment_groupe_bdtopo_bat.csv'
+INTO TABLE tmp_topo_999
+CHARACTER SET 'UTF8mb4'
+FIELDS TERMINATED BY ','
+OPTIONALLY ENCLOSED BY '"'
+IGNORE 1 LINES;
+
+-- Table temp pour Usage principal
+DROP TABLE IF EXISTS tmp_usage_999;
+CREATE TABLE tmp_usage_999 (
+ batiment_groupe_id VARCHAR(50),
+ code_departement_insee VARCHAR(5),
+ usage_principal_bdnb_open VARCHAR(100),
+ KEY (batiment_groupe_id)
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
+
+LOAD DATA LOCAL INFILE '/var/osm/csv/batiment_groupe_synthese_propriete_usage.csv'
+INTO TABLE tmp_usage_999
+CHARACTER SET 'UTF8mb4'
+FIELDS TERMINATED BY ','
+OPTIONALLY ENCLOSED BY '"'
+IGNORE 1 LINES;
+
+-- Table temp pour DLE Enedis (compteurs électriques)
+DROP TABLE IF EXISTS tmp_dle_999;
+CREATE TABLE tmp_dle_999 (
+ batiment_groupe_id VARCHAR(50),
+ code_departement_insee VARCHAR(5),
+ millesime VARCHAR(10),
+ nb_pdl_res INT,
+ nb_pdl_pro INT,
+ nb_pdl_tot INT,
+ conso_res DECIMAL(12,2),
+ conso_pro DECIMAL(12,2),
+ conso_tot DECIMAL(12,2),
+ conso_res_par_pdl DECIMAL(12,2),
+ conso_pro_par_pdl DECIMAL(12,2),
+ conso_tot_par_pdl DECIMAL(12,2),
+ KEY (batiment_groupe_id)
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
+
+LOAD DATA LOCAL INFILE '/var/osm/csv/batiment_groupe_dle_elec_multimillesime.csv'
+INTO TABLE tmp_dle_999
+CHARACTER SET 'UTF8mb4'
+FIELDS TERMINATED BY ','
+OPTIONALLY ENCLOSED BY '"'
+IGNORE 1 LINES;
+
+-- Création de la table finale avec jointure et filtre
+DROP TABLE IF EXISTS bat999;
+CREATE TABLE bat999 (
+ batiment_groupe_id VARCHAR(50) PRIMARY KEY,
+ code_departement_insee VARCHAR(5),
+ cle_interop_adr VARCHAR(50),
+ nb_niveau INT,
+ nb_log INT,
+ nb_pdl_tot INT,
+ annee_construction INT,
+ residence VARCHAR(200),
+ usage_principal VARCHAR(100),
+ altitude_sol_mean DECIMAL(10,2),
+ gps_lat DECIMAL(10,7),
+ gps_lng DECIMAL(10,7),
+ KEY (cle_interop_adr),
+ KEY (usage_principal),
+ KEY (nb_log)
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;
+
+INSERT INTO bat999
+SELECT
+ f.batiment_groupe_id,
+ f.code_departement_insee,
+ a.cle_interop_adr,
+ f.nb_niveau,
+ f.nb_log,
+ d.nb_pdl_tot,
+ f.annee_construction,
+ REPLACE(REPLACE(REPLACE(REPLACE(r.l_nom_copro, '[', ''), ']', ''), '"', ''), ' ', ' ') as residence,
+ u.usage_principal_bdnb_open as usage_principal,
+ t.altitude_sol_mean,
+ NULL as gps_lat,
+ NULL as gps_lng
+FROM tmp_ffo_999 f
+INNER JOIN tmp_adr_999 a ON f.batiment_groupe_id = a.batiment_groupe_id AND a.lien_valide = 1
+LEFT JOIN tmp_rnc_999 r ON f.batiment_groupe_id = r.batiment_groupe_id
+LEFT JOIN tmp_topo_999 t ON f.batiment_groupe_id = t.batiment_groupe_id
+LEFT JOIN tmp_usage_999 u ON f.batiment_groupe_id = u.batiment_groupe_id
+LEFT JOIN tmp_dle_999 d ON f.batiment_groupe_id = d.batiment_groupe_id
+WHERE u.usage_principal_bdnb_open IN ('Résidentiel individuel', 'Résidentiel collectif', 'Secondaire', 'Tertiaire')
+AND f.nb_log > 1
+AND a.cle_interop_adr IS NOT NULL
+GROUP BY f.batiment_groupe_id;
+
+-- Mise à jour des coordonnées GPS depuis la base adresses
+UPDATE bat999 b
+JOIN adresses.cp999 a ON b.cle_interop_adr = a.id
+SET b.gps_lat = a.gps_lat, b.gps_lng = a.gps_lng
+WHERE b.cle_interop_adr IS NOT NULL;
+
+-- Nettoyage des tables temporaires
+DROP TABLE IF EXISTS tmp_ffo_999;
+DROP TABLE IF EXISTS tmp_adr_999;
+DROP TABLE IF EXISTS tmp_rnc_999;
+DROP TABLE IF EXISTS tmp_topo_999;
+DROP TABLE IF EXISTS tmp_usage_999;
+DROP TABLE IF EXISTS tmp_dle_999;
+
+-- Historique
+INSERT INTO _histo SET date_import=NOW(), dept='999', nb_batiments=(SELECT COUNT(*) FROM bat999);
diff --git a/api/index.php b/api/index.php
index 565e9b42..4334a522 100755
--- a/api/index.php
+++ b/api/index.php
@@ -42,6 +42,8 @@ require_once __DIR__ . '/src/Controllers/ChatController.php';
require_once __DIR__ . '/src/Controllers/SecurityController.php';
require_once __DIR__ . '/src/Controllers/StripeController.php';
require_once __DIR__ . '/src/Controllers/StripeWebhookController.php';
+require_once __DIR__ . '/src/Controllers/MigrationController.php';
+require_once __DIR__ . '/src/Controllers/HealthController.php';
// Initialiser la configuration
$appConfig = AppConfig::getInstance();
diff --git a/api/scripts/CORRECTIONS_MIGRATE.md b/api/scripts/CORRECTIONS_MIGRATE.md
new file mode 100644
index 00000000..9f25869b
--- /dev/null
+++ b/api/scripts/CORRECTIONS_MIGRATE.md
@@ -0,0 +1,290 @@
+# 🔧 CORRECTIONS CRITIQUES - migrate_from_backup.php
+
+## ❌ ERREURS DÉTECTÉES
+
+### 1. **migrateUsers** (ligne 456)
+```sql
+-- ERREUR
+u.nom, u.prenom, u.nom_sect, u.username, u.password, u.phone, u.mobile
+
+-- CORRECTION (noms réels dans geosector.users)
+u.libelle, u.prenom, u.nom_tournee, u.username, u.userpass, u.telephone, u.mobile
+```
+
+### 2. **migrateOpePass** (ligne 1043)
+```sql
+-- ERREUR
+p.passed_at, p.libelle, p.email, p.phone
+
+-- CORRECTION (noms réels dans geosector.ope_pass)
+p.date_eve AS passed_at, p.libelle AS encrypted_name, p.email, p.phone
+```
+
+### 3. **migrateSectorsAdresses** (ligne 777)
+```sql
+-- ERREUR
+sa.osm_id, sa.osm_name, sa.osm_date_creat
+
+-- CORRECTION (ces champs n'existent PAS dans geosector.sectors_adresses)
+-- Ces champs doivent être mis à 0 ou NULL dans la cible
+0 AS osm_id, '' AS osm_name, NULL AS osm_date_creat
+```
+
+### 4. **migrateOpeUsersSectors** (ligne 955)
+```sql
+-- ERREUR
+ous.date_creat, ous.fk_user_creat, ous.date_modif, ous.fk_user_modif
+
+-- CORRECTION (geosector.ope_users_sectors n'a PAS ces champs)
+NULL AS created_at, NULL AS fk_user_creat, NULL AS updated_at, NULL AS fk_user_modif
+```
+
+### 5. **migrateMedias** (à vérifier)
+```sql
+-- ERREUR potentielle
+m.support_rowid
+
+-- CORRECTION
+m.support_rowid AS support_id
+```
+
+### 6. **migrateOperations** (erreur NOT NULL)
+```sql
+-- PROBLÈME: Column 'fk_user_modif' cannot be null
+-- CORRECTION: Utiliser 0 au lieu de NULL
+'fk_user_modif' => $row['fk_user_modif'] ?? 0
+```
+
+---
+
+## ✅ SOLUTION RAPIDE
+
+Créez un script `HOTFIX_migrate.sql` pour corriger rapidement :
+
+```sql
+-- Permettre NULL sur les champs problématiques
+ALTER TABLE operations MODIFY COLUMN fk_user_modif INT(10) UNSIGNED NULL DEFAULT NULL;
+ALTER TABLE ope_sectors MODIFY COLUMN fk_user_modif INT(10) UNSIGNED NULL DEFAULT NULL;
+ALTER TABLE ope_users MODIFY COLUMN fk_user_creat INT(10) UNSIGNED NULL DEFAULT NULL;
+ALTER TABLE ope_users MODIFY COLUMN fk_user_modif INT(10) UNSIGNED NULL DEFAULT NULL;
+ALTER TABLE ope_users_sectors MODIFY COLUMN fk_user_creat INT(10) UNSIGNED NULL DEFAULT NULL;
+ALTER TABLE ope_users_sectors MODIFY COLUMN fk_user_modif INT(10) UNSIGNED NULL DEFAULT NULL;
+```
+
+OU utiliser `0` à la place de `NULL` systématiquement dans le script PHP.
+
+---
+
+## 📋 STATUT DES CORRECTIONS (10/10/2025)
+
+1. ✅ **migrateEntites** - CORRIGÉ (cp, tel1, tel2, demo)
+2. ✅ **migrateUsers** - CORRIGÉ (libelle, nom_tournee, telephone, userpass, alert_email) - Lignes 455-537
+3. ✅ **migrateOperations** - CORRIGÉ (fk_user_modif ?? 0, fk_user_creat ?? 0) - Lignes 614-625
+4. ✅ **migrateOpeSectors** - CORRIGÉ (fk_user_modif ?? 0, fk_user_creat ?? 0) - Lignes 727-738
+5. ✅ **migrateSectorsAdresses** - CORRIGÉ (osm_id=0, osm_name='', osm_date_creat=null, created_at/updated_at=null) - Lignes 776-855
+6. ✅ **migrateOpeUsers** - CORRIGÉ (vérification existence user dans TARGET avant insertion) - Lignes 960-1020
+7. ✅ **migrateOpeUsersSectors** - CORRIGÉ (date_creat, fk_user_creat, date_modif, fk_user_modif = null + vérification user) - Lignes 1054-1135
+8. ✅ **migrateOpePass** - CORRIGÉ (date_eve, libelle, recu + fk_type_reglement forcé à 4 si invalide + vérification user) - Lignes 1215-1330
+9. ✅ **migrateMedias** - CORRIGÉ (support_rowid, type_fichier, hauteur/largeur) - Lignes 1281-1343
+10. ✅ **countTargetRows()** - CORRIGÉ (requêtes SQL spécifiques par table avec JOINs corrects) - Lignes 303-355
+
+---
+
+## ✅ CORRECTIONS APPLIQUÉES
+
+**Toutes les erreurs ont été corrigées dans `migrate_from_backup.php`.**
+
+Les corrections incluent :
+- Utilisation des vrais noms de colonnes SOURCE (`geosector-structure.sql`)
+- Gestion des champs manquants dans SOURCE avec valeurs par défaut
+- Utilisation de `?? 0` au lieu de `?? null` pour les FK NOT NULL
+- Suppression des champs inexistants dans les requêtes SELECT
+
+**ATTENTION** : Les noms de colonnes TARGET n'ont PAS été vérifiés contre `geo_app_structure.sql`.
+Le script utilise peut-être les mauvais noms TARGET (à vérifier avec `migrate_users.php` et autres `migrate_*.php` de référence).
+
+---
+
+## 🔧 CORRECTIONS RÉCENTES (Session actuelle)
+
+### 10. **Vérification FK users** (lignes 1008-1015, 1117-1125, 1257-1266)
+**Problème** : Violations de contraintes FK car certains `fk_user` référencent des utilisateurs absents dans TARGET.
+
+**Solution** : Ajout de vérification d'existence avant insertion :
+```php
+// Vérifier que fk_user existe dans users de la TARGET
+$checkUser = $this->targetDb->prepare("SELECT id FROM users WHERE id = ?");
+$checkUser->execute([$row['fk_user']]);
+if (!$checkUser->fetch()) {
+ $this->log(" ⚠ Record {$row['rowid']}: user {$row['fk_user']} non trouvé, ignoré", 'WARNING');
+ continue;
+}
+```
+
+**Appliqué sur** :
+- `migrateOpeUsers()` - ligne 1008
+- `migrateOpeUsersSectors()` - ligne 1117
+- `migrateOpePass()` - ligne 1257
+
+**Résultat** : Les enregistrements avec FK invalides sont ignorés avec un WARNING au lieu de provoquer une erreur fatale.
+
+### 11. **countTargetRows() - Requêtes SQL spécifiques** (lignes 303-355)
+**Problème** : Erreurs SQL car toutes les tables n'ont pas les mêmes colonnes/relations :
+- `Unknown column 'fk_entite' in 'WHERE'` pour `entites`
+- `Unknown column 't.fk_operation' in 'ON'` pour `operations`, `ope_pass_histo`, `medias`
+
+**Solution** : Requêtes SQL personnalisées par table :
+```php
+// Pour entites : pas de FK, juste l'ID
+if ($tableName === 'entites') {
+ $sql = "SELECT COUNT(*) as count FROM $tableName WHERE id = :entity_id";
+}
+// Pour operations : FK directe vers entites
+else if ($tableName === 'operations') {
+ $sql = "SELECT COUNT(*) as count FROM $tableName WHERE fk_entite = :entity_id";
+}
+// Pour sectors_adresses : JOIN via ope_sectors -> operations
+else if ($tableName === 'sectors_adresses') {
+ $sql = "SELECT COUNT(*) as count FROM $tableName sa
+ INNER JOIN ope_sectors s ON sa.fk_sector = s.id
+ INNER JOIN operations o ON s.fk_operation = o.id
+ WHERE o.fk_entite = :entity_id";
+}
+// Pour tables avec fk_operation directe
+else if (in_array($tableName, ['ope_sectors', 'ope_users', 'ope_users_sectors', 'ope_pass', 'ope_pass_histo', 'medias'])) {
+ $sql = "SELECT COUNT(*) as count FROM $tableName t
+ INNER JOIN operations o ON t.fk_operation = o.id
+ WHERE o.fk_entite = :entity_id";
+}
+```
+
+**Résultat** : Comptages TARGET précis et sans erreurs SQL pour toutes les tables.
+
+### 12. **fk_type_reglement validation** (lignes 1237-1241)
+**Problème** : FK violations car certains `fk_type_reglement` référencent des IDs inexistants dans `x_types_reglements` (IDs valides : 1, 2, 3).
+
+**Solution** : Forcer à 4 ("-") si valeur invalide (comme dans `migrate_ope_pass.php`) :
+```php
+// Vérification et correction du type de règlement
+$fkTypeReglement = $row['fk_type_reglement'] ?? 1;
+if (!in_array($fkTypeReglement, [1, 2, 3])) {
+ $fkTypeReglement = 4; // Forcer à 4 ("-") si différent de 1, 2 ou 3
+}
+```
+
+**Résultat** : Tous les `ope_pass` sont migrés sans violation de FK sur `fk_type_reglement`.
+
+### 13. **Limitation aux 3 dernières opérations** (lignes 646-647) ⚠️ IMPORTANT
+**Problème** : Migration de TOUTES les opérations au lieu des 3 dernières uniquement.
+
+**Solution** : Ajout de `ORDER BY rowid DESC LIMIT 3` dans la requête :
+```php
+// Ne migrer que les 3 dernières opérations (plus récentes)
+$sql .= " ORDER BY rowid DESC LIMIT 3";
+```
+
+**Résultat** : Seules les 3 opérations les plus récentes (par rowid DESC) sont migrées par entité.
+**Impact** : Réduit considérablement le volume de données migrées et toutes les tables liées (ope_sectors, ope_users, ope_users_sectors, ope_pass, medias, sectors_adresses).
+
+### 14. **Option de suppression avant migration** (lignes 127-200, 1692, 1722, 1776) ⭐ NOUVELLE FONCTIONNALITÉ
+**Besoin** : Permettre de supprimer les données existantes d'une entité dans TARGET avant migration pour repartir à zéro.
+
+**Solution** : Ajout du paramètre `--delete-before` :
+
+**Script bash** (lignes 174-183) :
+```bash
+# Demander si suppression des données de l'entité avant migration
+echo -ne "${YELLOW}3️⃣ Supprimer les données existantes de cette entité dans la TARGET avant migration ? (y/N): ${NC}"
+read -r DELETE_BEFORE
+DELETE_FLAG=""
+if [[ $DELETE_BEFORE =~ ^[Yy]$ ]]; then
+ echo -e "${GREEN}✓${NC} Les données seront supprimées avant migration"
+ DELETE_FLAG="--delete-before"
+fi
+```
+
+**Script PHP** - Méthode `deleteEntityData()` (lignes 127-200) :
+```php
+private function deleteEntityData($entityId) {
+ // Ordre de suppression inverse pour respecter les FK
+ $deletionOrder = [
+ 'medias', 'ope_pass_histo', 'ope_pass', 'ope_users_sectors',
+ 'ope_users', 'sectors_adresses', 'ope_sectors', 'operations', 'users'
+ ];
+
+ foreach ($deletionOrder as $table) {
+ // Suppression via JOIN avec operations pour respecter FK
+ DELETE t FROM $table t
+ INNER JOIN operations o ON t.fk_operation = o.id
+ WHERE o.fk_entite = ?
+ }
+}
+```
+
+**Résultat** :
+- En mode interactif, l'utilisateur peut choisir de supprimer les données existantes avant migration
+- Suppression propre dans l'ordre inverse des FK (pas d'erreur de contrainte)
+- L'entité elle-même n'est PAS supprimée (car peut avoir d'autres données liées)
+- Transaction avec rollback en cas d'erreur
+
+**Usage** :
+```bash
+# Interactif
+./scripts/migrate_batch.sh
+# Choisir option d) puis répondre 'y' à la question de suppression
+
+# Direct
+php migrate_from_backup.php --source-db=geosector_20251008 --mode=entity --entity-id=5 --delete-before
+```
+
+---
+
+## 📊 RÉSULTATS MIGRATION TEST (Entité #5)
+
+Dernière exécution avec toutes les corrections :
+- ✅ **Entités** : 1 SOURCE → 1 TARGET
+- ✅ **Users** : 21 SOURCE → 21 TARGET (100%)
+- ✅ **Operations** : 4 SOURCE → 4 TARGET (100%)
+- ✅ **Ope_sectors** : 64 SOURCE → 64 TARGET (100%)
+- ⚠️ **Sectors_adresses** : 1975 SOURCE → 1040 TARGET (différence de -935, à investiguer)
+- ✅ **Ope_users** : 20 migrés (0 erreurs après vérification FK)
+- ✅ **Ope_users_sectors** : 20 migrés (0 erreurs après vérification FK)
+- ⚠️ **Ope_pass** : 466 erreurs (users manquants - comportement attendu avec validation FK)
+- ✅ **Medias** : Migration réussie
+
+### 15. **Ajout de contraintes UNIQUE pour éviter les doublons** (10/10/2025) ⭐ CONTRAINTES MANQUANTES
+**Problème** : Les tables `ope_users` et `ope_users_sectors` n'avaient PAS de contrainte UNIQUE sur leurs combinaisons de FK, permettant des doublons massifs.
+
+**Diagnostic** :
+- Table `ope_users` : 186+ doublons pour la même paire (fk_operation, fk_user)
+- Table `ope_users_sectors` : Risque de doublons sur (fk_operation, fk_user, fk_sector)
+- Le `ON DUPLICATE KEY UPDATE` ne fonctionnait pas car aucune contrainte UNIQUE n'existait
+
+**Solution** : Création du script `scripts/sql/add_unique_constraints_ope_tables.sql` qui :
+1. Supprime les doublons existants (garde la première occurrence, supprime les duplicatas)
+2. Ajoute `UNIQUE KEY idx_operation_user (fk_operation, fk_user)` sur `ope_users`
+3. Ajoute `UNIQUE KEY idx_operation_user_sector (fk_operation, fk_user, fk_sector)` sur `ope_users_sectors`
+4. Vérifie les contraintes et compte les doublons restants
+
+**Fichiers modifiés** :
+- `scripts/sql/add_unique_constraints_ope_tables.sql` - Script SQL d'ajout des contraintes
+- `scripts/php/geo_app_structure.sql` - Documentation de la structure cible avec contraintes
+
+**À exécuter AVANT la prochaine migration** :
+```bash
+mysql -u root -p pra_geo < scripts/sql/add_unique_constraints_ope_tables.sql
+```
+
+**Puis re-migrer l'entité** :
+```bash
+php migrate_from_backup.php --source-db=geosector_20251008 --mode=entity --entity-id=5 --delete-before
+```
+
+---
+
+**Prochaines étapes** :
+1. ✅ Exécuter le script SQL pour ajouter les contraintes UNIQUE
+2. ✅ Re-migrer l'entité #5 avec `--delete-before` pour vérifier l'absence de doublons
+3. Investiguer la différence de -935 sur `sectors_adresses`
+4. Analyser les 466 erreurs sur `ope_pass` (probablement des références à des users d'autres entités)
+5. Tester sur une autre entité pour valider la stabilité des corrections
diff --git a/api/scripts/MIGRATION_PATCH_INSTRUCTIONS.md b/api/scripts/MIGRATION_PATCH_INSTRUCTIONS.md
new file mode 100644
index 00000000..b7c2e944
--- /dev/null
+++ b/api/scripts/MIGRATION_PATCH_INSTRUCTIONS.md
@@ -0,0 +1,350 @@
+# Instructions de modification des scripts de migration
+
+## Modifications à effectuer
+
+### 1. migrate_from_backup.php
+
+#### A. Remplacer les lignes 31-50 (configuration DB)
+
+**ANCIEN** :
+```php
+ private $sourceDbName;
+ private $targetDbName;
+ private $sourceDb;
+ private $targetDb;
+ private $mode;
+ private $entityId;
+ private $logFile;
+ private $deleteBefore;
+
+ // Configuration MariaDB (maria4 sur IN4)
+ // pra-geo se connecte à maria4 via l'IP du container
+ private const DB_HOST = '13.23.33.4'; // maria4 sur IN4
+ private const DB_PORT = 3306;
+ private const DB_USER = 'pra_geo_user';
+ private const DB_PASS = 'd2jAAGGWi8fxFrWgXjOA';
+
+ // Pour la base source (backup), on utilise pra_geo_user (avec SELECT sur geosector_*)
+ // L'utilisateur root n'est pas accessible depuis pra-geo (13.23.33.22)
+ private const DB_USER_ROOT = 'pra_geo_user';
+ private const DB_PASS_ROOT = 'd2jAAGGWi8fxFrWgXjOA';
+```
+
+**NOUVEAU** :
+```php
+ private $sourceDbName;
+ private $targetDbName;
+ private $sourceDb;
+ private $targetDb;
+ private $mode;
+ private $entityId;
+ private $logFile;
+ private $deleteBefore;
+ private $env;
+
+ // Configuration multi-environnement
+ private const ENVIRONMENTS = [
+ 'rca' => [
+ 'host' => '13.23.33.3', // maria3 sur IN3
+ 'port' => 3306,
+ 'user' => 'rca_geo_user',
+ 'pass' => 'UPf3C0cQ805LypyM71iW',
+ 'target_db' => 'rca_geo',
+ 'source_db' => 'geosector' // Base synchronisée par PM7
+ ],
+ 'pra' => [
+ 'host' => '13.23.33.4', // maria4 sur IN4
+ 'port' => 3306,
+ 'user' => 'pra_geo_user',
+ 'pass' => 'd2jAAGGWi8fxFrWgXjOA',
+ 'target_db' => 'pra_geo',
+ 'source_db' => 'geosector' // Base synchronisée par PM7
+ ]
+ ];
+```
+
+#### B. Modifier le constructeur (ligne 67)
+
+**ANCIEN** :
+```php
+ public function __construct($sourceDbName, $targetDbName, $mode = 'global', $entityId = null, $logFile = null, $deleteBefore = false) {
+ $this->sourceDbName = $sourceDbName;
+ $this->targetDbName = $targetDbName;
+ $this->mode = $mode;
+ $this->entityId = $entityId;
+ $this->logFile = $logFile ?? '/var/back/migration_' . date('Ymd_His') . '.log';
+ $this->deleteBefore = $deleteBefore;
+
+ $this->log("=== Migration depuis backup PM7 ===");
+ $this->log("Source: {$sourceDbName}");
+ $this->log("Cible: {$targetDbName}");
+ $this->log("Mode: {$mode}");
+```
+
+**NOUVEAU** :
+```php
+ public function __construct($env, $mode = 'global', $entityId = null, $logFile = null, $deleteBefore = false) {
+ // Validation de l'environnement
+ if (!isset(self::ENVIRONMENTS[$env])) {
+ throw new Exception("Invalid environment: $env. Use 'rca' or 'pra'");
+ }
+
+ $this->env = $env;
+ $config = self::ENVIRONMENTS[$env];
+ $this->sourceDbName = $config['source_db'];
+ $this->targetDbName = $config['target_db'];
+ $this->mode = $mode;
+ $this->entityId = $entityId;
+ $this->logFile = $logFile ?? '/var/back/migration_' . date('Ymd_His') . '.log';
+ $this->deleteBefore = $deleteBefore;
+
+ $this->log("=== Migration depuis backup PM7 ===");
+ $this->log("Environment: {$env}");
+ $this->log("Source: {$this->sourceDbName} → Cible: {$this->targetDbName}");
+ $this->log("Mode: {$mode}");
+```
+
+#### C. Modifier connect() (lignes 90-112)
+
+**Remplacer toutes les constantes** :
+- `self::DB_HOST` → `self::ENVIRONMENTS[$this->env]['host']`
+- `self::DB_PORT` → `self::ENVIRONMENTS[$this->env]['port']`
+- `self::DB_USER_ROOT` → `self::ENVIRONMENTS[$this->env]['user']`
+- `self::DB_PASS_ROOT` → `self::ENVIRONMENTS[$this->env]['pass']`
+- `self::DB_USER` → `self::ENVIRONMENTS[$this->env]['user']`
+- `self::DB_PASS` → `self::ENVIRONMENTS[$this->env]['pass']`
+
+#### D. Modifier parseArguments() (vers la fin du fichier)
+
+**ANCIEN** :
+```php
+ $args = [
+ 'source-db' => null,
+ 'target-db' => 'pra_geo',
+ 'mode' => 'global',
+ 'entity-id' => null,
+ 'log' => null,
+ 'delete-before' => true,
+ 'help' => false
+ ];
+```
+
+**NOUVEAU** :
+```php
+ $args = [
+ 'env' => 'rca', // Défaut: recette
+ 'mode' => 'global',
+ 'entity-id' => null,
+ 'log' => null,
+ 'delete-before' => true,
+ 'help' => false
+ ];
+```
+
+#### E. Modifier showHelp()
+
+**ANCIEN** :
+```php
+ --source-db=NAME Nom de la base source (backup restauré, ex: geosector_20251007) [REQUIS]
+ --target-db=NAME Nom de la base cible (défaut: pra_geo)
+```
+
+**NOUVEAU** :
+```php
+ --env=ENV Environment: 'rca' (recette) ou 'pra' (production) [défaut: rca]
+```
+
+**ANCIEN** (exemples) :
+```php
+ php migrate_from_backup.php --source-db=geosector_20251007 --target-db=pra_geo --mode=global
+```
+
+**NOUVEAU** :
+```php
+ php migrate_from_backup.php --env=pra --mode=global
+ php migrate_from_backup.php --env=rca --mode=entity --entity-id=45
+```
+
+#### F. Modifier validation des arguments
+
+**ANCIEN** :
+```php
+if (!$args['source-db']) {
+ echo "Erreur: --source-db est requis\n\n";
+ showHelp();
+ exit(1);
+}
+```
+
+**NOUVEAU** :
+```php
+if (!in_array($args['env'], ['rca', 'pra'])) {
+ echo "Erreur: --env doit être 'rca' ou 'pra'\n\n";
+ showHelp();
+ exit(1);
+}
+```
+
+#### G. Modifier instanciation BackupMigration
+
+**ANCIEN** :
+```php
+$migration = new BackupMigration(
+ $args['source-db'],
+ $args['target-db'],
+ $args['mode'],
+ $args['entity-id'],
+ $args['log'],
+ (bool)$args['delete-before']
+);
+```
+
+**NOUVEAU** :
+```php
+$migration = new BackupMigration(
+ $args['env'],
+ $args['mode'],
+ $args['entity-id'],
+ $args['log'],
+ (bool)$args['delete-before']
+);
+```
+
+---
+
+### 2. migrate_batch.sh
+
+#### A. Ajouter détection automatique de l'environnement (après ligne 22)
+
+**AJOUTER** :
+```bash
+# Détection automatique de l'environnement
+if [ -f "/etc/hostname" ]; then
+ CONTAINER_NAME=$(cat /etc/hostname)
+ case $CONTAINER_NAME in
+ rca-geo)
+ ENV="rca"
+ ;;
+ pra-geo)
+ ENV="pra"
+ ;;
+ *)
+ ENV="rca" # Défaut
+ ;;
+ esac
+else
+ ENV="rca" # Défaut
+fi
+```
+
+#### B. Remplacer lignes 26-27
+
+**ANCIEN** :
+```bash
+SOURCE_DB="geosector_20251013_13"
+TARGET_DB="pra_geo"
+```
+
+**NOUVEAU** :
+```bash
+# SOURCE_DB et TARGET_DB ne sont plus utilisés
+# Ils sont déduits de --env dans migrate_from_backup.php
+```
+
+#### C. Ajouter option --env dans le parsing (ligne 68)
+
+**AJOUTER avant `--interactive|-i)` ** :
+```bash
+ --env)
+ ENV="$2"
+ shift 2
+ ;;
+```
+
+#### D. Modifier les appels PHP - ligne 200-206
+
+**ANCIEN** :
+```bash
+ php "$MIGRATION_SCRIPT" \
+ --source-db="$SOURCE_DB" \
+ --target-db="$TARGET_DB" \
+ --mode=entity \
+ --entity-id="$SPECIFIC_ENTITY_ID" \
+ --log="$ENTITY_LOG" \
+ $DELETE_FLAG
+```
+
+**NOUVEAU** :
+```bash
+ php "$MIGRATION_SCRIPT" \
+ --env="$ENV" \
+ --mode=entity \
+ --entity-id="$SPECIFIC_ENTITY_ID" \
+ --log="$ENTITY_LOG" \
+ $DELETE_FLAG
+```
+
+#### E. Modifier les appels PHP - ligne 374-379
+
+**ANCIEN** :
+```bash
+ php "$MIGRATION_SCRIPT" \
+ --source-db="$SOURCE_DB" \
+ --target-db="$TARGET_DB" \
+ --mode=entity \
+ --entity-id="$ENTITY_ID" \
+ --log="$ENTITY_LOG" > /tmp/migration_output_$$.txt 2>&1
+```
+
+**NOUVEAU** :
+```bash
+ php "$MIGRATION_SCRIPT" \
+ --env="$ENV" \
+ --mode=entity \
+ --entity-id="$ENTITY_ID" \
+ --log="$ENTITY_LOG" > /tmp/migration_output_$$.txt 2>&1
+```
+
+#### F. Modifier les messages de log (lignes 289-291)
+
+**ANCIEN** :
+```bash
+log "📅 Date: $(date '+%Y-%m-%d %H:%M:%S')"
+log "📁 Source: $SOURCE_DB"
+log "📁 Cible: $TARGET_DB"
+```
+
+**NOUVEAU** :
+```bash
+log "📅 Date: $(date '+%Y-%m-%d %H:%M:%S')"
+log "🌍 Environment: $ENV"
+log "📁 Source: geosector → Target: (déduit de \$ENV)"
+```
+
+---
+
+## Nouveaux usages
+
+### Sur rca-geo (IN3)
+```bash
+# Détection automatique
+./migrate_batch.sh
+
+# Ou explicite
+./migrate_batch.sh --env=rca
+
+# Migration PHP directe
+php php/migrate_from_backup.php --env=rca --mode=entity --entity-id=45
+```
+
+### Sur pra-geo (IN4)
+```bash
+# Détection automatique
+./migrate_batch.sh
+
+# Ou explicite
+./migrate_batch.sh --env=pra
+
+# Migration PHP directe
+php php/migrate_from_backup.php --env=pra --mode=entity --entity-id=45
+```
diff --git a/api/scripts/README-migration.md b/api/scripts/README-migration.md
new file mode 100644
index 00000000..3befdfcf
--- /dev/null
+++ b/api/scripts/README-migration.md
@@ -0,0 +1,1925 @@
+# Documentation des Scripts de Migration GeoSector
+
+## 📋 Vue d'ensemble
+
+Ce dossier contient l'ensemble des scripts PHP permettant de migrer les données de l'ancienne base **geosector** (MySQL distante) vers la nouvelle base **pra_geo** (MariaDB sur maria4/IN4).
+
+### Statistiques
+- **Nombre total de scripts** : 21 fichiers
+- **Base source** : MySQL geosector (sauvegardée depuis PM7)
+- **Base cible** : MariaDB 11.4 pra_geo (maria4 sur IN4)
+- **Méthode** : Restauration backup PM7 → Migration sur maria4
+- **Ordre d'exécution** : Défini dans `migrate.php`
+
+---
+
+## 🚀 Guide Rapide de Migration depuis Backup PM7
+
+### Architecture de migration
+
+```
+PM7 (11.1.2.17)
+ └─ Backup nocturne chiffré : geosector_YYYYMMDD.sql.tar.gz.enc
+ ↓
+ Déchiffrement avec decpm7.sh
+ ↓
+ Transfert SCP vers IN4:/var/back/
+ ↓
+ Push vers maria4:/var/back/
+ ↓
+ Décompression et import dans maria4
+ ↓
+ Base geosector_YYYYMMDD créée dans maria4
+ ↓
+ Migration vers pra_geo (même serveur maria4)
+```
+
+### Étape 1️⃣ : Récupération et déchiffrement du backup depuis PM7
+
+**Sur le serveur PM7 (11.1.2.17)** :
+
+```bash
+# Se connecter à PM7
+ssh root@11.1.2.17
+
+# Aller dans le dossier des backups
+cd /var/back/
+
+# Lister les backups disponibles
+ls -lh geosector_*.sql.tar.gz.enc | tail -5
+
+# Identifier le dernier backup (exemple : geosector_20251007.sql.tar.gz.enc)
+BACKUP_DATE=$(date +%Y%m%d)
+BACKUP_FILE="geosector_${BACKUP_DATE}.sql.tar.gz.enc"
+
+# Déchiffrer le backup avec le script decpm7.sh
+./decpm7.sh ${BACKUP_FILE}
+
+# Résultat : fichier geosector_20251007.sql.tar.gz
+```
+
+### Étape 2️⃣ : Transfert du backup vers IN4
+
+**Depuis PM7 ou depuis votre poste local** :
+
+```bash
+# Définir les variables
+BACKUP_DATE=$(date +%Y%m%d)
+BACKUP_FILE="geosector_${BACKUP_DATE}.sql.tar.gz"
+
+# Transférer depuis PM7 vers IN4
+scp root@11.1.2.17:/var/back/${BACKUP_FILE} root@51.159.7.190:/var/back/
+
+# Vérifier la présence du fichier sur IN4
+ssh root@51.159.7.190 "ls -lh /var/back/${BACKUP_FILE}"
+```
+
+### Étape 3️⃣ : Push du backup vers le container maria4
+
+**Sur le serveur IN4 (51.159.7.190)** :
+
+```bash
+# Se connecter à IN4
+ssh root@51.159.7.190
+
+# Définir la variable
+BACKUP_DATE=$(date +%Y%m%d)
+BACKUP_FILE="geosector_${BACKUP_DATE}.sql.tar.gz"
+
+# Pousser le fichier vers le container maria4
+incus file push /var/back/${BACKUP_FILE} maria4/var/back/
+
+# Vérifier le fichier dans le container
+incus exec maria4 -- ls -lh /var/back/${BACKUP_FILE}
+```
+
+### Étape 4️⃣ : Décompression et import dans maria4
+
+**Dans le container maria4** :
+
+```bash
+# Se connecter au container maria4
+incus exec maria4 bash
+
+# Définir les variables
+BACKUP_DATE=$(date +%Y%m%d)
+BACKUP_FILE="geosector_${BACKUP_DATE}.sql.tar.gz"
+SQL_FILE="geosector_${BACKUP_DATE}.sql"
+
+# Décompresser l'archive
+cd /var/back
+tar -xzf ${BACKUP_FILE}
+
+# Vérifier la présence du fichier SQL
+ls -lh ${SQL_FILE}
+
+# Importer le SQL dans MariaDB
+# Note : Le fichier SQL contient déjà CREATE DATABASE et USE geosector_YYYYMMDD
+mariadb -u root -p'MyAlpLocal,90b' < ${SQL_FILE}
+
+# Vérifier la création de la base
+mariadb -u root -p'MyAlpLocal,90b' -e "SHOW DATABASES LIKE 'geosector_%';"
+
+# Vérifier le nombre de tables
+mariadb -u root -p'MyAlpLocal,90b' geosector_${BACKUP_DATE} -e "SHOW TABLES;"
+
+# Vérifier quelques comptages
+mariadb -u root -p'MyAlpLocal,90b' geosector_${BACKUP_DATE} -e "
+SELECT 'users' as table_name, COUNT(*) as count FROM users
+UNION ALL SELECT 'operations', COUNT(*) FROM operations
+UNION ALL SELECT 'ope_pass', COUNT(*) FROM ope_pass
+UNION ALL SELECT 'entites', COUNT(*) FROM users_entites;"
+
+# Sortir du container
+exit
+```
+
+### Étape 5️⃣ : Migration des données vers pra_geo
+
+**Important** : Le script PHP s'exécute sur le container **pra-geo** (pas maria4), car :
+- ✅ PHP 8.3 est disponible sur pra-geo
+- ✅ L'API GeoSector avec `ApiService::encryptData()` est présente
+- ✅ pra-geo se connecte à maria4 via l'IP 13.23.33.4
+
+**Option A : Migration globale (toutes les amicales)**
+
+```bash
+# Sur IN4, exécuter le script depuis le container pra-geo
+incus exec pra-geo bash
+
+# Aller dans le dossier de l'API
+cd /var/www/geosector/api
+
+# Lister les bases disponibles dans maria4 pour vérifier
+# (optionnel - pour voir les backups restaurés)
+echo "SHOW DATABASES LIKE 'geosector_%';" | \
+ mysql -h 13.23.33.4 -u root -p'MyAlpLocal,90b'
+
+# Lancer la migration complète
+php scripts/php/migrate_from_backup.php \
+ --source-db=geosector_20251007 \
+ --target-db=pra_geo \
+ --mode=global \
+ --log=/var/www/geosector/api/logs/migration_global_20251007.log
+
+# Suivre la progression dans les logs
+tail -f /var/www/geosector/api/logs/migration_global_20251007.log
+```
+
+**Option B : Migration par amicale (recommandé)**
+
+```bash
+# Sur IN4, dans le container pra-geo
+incus exec pra-geo bash
+
+# Aller dans le dossier de l'API
+cd /var/www/geosector/api
+
+# Lister les amicales disponibles dans la base source
+mysql -h 13.23.33.4 -u root -p'MyAlpLocal,90b' geosector_20251007 -e "
+SELECT rowid, libelle, cp, ville, active
+FROM users_entites
+WHERE active = 1
+ORDER BY rowid;"
+
+# Migrer une amicale spécifique (exemple : ID 45)
+php scripts/php/migrate_from_backup.php \
+ --source-db=geosector_20251007 \
+ --target-db=pra_geo \
+ --mode=entity \
+ --entity-id=45 \
+ --log=/var/www/geosector/api/logs/migration_entity_45_20251007.log
+
+# Suivre la progression dans un autre terminal
+incus exec pra-geo -- tail -f /var/www/geosector/api/logs/migration_entity_45_20251007.log
+```
+
+### Étape 6️⃣ : Vérification de la migration
+
+```bash
+# Depuis pra-geo ou directement sur IN4, vérifier les données migrées
+
+# Comparer les comptages source vs cible pour l'entité 45
+mysql -h 13.23.33.4 -u root -p'MyAlpLocal,90b' << 'EOF'
+SELECT
+ 'Users' as table_name,
+ (SELECT COUNT(*) FROM geosector_20251007.users WHERE fk_entite = 45) as source,
+ (SELECT COUNT(*) FROM pra_geo.users WHERE fk_entite = 45) as cible
+UNION ALL SELECT
+ 'Operations',
+ (SELECT COUNT(*) FROM geosector_20251007.operations WHERE fk_entite = 45),
+ (SELECT COUNT(*) FROM pra_geo.operations WHERE fk_entite = 45)
+UNION ALL SELECT
+ 'Passages',
+ (SELECT COUNT(*) FROM geosector_20251007.ope_pass
+ WHERE fk_operation IN (SELECT rowid FROM geosector_20251007.operations WHERE fk_entite = 45)),
+ (SELECT COUNT(*) FROM pra_geo.ope_pass
+ WHERE fk_operation IN (SELECT id FROM pra_geo.operations WHERE fk_entite = 45));
+EOF
+
+# Vérifier l'intégrité des montants
+mariadb -u root -p'MyAlpLocal,90b' pra_geo << 'EOF'
+SELECT p.fk_type, COUNT(*) as nb_passages, SUM(p.montant) as total_montant
+FROM ope_pass p
+JOIN operations o ON p.fk_operation = o.id
+WHERE o.fk_entite = 45
+GROUP BY p.fk_type;
+EOF
+```
+
+### Étape 7️⃣ : Nettoyage (optionnel)
+
+```bash
+# Supprimer la base temporaire geosector_YYYYMMDD après migration réussie
+mariadb -u root -p'MyAlpLocal,90b' -e "DROP DATABASE IF EXISTS geosector_20251007;"
+
+# Supprimer les fichiers de backup
+rm -f /var/back/geosector_20251007.sql.tar.gz
+rm -f /var/back/geosector_20251007.sql
+```
+
+---
+
+## 📝 Notes importantes
+
+### Sécurité et mot de passe decpm7.sh
+
+Le script `decpm7.sh` sur PM7 nécessite un mot de passe de déchiffrement. Assurez-vous de :
+- Connaître le mot de passe avant de lancer le déchiffrement
+- Ne jamais commiter ce mot de passe dans Git
+- Le stocker de manière sécurisée (gestionnaire de mots de passe)
+
+### Durée estimée de la migration
+
+| Taille de l'amicale | Temps estimé |
+|---------------------|--------------|
+| Petite (< 5000 passages) | 2-5 minutes |
+| Moyenne (5000-20000 passages) | 5-15 minutes |
+| Grande (> 20000 passages) | 15-60 minutes |
+
+### Espace disque requis
+
+- Backup chiffré : ~500 MB
+- Backup déchiffré : ~1.2 GB
+- Import dans MariaDB : ~2 GB
+- Total recommandé : **5 GB d'espace libre**
+
+### Gestion des erreurs courantes
+
+**Erreur : "Disk full"**
+```bash
+# Vérifier l'espace disque
+df -h /var/back
+
+# Nettoyer les anciens backups
+find /var/back -name "geosector_*.sql" -mtime +7 -delete
+```
+
+**Erreur : "Table already exists"**
+```bash
+# Si migration échouée à moitié, supprimer les données partielles
+mariadb -u root -p'MyAlpLocal,90b' pra_geo -e "
+DELETE FROM ope_pass WHERE fk_operation IN (SELECT id FROM operations WHERE fk_entite = 45);
+DELETE FROM operations WHERE fk_entite = 45;
+DELETE FROM users WHERE fk_entite = 45;
+DELETE FROM entites WHERE id = 45;"
+```
+
+---
+
+## 🏗️ Architecture
+
+### Fichiers de configuration
+
+#### `config.php`
+**Rôle** : Configuration principale des connexions bases de données et tunnel SSH
+
+**Fonctionnalités** :
+- Constantes de connexion SSH (host, port, clé privée)
+- Constantes de connexion base source (via tunnel SSH sur port 13306)
+- Constantes de connexion base cible (localhost:3306)
+- Fonctions utilitaires :
+ - `createSshTunnel()` : Établit le tunnel SSH
+ - `closeSshTunnel()` : Ferme le tunnel SSH
+ - `getSourceConnection()` : Retourne PDO vers base source
+ - `getTargetConnection()` : Retourne PDO vers base cible
+ - `logOperation()` : Journalise les opérations dans `logs/migration_YYYY-MM-DD.log`
+
+**⚠️ Problèmes identifiés** :
+- Mots de passe et clés SSH en dur (lignes 8-26)
+- Pas de gestion de fichier `.env`
+- Credentials visibles dans le code source
+
+#### `MigrationConfig.php`
+**Rôle** : Configuration simplifiée pour le chiffrement
+
+**Fonctionnalités** :
+- Classe `AppConfig` singleton
+- Fournit la clé de chiffrement pour les données sensibles
+- Pas de dépendance aux en-têtes HTTP (adapté pour CLI)
+
+#### `migrate.php`
+**Rôle** : Script orchestrateur principal
+
+**Fonctionnalités** :
+- Exécute toutes les migrations dans l'ordre
+- Gestion des arguments CLI (`--truncate`, `--create-table`, `--help`)
+- Peut exécuter une migration spécifique : `php migrate.php users`
+- Peut exécuter toutes les migrations : `php migrate.php`
+- Affiche des statistiques de migration (temps, succès, erreurs)
+
+**Ordre d'exécution** :
+1. Tables de référence `x_*` (devises, types, pays, régions, départements, villes)
+2. `entites` (entities)
+3. `users` (utilisateurs)
+4. `operations` (opérations)
+5. `ope_sectors` (secteurs d'opération)
+6. `sectors_adresses` (adresses de secteurs)
+7. `ope_users` (associations utilisateurs-opérations)
+8. `ope_users_sectors` (associations utilisateurs-secteurs)
+9. `ope_pass` (passages)
+10. `ope_pass_histo` (historique des passages)
+11. `medias` (fichiers médias)
+
+---
+
+## 🔄 Différences Structurelles entre les Bases
+
+### Vue d'ensemble
+
+**Base source (geosector)** - MariaDB 10.11.9 :
+- Serveur version : `10.11.9-MariaDB-deb12`
+- Clés primaires : `rowid` (int(11))
+- Statut : `active` (tinyint(1))
+- Dates : `date_creat`, `date_modif` (datetime)
+- **Données en clair** (pas de chiffrement)
+- **Pas de contraintes FK définies**
+
+**Base cible (geo_app)** - MariaDB 11.4.5 :
+- Serveur version : `11.4.5-MariaDB`
+- Clés primaires : `id` (int(10) unsigned)
+- Statut : `chk_active` (tinyint(1) unsigned)
+- Dates : `created_at`, `updated_at` (timestamp avec ON UPDATE CURRENT_TIMESTAMP)
+- **Données chiffrées** : `encrypted_name`, `encrypted_email`, `encrypted_phone`, `encrypted_user_name`, etc.
+- **Contraintes FK définies** avec ON DELETE/ON UPDATE CASCADE
+
+### Mappings globaux systématiques
+
+```
+rowid → id
+active → chk_active
+date_creat → created_at
+date_modif → updated_at
+date_eve → passed_at (ope_pass uniquement)
+support_rowid → support_id (medias uniquement)
+```
+
+### Tables absentes de la source (nouvelles fonctionnalités)
+
+Ces tables n'existent que dans `geo_app` et **ne sont pas migrées** :
+
+#### Système de chat (4 tables)
+- `chat_rooms` : Salles de conversation
+- `chat_messages` : Messages
+- `chat_participants` : Participants aux conversations
+- `chat_read_receipts` : Accusés de lecture
+
+#### Intégration Stripe (7 tables)
+- `stripe_accounts` : Comptes Stripe connectés
+- `stripe_terminal_readers` : Lecteurs de cartes (Tap to Pay)
+- `stripe_android_certified_devices` : Devices Android certifiés
+- `stripe_payment_history` : Historique des paiements
+- `stripe_refunds` : Remboursements
+- `stripe_webhooks` : Événements webhook Stripe
+
+#### Sécurité et monitoring (4 tables)
+- `sec_alerts` : Alertes de sécurité
+- `sec_blocked_ips` : IPs bloquées
+- `sec_failed_login_attempts` : Tentatives de connexion échouées
+- `sec_performance_metrics` : Métriques de performance
+
+#### Autres nouvelles tables
+- `user_devices` : Informations des devices mobiles
+- `x_departements_contours` : Contours géographiques des départements
+- `x_users_titres` : Titres des utilisateurs (Mme, M., etc.)
+
+### Tables renommées
+
+| Source | Cible | Notes |
+|--------|-------|-------|
+| `users_entites` | `entites` | Migration via `migrate_entites.php` |
+| `sectors` | `ope_sectors` | Intégrée dans les opérations |
+
+### Différences par table migrée
+
+#### ⚠️ `x_villes`
+**Modifications structurelles** :
+- Source : `cp` → Cible : `code_postal`
+- Source : `departement` (varchar 65) → **Supprimé** (redondant avec `fk_departement`)
+
+#### ⚠️ `x_departements`
+**Ajouts dans cible** :
+- `dept_limitrophes` (varchar 100) : Départements limitrophes
+- `contour` (geometry) : Contour géographique
+
+#### ✅ `entites` (source: `users_entites`)
+**Champs chiffrés** :
+- `libelle` → `encrypted_name`
+- `email` → `encrypted_email`
+- `tel1`, `tel2` → `encrypted_phone`, `encrypted_mobile` (avec détection 06/07)
+- `iban` → `encrypted_iban`
+- `bic` → `encrypted_bic`
+
+**Nouveaux champs** :
+- `chk_stripe` (tinyint) : Intégration Stripe activée
+- `encrypted_stripe_id` (varchar 255) : ID Stripe chiffré
+- `chk_username_manuel` (tinyint) : Gestion usernames manuelle/auto
+- `chk_user_delete_pass` (tinyint) : Autorisation suppression passages
+- `chk_lot_actif` (tinyint) : Lots actifs
+
+**Champs supprimés** :
+- Tous les champs métier spécifiques (`appname`, `http_host`, `tva_intra`, `rcs`, `siret`, `ape`, `couleur`, `prefecture`, `gerant_*`, `banque_*`, `genbase`, `groupebase`, `userbase`, `passbase`, `demo`, `lib_*`, `icon_*`, `btn_width`, `nbmembres`, `nbconnex`)
+
+#### ✅ `users`
+**Champs chiffrés** :
+- `libelle` → `encrypted_name`
+- `username` → `encrypted_user_name` (chiffrement recherchable)
+- `telephone` → `encrypted_phone`
+- `mobile` → `encrypted_mobile`
+- `email` → `encrypted_email` (chiffrement recherchable)
+
+**Mappings spécifiques** :
+- `userpswd` ou `userpass` → `user_pass_hash`
+- `prenom` → `first_name`
+- `nom_tournee` → `sect_name`
+- `alert_email` → `chk_alert_email`
+
+**Champs supprimés** :
+- Nombreux champs métier : `num_adherent`, `libelle_naissance`, `josh`, `email_secondaire`, `infos`, `ltt`, `lng`, `sector`, `dept_naissance`, `commune_naissance`, `anciennete`, `fk_categorie`, `fk_sous_categorie`, `adresse_*`, `cp`, `ville`, `matricule`, `fk_grade`, `chk_adherent_*`, `chk_archive`, `chk_double_affectation`
+
+#### ✅ `operations`
+**Modifications** :
+- `chk_api_adresse` : **Supprimé** dans cible
+- Dates : `date_deb`, `date_fin` changent de `date` à `NOT NULL DEFAULT '0000-00-00'`
+
+#### ✅ `ope_sectors`
+**Nouveaux champs** :
+- `fk_old_sector` (int unsigned) : Référence à l'ancien `sectors.rowid` pour le mapping
+
+#### ✅ `ope_users`
+**Nouveaux champs dans cible** :
+- `fk_role` (int unsigned) : Rôle de l'utilisateur dans l'opération
+- `first_name` (varchar 45) : Prénom
+- `encrypted_name` (varchar 255) : Nom chiffré
+- `sect_name` (varchar 60) : Nom de tournée
+
+**Impact** : Le script `migrate_ope_users.php` doit remplir ces champs ou les laisser vides
+
+#### ✅ `ope_pass`
+**Champs chiffrés** :
+- `libelle` → `encrypted_name`
+- `email` → `encrypted_email` (chiffrement recherchable)
+- `phone` → `encrypted_phone`
+
+**Nouveaux champs dans cible** :
+- `residence` (varchar 75) : Nom de la résidence
+- `date_recu` (timestamp) : Date de réception
+- `date_creat_recu` (timestamp) : Date de création du reçu
+- `date_sent_recu` (timestamp) : Date d'envoi du reçu
+- `stripe_payment_id` (varchar 50) : ID du PaymentIntent Stripe
+
+**Champs supprimés** :
+- `lieudit` (varchar 75)
+- `chk_habitat_vide` (tinyint)
+- `lot_nb_passages` (int)
+
+**Mappings spécifiques** :
+- `recu` → `nom_recu`
+- `date_eve` → `passed_at`
+- `fk_type` : transformation 8→5, 9→6
+
+#### ✅ `ope_pass_histo`
+**Champ supprimé** :
+- `fk_user` : **N'existe plus** dans la nouvelle structure
+
+#### ✅ `medias`
+**Mapping spécifique** :
+- `support_rowid` → `support_id`
+
+**Nouveaux champs dans cible** :
+- `fk_entite` (int unsigned) : Propriétaire du média
+- `fk_operation` (int unsigned) : Opération liée
+- `file_type`, `file_category`, `file_size`, `mime_type` : Métadonnées fichier
+- `original_name`, `file_path` : Informations fichier
+- `original_width`, `original_height`, `processed_width`, `processed_height` : Dimensions images
+- `is_processed` : Statut traitement image
+
+**Champs supprimés** :
+- `dir0`, `dir1`, `dir2` : Ancienne structure de dossiers
+- `type_fichier`, `position`, `hauteur`, `largeur`, `niveaugris` : Anciens champs métier
+
+#### ✅ `sectors_adresses`
+**Nouveaux champs dans cible** :
+- `id` (auto-increment) : Clé primaire ajoutée
+- `osm_id` (int) : ID OpenStreetMap
+- `osm_name` (varchar 50) : Nom OSM
+- `osm_date_creat` (timestamp) : Date de création OSM
+- `created_at`, `updated_at` : Timestamps standards
+
+**Mappings** :
+- `fk_sector` : ancien ID → nouvel ID via mapping `ope_sectors`
+
+### Tables non migrées de la source
+
+Ces tables existent dans `geosector` mais **ne sont pas migrées** vers `geo_app` :
+
+- `articles`, `articles_pages` : Système d'articles (obsolète ?)
+- `blog_articles`, `blog_pages` : Système de blog (obsolète ?)
+- `email_counter`, `email_queue` : File d'attente emails (recréée dans cible)
+- `ope_pass_recus` : Table séparée des reçus (intégrée dans `ope_pass`)
+- `ope_users_suivis` : Suivi GPS des utilisateurs (archivé ?)
+- `operations_docs`, `operations_eve_docs` : Documents opérations (archivé ?)
+- `params` : Paramètres globaux (reconfiguré ?)
+- `sectors`, `sectors_streets` : Tables sectors (transformée en `ope_sectors`)
+- `users_lastpos` : Dernière position utilisateurs (archivé ?)
+- `x_civilites` : Civilités (remplacé par `x_users_titres`)
+- `x_users_categories`, `x_users_sous_categories`, `x_users_grades` : Catégories utilisateurs (supprimé)
+- `y_conf`, `y_menus`, `y_modules`, `y_modules_rules`, `y_pages` : Configuration interface (obsolète)
+- `z_logs`, `z_sessions`, `z_stats` : Logs et sessions (recréés dans cible)
+
+### Contraintes de clés étrangères
+
+**Source** : Aucune contrainte FK définie explicitement
+
+**Cible** : Toutes les FK définies avec :
+- `ON UPDATE CASCADE` : Mise à jour en cascade
+- `ON DELETE CASCADE` ou `ON DELETE SET NULL` : Suppression gérée
+
+**Impact migration** : Les scripts doivent respecter l'ordre des dépendances pour éviter les erreurs d'intégrité référentielle.
+
+### Vues dans geo_app
+
+- `chat_rooms_with_last_message` : Vue des salles de chat avec dernier message
+- `v_stripe_entite_stats` : Statistiques Stripe par entité
+- `v_stripe_payment_stats` : Statistiques paiements Stripe
+
+**Note** : Les vues ne contiennent pas de données à migrer.
+
+---
+
+## 📊 Analyse Script par Script
+
+### 1. Tables de référence `x_*`
+
+#### `migrate_x_devises.php`
+**Table** : `x_devises` (Devises)
+**Mappings** :
+- `rowid` → `id`
+- `active` → `chk_active`
+
+**Particularités** :
+- Crée automatiquement la table si elle n'existe pas
+- Utilise `logOperation()` pour le logging
+- Pattern : `ON DUPLICATE KEY UPDATE`
+
+**Statut** : ✅ Fonctionnel
+
+---
+
+#### `migrate_x_entites_types.php`
+**Table** : `x_entites_types` (Types d'entités)
+**Mappings** :
+- `rowid` → `id`
+- `active` → `chk_active`
+
+**Particularités** :
+- Logging simple via `echo`
+- Pattern : `ON DUPLICATE KEY UPDATE`
+
+**Statut** : ✅ Fonctionnel
+
+---
+
+#### `migrate_x_types_passages.php`, `migrate_x_types_reglements.php`, `migrate_x_users_roles.php`
+**Tables** : Types de passages, types de règlements, rôles utilisateurs
+**Pattern** : Identique aux autres tables `x_*`
+
+**Statut** : ✅ Fonctionnel (non lu mais présumé identique)
+
+---
+
+#### `migrate_x_pays.php`
+**Table** : `x_pays` (Pays)
+**Mappings** :
+- `rowid` → `id`
+- `active` → `chk_active`
+- Conservation de `code`, `fk_continent`, `fk_devise`, `libelle`
+
+**Statut** : ✅ Fonctionnel
+
+---
+
+#### `migrate_x_regions.php`
+**Table** : `x_regions` (Régions)
+**Mappings** :
+- `rowid` → `id`
+- `active` → `chk_active`
+- Conservation de tous les champs métier (`fk_pays`, `libelle`, `libelle_long`, `table_osm`, `departements`)
+
+**Statut** : ✅ Fonctionnel
+
+---
+
+#### `migrate_x_departements.php`, `migrate_x_villes.php`
+**Tables** : Départements, Villes
+**Pattern** : Identique aux autres tables `x_*`
+
+**Statut** : ✅ Fonctionnel (non lu mais présumé identique)
+
+---
+
+### 2. Tables métier principales
+
+#### `migrate_entites.php`
+**Table source** : `users_entites`
+**Table cible** : `entites`
+
+**Mappings** :
+- `rowid` → `id`
+- `active` → `chk_active`
+- `libelle` → `encrypted_name` (chiffré)
+- `tel1`, `tel2` → `encrypted_phone`, `encrypted_mobile` (logique de détection 06/07)
+- `email` → `encrypted_email` (chiffré et recherchable)
+- `iban` → `encrypted_iban` (chiffré)
+- `bic` → `encrypted_bic` (chiffré)
+- `cp` → `code_postal`
+
+**Particularités** :
+- **Chiffrement** : Utilise `ApiService::encryptData()` et `ApiService::encryptSearchableData()`
+- **Logique téléphones** : Détecte mobiles (06/07) vs fixes
+- **Valeur par défaut** : `chk_demo = 0` (forcé)
+
+**Statut** : ✅ Fonctionnel
+
+---
+
+#### `migrate_users.php`
+**Table** : `users`
+
+**Mappings** :
+- `rowid` → `id`
+- `active` → `chk_active`
+- `libelle` → `encrypted_name` (chiffré)
+- `prenom` → `first_name`
+- `nom_tournee` → `sect_name`
+- `username` → `encrypted_user_name` (chiffré et recherchable)
+- `userpswd` ou `userpass` → `user_pass_hash`
+- `telephone` → `encrypted_phone` (chiffré)
+- `mobile` → `encrypted_mobile` (chiffré)
+- `email` → `encrypted_email` (chiffré et recherchable)
+- `alert_email` → `chk_alert_email`
+
+**Particularités** :
+- **Tests de chiffrement** : Pour les 100 premiers utilisateurs (lignes 130-158)
+ - Chiffre et déchiffre email et username
+ - Affiche les valeurs pour vérification
+ - **⚠️ Impact performance** : Ralentit la migration
+- **Gestion rôle** : Force `fk_role=1` si `fk_role=0`
+- **Gestion titre** : Force `fk_titre=1` si différent de 1 ou 2
+
+**⚠️ PROBLÈME CRITIQUE** (lignes 227-239) :
+```php
+if ($exists) {
+ $insertStmt->execute($userData); // OK : update
+ $successCount++;
+} else {
+ $errorCount++; // ❌ ERREUR : devrait insérer, pas compter comme erreur
+}
+```
+**Conséquence** : Les nouveaux utilisateurs ne sont jamais insérés, comptés comme erreurs
+
+**Statut** : 🔴 Bug critique - logique d'insertion inversée
+
+---
+
+#### `migrate_operations.php`
+**Table** : `operations`
+
+**Mappings** :
+- `rowid` → `id`
+- `active` → `chk_active`
+- `date_creat` → `created_at`
+- `date_modif` → `updated_at`
+
+**Particularités** :
+- **Limitation arbitraire** : Ne migre que les **3 dernières opérations par entité** (lignes 54-70)
+- Vérifie que les entités référencées ont été migrées
+- Filtre basé sur `fk_entite IN (IDs des entités migrées)`
+- Option `--truncate` pour vider la table avant migration
+
+**⚠️ Problème** :
+- Limite de 3 opérations non documentée et non paramétrable
+- Pourrait perdre des données historiques importantes
+
+**Statut** : ⚠️ Limitation fonctionnelle - à valider métier
+
+---
+
+#### `migrate_ope_sectors.php`
+**Table** : `ope_sectors`
+
+**Mappings** :
+- Fusionne données de `sectors` et `ope_users_sectors`
+- `rowid` de `sectors` → `fk_old_sector`
+- Génère un nouvel `id` auto-incrémenté
+- Conservation de `libelle`, `sector` (géométrie), `color`
+
+**Particularités** :
+- Crée une correspondance `fk_operation + fk_old_sector → id`
+- Utile pour les migrations suivantes (ope_users_sectors, ope_pass)
+- Ne migre que les secteurs liés aux opérations migrées
+- Filtre `active = 1` sur source
+
+**Statut** : ✅ Fonctionnel
+
+---
+
+#### `migrate_sectors_adresses.php`
+**Table** : `sectors_adresses`
+
+**Mappings** :
+- `fk_sector` (ancien ID) → `fk_sector` (nouvel ID via mapping)
+- Ajout de colonnes OSM avec valeurs par défaut :
+ - `osm_id = 0`
+ - `osm_name = ''`
+ - `osm_date_creat = '0000-00-00 00:00:00'`
+
+**Particularités** :
+- Utilise le mapping créé par `migrate_ope_sectors.php`
+- Recherche `fk_old_sector → id` dans `ope_sectors`
+- Ignore les adresses dont le secteur n'a pas été migré
+
+**Statut** : ✅ Fonctionnel
+
+---
+
+#### `migrate_ope_users.php`
+**Table** : `ope_users`
+
+**Mappings** :
+- `rowid` → `id`
+- `active` → `chk_active`
+- `date_creat` → `created_at`
+- `date_modif` → `updated_at`
+
+**Particularités** :
+- Vérifie que `fk_operation` et `fk_user` existent dans les tables cibles
+- Double filtrage par IDs migrés
+- Ne migre que les associations valides
+
+**Statut** : ✅ Fonctionnel
+
+---
+
+#### `migrate_ope_users_sectors.php`
+**Table** : `ope_users_sectors`
+
+**Mappings** :
+- `fk_sector` (ancien ID) → `fk_sector` (nouvel ID via mapping ope_sectors)
+- Pas de `rowid` → utilise clé composite `(fk_operation, fk_user, fk_sector)`
+
+**Particularités** :
+- Triple vérification :
+ 1. Opération migrée
+ 2. Utilisateur migré
+ 3. Secteur existe dans mapping ope_sectors
+- Génère la clé de recherche : `fk_operation . '_' . fk_old_sector`
+- Compteur `skipped` pour associations ignorées
+
+**Statut** : ✅ Fonctionnel
+
+---
+
+#### `migrate_ope_pass.php` ⭐ Script le plus complexe
+**Table** : `ope_pass` (Passages)
+
+**Mappings** :
+- `date_eve` → `passed_at` 📅
+- `libelle` → `encrypted_name` 🔒 (chiffré)
+- `email` → `encrypted_email` 🔒 (chiffré et recherchable)
+- `phone` → `encrypted_phone` 🔒 (chiffré)
+- `recu` → `nom_recu`
+- `fk_sector` (ancien ID) → `fk_sector` (nouvel ID via mapping)
+- `fk_type` : transformation 8→5, 9→6
+
+**Particularités** :
+- **Gestion avancée des timeouts** (lignes 46-80) :
+ - Configure `PDO::ATTR_TIMEOUT = 600` (10 minutes)
+ - Configure variables MariaDB 10.11 :
+ - `wait_timeout = 3600` (1h)
+ - `net_read_timeout = 3600` (1h)
+ - `net_write_timeout = 3600` (1h)
+ - `innodb_lock_wait_timeout = 3600` (1h)
+- **Suppression des contraintes FK** avant migration (lignes 98-120)
+- **Suppression par lots** : Delete par lots de 100 000 (lignes 136-160)
+- **Migration par lots** : 5 000 passages par lot (lignes 238-543)
+- **Transactions par lot** : `BEGIN TRANSACTION` → traitement → `COMMIT`
+- **Garbage collector** : Appel explicite `gc_collect_cycles()` pour libérer mémoire
+- **Validation email** : `filter_var($email, FILTER_VALIDATE_EMAIL)` avant chiffrement
+- **Gestion type_reglement** : Force à 4 si différent de 1, 2 ou 3
+- **Recréation des FK** après migration (lignes 547-575)
+
+**⚠️ Points d'attention** :
+- Tue le processus SSH sur port 13306 au démarrage (ligne 23)
+- Désactive `FOREIGN_KEY_CHECKS` pendant suppression
+- Gestion silencieuse des passages dont secteur/utilisateur non migré
+
+**Statut** : ✅ Fonctionnel - Optimisé pour gros volumes
+
+---
+
+#### `migrate_ope_pass_histo.php`
+**Table** : `ope_pass_histo`
+
+**Mappings** :
+- `rowid` (non conservé, auto-increment)
+- `date_histo` (conversion datetime)
+- `fk_user` → **supprimé** (n'existe plus dans nouvelle structure)
+
+**Particularités** :
+- Ne migre que si `fk_pass` existe dans `ope_pass` cible
+- Suppression complète avant migration (`DELETE FROM ope_pass_histo`)
+- Mode "silencieux" : affiche uniquement les erreurs
+
+**Statut** : ✅ Fonctionnel
+
+---
+
+#### `migrate_medias.php`
+**Table** : `medias`
+
+**Mappings** :
+- `support_rowid` → `support_id` 📝
+- `date_creat` → `created_at`
+- `date_modif` → `updated_at`
+
+**Particularités** :
+- Vérifie que `fk_user_creat` et `fk_user_modif` existent
+- Force à `0` (système) si utilisateur non migré
+- Suppression complète avant migration
+- Pas de chiffrement (fichiers référencés, pas de données sensibles)
+
+**Statut** : ✅ Fonctionnel
+
+---
+
+## 🔍 Problèmes Identifiés par Priorité
+
+### 🔴 CRITIQUES (Bloquants)
+
+#### 1. **Bug logique `migrate_users.php` (lignes 227-239)**
+**Impact** : Aucun nouvel utilisateur ne peut être inséré
+```php
+// Logique actuelle (INCORRECTE)
+if ($exists) {
+ $insertStmt->execute($userData); // Update OK
+ $successCount++;
+} else {
+ $errorCount++; // ❌ Devrait faire INSERT
+}
+```
+**Solution** :
+```php
+// Logique corrigée
+$insertStmt->execute($userData); // ON DUPLICATE KEY fait le travail
+$successCount++;
+```
+
+---
+
+### ⚠️ MAJEURS (Fonctionnels)
+
+#### 2. **Limitation arbitraire opérations `migrate_operations.php` (lignes 54-70)**
+**Impact** : Perte potentielle de données historiques
+- Ne migre que 3 dernières opérations par entité
+- Pas paramétrable
+- Pas documenté dans le code
+
+**Solution proposée** :
+- Ajouter paramètre `--limit-operations=N` (défaut : 0 = toutes)
+- Documenter dans `--help`
+
+---
+
+#### 3. **Tests de chiffrement dans production `migrate_users.php` (lignes 130-158)**
+**Impact** : Ralentissement x2 de la migration users
+- Teste chiffrement/déchiffrement pour 100 premiers users
+- S'exécute à chaque migration
+
+**Solution proposée** :
+- Créer flag `--test-encryption`
+- Désactiver par défaut
+
+---
+
+#### 4. **Configuration sensible en dur `config.php`**
+**Impact** : Sécurité compromise
+- Mots de passe en clair
+- Clé SSH en dur
+- Visible dans Git
+
+**Solution proposée** :
+- Créer `scripts/.env.example`
+- Utiliser `vlucas/phpdotenv` ou parser manuel
+- Ajouter `scripts/.env` au `.gitignore`
+
+---
+
+### ℹ️ MINEURS (Améliorations)
+
+#### 5. **Incohérence logging**
+**Impact** : Difficulté debugging
+- Certains scripts : `logOperation()`
+- D'autres : `echo` direct
+
+**Solution** : Uniformiser avec `logOperation()` partout
+
+---
+
+#### 6. **Pas de transaction globale**
+**Impact** : État incohérent si échec en milieu de migration
+- Chaque table = migration indépendante
+- Si échec table N, tables 1..N-1 déjà modifiées
+
+**Solution proposée** :
+- Option `--transactional` pour tout englobé dans 1 transaction
+- Par défaut : comportement actuel (plus sûr)
+
+---
+
+#### 7. **Gestion FK manuelle dans `migrate_ope_pass.php`**
+**Impact** : Complexité maintenance
+- Désactivation/réactivation manuelle
+- Suppression/recréation manuelle
+
+**Solution** :
+- Vérifier si vraiment nécessaire
+- Documenter pourquoi (gros volumes)
+
+---
+
+#### 8. **Dates avec valeurs `0000-00-00`**
+**Impact** : Warnings MariaDB en mode strict
+- Plusieurs scripts utilisent `'0000-00-00'` ou `'0000-00-00 00:00:00'`
+- MariaDB 10.11 en mode strict refuse ces valeurs
+
+**Solution** : Remplacer par `NULL`
+
+---
+
+#### 9. **Pas de rapport détaillé post-migration**
+**Impact** : Difficile de valider la migration
+- Pas de récapitulatif des données migrées
+- Pas de comparaison source vs cible
+
+**Solution proposée** :
+- Créer script `scripts/php/verify_migration.php`
+- Compare counts par table
+- Liste les incohérences
+
+---
+
+## ✅ Points Positifs
+
+1. ✨ **Idempotence** : Utilisation systématique de `ON DUPLICATE KEY UPDATE`
+2. 🔒 **Sécurité données** : Chiffrement des données sensibles
+3. 🎯 **Filtrage intelligent** : Ne migre que les données liées (pas d'orphelins)
+4. 📊 **Optimisation gros volumes** : Migration par lots (`ope_pass`)
+5. 🔗 **Gestion dépendances** : Ordre d'exécution respecté
+6. 🧹 **Nettoyage mémoire** : Garbage collection explicite
+7. 📝 **Logging** : Historique des migrations dans `logs/`
+8. 🚀 **Tunnel SSH automatique** : Connexion transparente
+
+---
+
+## ⚙️ Incohérences Détectées (Scripts vs Structures Réelles)
+
+Suite à l'analyse comparative des scripts de migration avec les structures SQL réelles, voici les incohérences identifiées :
+
+### 🔴 CRITIQUES
+
+#### 1. `migrate_users.php` - Logique d'insertion inversée
+**Lignes** : 227-239
+**Problème** : Les nouveaux utilisateurs ne sont jamais insérés
+```php
+// Code actuel (INCORRECT)
+if ($exists) {
+ $insertStmt->execute($userData); // Update OK
+ $successCount++;
+} else {
+ $errorCount++; // ❌ Devrait faire INSERT, pas compter comme erreur
+}
+```
+**Impact** : **BLOQUANT** - Impossible d'ajouter de nouveaux utilisateurs
+**Solution** : Utiliser directement `ON DUPLICATE KEY UPDATE` sans vérification préalable
+
+---
+
+### ⚠️ MAJEURS
+
+#### 2. `migrate_x_departements.php` - Nouveaux champs non remplis
+**Champs manquants** :
+- `dept_limitrophes` (varchar 100) : Départements limitrophes
+- `contour` (geometry) : Contour géographique
+
+**Impact** : Ces champs resteront `NULL` après migration
+**Solution** : Acceptable si ces données seront ajoutées ultérieurement
+
+---
+
+#### 3. `migrate_ope_users.php` - Nouveaux champs non remplis
+**Champs manquants** :
+- `fk_role` (int) : Rôle de l'utilisateur dans l'opération
+- `first_name` (varchar 45) : Prénom
+- `encrypted_name` (varchar 255) : Nom chiffré
+- `sect_name` (varchar 60) : Nom de tournée
+
+**Impact** : Fonctionnalités limitées - ces informations manqueront dans les opérations
+**Solution** : Enrichir le script pour copier ces données depuis `users` si disponibles
+
+---
+
+#### 4. `migrate_medias.php` - Métadonnées fichiers manquantes
+**Champs manquants** :
+- `fk_entite`, `fk_operation` : Liens vers entités/opérations
+- `file_type`, `file_category`, `file_size`, `mime_type` : Métadonnées
+- `original_name`, `file_path` : Informations fichier
+- `original_width`, `original_height`, `processed_width`, `processed_height` : Dimensions images
+- `is_processed` : Statut traitement
+
+**Impact** : Fonctionnalités de gestion de médias limitées
+**Solution** :
+- Analyser les fichiers existants pour extraire les métadonnées
+- Déduire `fk_entite` et `fk_operation` depuis `support` et `support_id`
+
+---
+
+### ℹ️ MINEURS (Informations)
+
+#### 5. `migrate_x_villes.php` - Champ `departement` ignoré
+**Champ supprimé** : `departement` (varchar 65)
+**Raison** : Redondant avec `fk_departement`
+**Impact** : ✅ Correct - pas de perte de données
+
+---
+
+#### 6. `migrate_ope_pass.php` - Nouveaux champs Stripe
+**Champs non remplis** :
+- `residence` (varchar 75)
+- `date_recu`, `date_creat_recu`, `date_sent_recu` (timestamps)
+- `stripe_payment_id` (varchar 50)
+
+**Impact** : Attendu - ces fonctionnalités sont nouvelles
+**Solution** : ✅ Aucune action requise - champs remplis lors de l'utilisation future
+
+---
+
+#### 7. `migrate_entites.php` - Nouveaux champs Stripe/Config
+**Champs non remplis** :
+- `chk_stripe`, `encrypted_stripe_id` : Intégration Stripe
+- `chk_username_manuel` : Gestion usernames
+- `chk_user_delete_pass` : Autorisation suppression
+- `chk_lot_actif` : Gestion lots
+
+**Impact** : Attendu - nouvelles fonctionnalités
+**Solution** : ✅ Valeurs par défaut appropriées définies dans la structure
+
+---
+
+### 📊 Résumé des vérifications
+
+| Script | Statut | Problèmes critiques | Problèmes majeurs | Avertissements |
+|--------|--------|---------------------|-------------------|----------------|
+| `migrate_users.php` | 🔴 | 1 | 0 | 0 |
+| `migrate_x_departements.php` | ⚠️ | 0 | 1 | 0 |
+| `migrate_ope_users.php` | ⚠️ | 0 | 1 | 0 |
+| `migrate_medias.php` | ⚠️ | 0 | 1 | 0 |
+| `migrate_x_villes.php` | ✅ | 0 | 0 | 1 |
+| `migrate_ope_pass.php` | ✅ | 0 | 0 | 1 |
+| `migrate_entites.php` | ✅ | 0 | 0 | 1 |
+| Autres scripts `x_*` | ✅ | 0 | 0 | 0 |
+
+### 🛠️ Outil de vérification
+
+Un script de vérification automatique a été créé : **`verify_migration_structure.php`**
+
+**Usage** :
+```bash
+php scripts/php/verify_migration_structure.php
+```
+
+**Fonctionnalités** :
+- Compare les colonnes source vs cible pour chaque table
+- Identifie les colonnes non mappées
+- Liste les nouvelles colonnes qui seront NULL
+- Affichage coloré avec compteurs d'erreurs/avertissements
+
+---
+
+---
+
+## 🚀 Migration via Endpoint API (Approche Recommandée)
+
+### Vue d'ensemble
+
+Au lieu d'exécuter les scripts PHP en ligne de commande, nous recommandons d'utiliser un **endpoint API REST** qui permet de migrer **UNE entité (amicale) à la fois** de manière contrôlée et testable.
+
+### Avantages de cette approche
+
+✅ **Migration progressive** : Une entité à la fois, avec validation entre chaque étape
+✅ **Interface utilisateur** : Suivi visuel de la progression depuis Flutter/Web
+✅ **Tests granulaires** : Vérification table par table avant de continuer
+✅ **Rollback possible** : Annulation par entité en cas de problème
+✅ **Logs détaillés** : Traçabilité complète dans l'API
+✅ **Sécurité renforcée** : Authentification et autorisation via l'API
+✅ **Moins risqué** : Pas de migration globale "big bang"
+
+### Architecture de l'endpoint
+
+#### Endpoint principal
+```http
+POST /api/migrations/entity
+Authorization: Bearer {session_id}
+Content-Type: application/json
+
+{
+ "entity_id": 45, // ID de l'entité dans l'ancienne base
+ "steps": ["users", "operations", "ope_pass"], // Étapes à exécuter (optionnel)
+ "dry_run": false, // Mode simulation (optionnel)
+ "truncate": false // Vider les tables cible avant migration (optionnel)
+}
+```
+
+#### Réponse
+```json
+{
+ "status": "success",
+ "entity_id": 45,
+ "entity_name": "Amicale de Grenoble",
+ "migration_id": "mig_abc123",
+ "steps_completed": [
+ {
+ "step": "x_devises",
+ "status": "success",
+ "records_migrated": 1,
+ "duration_ms": 45
+ },
+ {
+ "step": "users",
+ "status": "success",
+ "records_migrated": 38,
+ "duration_ms": 234
+ }
+ ],
+ "total_duration_ms": 1234,
+ "summary": {
+ "total_records": 156,
+ "total_errors": 0,
+ "total_warnings": 2
+ }
+}
+```
+
+### Plan de migration étape par étape
+
+#### Phase 0️⃣ : Préparation
+
+**Actions à effectuer :**
+- [ ] Créer le controller `MigrationController.php`
+- [ ] Créer le service `MigrationService.php`
+- [ ] Ajouter les routes dans `index.php`
+- [ ] Configurer la connexion à la base source dans `AppConfig.php`
+- [ ] Tester la connexion aux deux bases
+
+**Tests de préparation :**
+```bash
+# Tester la connexion aux bases
+GET /api/migrations/test-connections
+
+# Lister les entités disponibles à migrer
+GET /api/migrations/entities/available
+
+# Récupérer les détails d'une entité source
+GET /api/migrations/entities/{source_id}
+```
+
+---
+
+#### Phase 1️⃣ : Tables de référence `x_*`
+
+**Ordre d'exécution :**
+1. `x_devises`
+2. `x_entites_types`
+3. `x_types_passages`
+4. `x_types_reglements`
+5. `x_users_roles`
+6. `x_pays`
+7. `x_regions`
+8. `x_departements`
+9. `x_villes`
+
+**Endpoint de test :**
+```http
+POST /api/migrations/entity/step
+{
+ "entity_id": 45,
+ "step": "x_devises",
+ "dry_run": true
+}
+```
+
+**Vérifications à effectuer :**
+- [ ] Nombre d'enregistrements source == cible
+- [ ] Champs `rowid` → `id` correctement mappés
+- [ ] Champs `active` → `chk_active` correctement mappés
+- [ ] Aucune erreur de contrainte FK
+
+**Requêtes SQL de vérification :**
+```sql
+-- Comparer les counts
+SELECT 'source' as db, COUNT(*) as count FROM geosector.x_devises
+UNION ALL
+SELECT 'cible' as db, COUNT(*) as count FROM geo_app.x_devises;
+
+-- Vérifier les IDs manquants
+SELECT s.rowid
+FROM geosector.x_devises s
+LEFT JOIN geo_app.x_devises t ON s.rowid = t.id
+WHERE t.id IS NULL;
+```
+
+---
+
+#### Phase 2️⃣ : Entité (Amicale)
+
+**Tables concernées :**
+- `users_entites` → `entites`
+
+**Endpoint :**
+```http
+POST /api/migrations/entity/step
+{
+ "entity_id": 45,
+ "step": "entites"
+}
+```
+
+**Mappings critiques à vérifier :**
+- [ ] `libelle` → `encrypted_name` (chiffrement AES-256)
+- [ ] `email` → `encrypted_email` (chiffrement recherchable)
+- [ ] `tel1`, `tel2` → `encrypted_phone`, `encrypted_mobile` (détection 06/07)
+- [ ] `iban` → `encrypted_iban`
+- [ ] `bic` → `encrypted_bic`
+- [ ] `cp` → `code_postal`
+
+**Tests fonctionnels :**
+```http
+# Récupérer l'entité migrée
+GET /api/entites/45
+
+# Vérifier le déchiffrement
+# Le nom doit être lisible dans la réponse
+```
+
+**Vérifications SQL :**
+```sql
+-- Vérifier la présence de l'entité
+SELECT id, encrypted_name, encrypted_email
+FROM geo_app.entites
+WHERE id = 45;
+
+-- Les champs chiffrés doivent contenir des données base64
+-- encrypted_name devrait ressembler à : "eyJpdiI6Ij..."
+```
+
+---
+
+#### Phase 3️⃣ : Utilisateurs de l'entité
+
+**Tables concernées :**
+- `users` (filtrés par `fk_entite = 45`)
+
+**Endpoint :**
+```http
+POST /api/migrations/entity/step
+{
+ "entity_id": 45,
+ "step": "users"
+}
+```
+
+**Mappings critiques :**
+- [ ] `libelle` → `encrypted_name`
+- [ ] `username` → `encrypted_user_name` (chiffrement recherchable)
+- [ ] `userpswd` ou `userpass` → `user_pass_hash`
+- [ ] `prenom` → `first_name`
+- [ ] `nom_tournee` → `sect_name`
+- [ ] `telephone` → `encrypted_phone`
+- [ ] `mobile` → `encrypted_mobile`
+- [ ] `email` → `encrypted_email`
+
+**Tests fonctionnels :**
+```http
+# Lister les utilisateurs de l'entité
+GET /api/users?fk_entite=45
+
+# Tester un login avec un utilisateur migré
+POST /api/login
+{
+ "username": "j.dupont",
+ "password": "MotDePasseOriginal123"
+}
+```
+
+**Vérifications SQL :**
+```sql
+-- Comparer les counts
+SELECT COUNT(*) FROM geosector.users WHERE fk_entite = 45;
+SELECT COUNT(*) FROM geo_app.users WHERE fk_entite = 45;
+
+-- Vérifier les hash de mots de passe (doivent être identiques)
+SELECT u1.rowid, u1.userpswd, u2.user_pass_hash
+FROM geosector.users u1
+JOIN geo_app.users u2 ON u1.rowid = u2.id
+WHERE u1.fk_entite = 45
+LIMIT 5;
+```
+
+**⚠️ Point critique :**
+- Les mots de passe doivent être migrés **tels quels** (hash déjà fait)
+- Ne PAS re-hasher les mots de passe
+- Vérifier que le login fonctionne avec les anciens identifiants
+
+---
+
+#### Phase 4️⃣ : Opérations de l'entité
+
+**Tables concernées :**
+- `operations` (filtrées par `fk_entite = 45`)
+
+**Endpoint :**
+```http
+POST /api/migrations/entity/step
+{
+ "entity_id": 45,
+ "step": "operations",
+ "options": {
+ "limit": 0 // 0 = toutes les opérations (pas de limite à 3)
+ }
+}
+```
+
+**Mappings :**
+- [ ] `rowid` → `id`
+- [ ] `date_creat` → `created_at`
+- [ ] `date_modif` → `updated_at`
+- [ ] `active` → `chk_active`
+
+**Tests fonctionnels :**
+```http
+# Lister les opérations de l'entité
+GET /api/operations?fk_entite=45
+
+# Récupérer une opération spécifique
+GET /api/operations/{operation_id}
+```
+
+**Vérifications SQL :**
+```sql
+-- Comparer les counts
+SELECT COUNT(*) FROM geosector.operations WHERE fk_entite = 45;
+SELECT COUNT(*) FROM geo_app.operations WHERE fk_entite = 45;
+
+-- Vérifier les 3 dernières opérations
+SELECT id, libelle, date_deb, date_fin
+FROM geo_app.operations
+WHERE fk_entite = 45
+ORDER BY id DESC
+LIMIT 3;
+```
+
+---
+
+#### Phase 5️⃣ : Secteurs des opérations
+
+**Tables concernées :**
+- `sectors` + `ope_users_sectors` → `ope_sectors`
+
+**Endpoint :**
+```http
+POST /api/migrations/entity/step
+{
+ "entity_id": 45,
+ "step": "ope_sectors"
+}
+```
+
+**Logique spécifique :**
+- Fusion de `sectors.rowid` → `ope_sectors.fk_old_sector`
+- Génération de nouveaux `ope_sectors.id` (auto-increment)
+- Création d'un mapping `old_id → new_id` pour les étapes suivantes
+
+**Vérifications SQL :**
+```sql
+-- Vérifier le mapping des secteurs
+SELECT os.id, os.fk_old_sector, os.libelle, os.fk_operation
+FROM geo_app.ope_sectors os
+JOIN geo_app.operations o ON os.fk_operation = o.id
+WHERE o.fk_entite = 45
+ORDER BY os.fk_operation, os.id;
+```
+
+---
+
+#### Phase 6️⃣ : Adresses des secteurs
+
+**Tables concernées :**
+- `sectors_adresses`
+
+**Endpoint :**
+```http
+POST /api/migrations/entity/step
+{
+ "entity_id": 45,
+ "step": "sectors_adresses"
+}
+```
+
+**Mappings :**
+- [ ] Utilisation du mapping `fk_old_sector → new_id` créé en Phase 5
+- [ ] Ajout de colonnes OSM avec valeurs par défaut
+
+**Vérifications SQL :**
+```sql
+-- Comparer les counts
+SELECT COUNT(*) FROM geosector.sectors_adresses sa
+JOIN geosector.sectors s ON sa.fk_sector = s.rowid
+JOIN geosector.ope_users_sectors ous ON ous.fk_sector = s.rowid
+JOIN geosector.operations o ON ous.fk_operation = o.rowid
+WHERE o.fk_entite = 45;
+
+-- Vérifier dans la cible
+SELECT COUNT(*) FROM geo_app.sectors_adresses sa
+JOIN geo_app.ope_sectors os ON sa.fk_sector = os.id
+JOIN geo_app.operations o ON os.fk_operation = o.id
+WHERE o.fk_entite = 45;
+```
+
+---
+
+#### Phase 7️⃣ : Associations opérations-utilisateurs
+
+**Tables concernées :**
+- `ope_users`
+- `ope_users_sectors`
+
+**Endpoint :**
+```http
+POST /api/migrations/entity/step
+{
+ "entity_id": 45,
+ "step": "ope_users"
+}
+```
+
+**⚠️ Point d'attention :**
+- Nouveaux champs dans cible : `fk_role`, `first_name`, `encrypted_name`, `sect_name`
+- **Solution recommandée** : Enrichir avec les données depuis `users` si disponibles
+
+**Vérifications SQL :**
+```sql
+-- Vérifier les associations
+SELECT ou.id, ou.fk_operation, ou.fk_user, u.encrypted_user_name
+FROM geo_app.ope_users ou
+JOIN geo_app.users u ON ou.fk_user = u.id
+JOIN geo_app.operations o ON ou.fk_operation = o.id
+WHERE o.fk_entite = 45
+LIMIT 10;
+```
+
+---
+
+#### Phase 8️⃣ : Passages (Données critiques)
+
+**Tables concernées :**
+- `ope_pass`
+
+**Endpoint :**
+```http
+POST /api/migrations/entity/step
+{
+ "entity_id": 45,
+ "step": "ope_pass",
+ "options": {
+ "batch_size": 5000 // Migration par lots de 5000
+ }
+}
+```
+
+**Mappings critiques :**
+- [ ] `date_eve` → `passed_at`
+- [ ] `libelle` → `encrypted_name` (chiffrement)
+- [ ] `email` → `encrypted_email` (chiffrement recherchable + validation)
+- [ ] `phone` → `encrypted_phone` (chiffrement)
+- [ ] `recu` → `nom_recu`
+- [ ] `fk_type` : transformation 8→5, 9→6
+- [ ] `fk_sector` : ancien ID → nouvel ID via mapping Phase 5
+
+**Tests fonctionnels :**
+```http
+# Compter les passages de l'entité
+GET /api/passages/count?entity_id=45
+
+# Récupérer les derniers passages
+GET /api/passages?entity_id=45&limit=10&order=desc
+```
+
+**Vérifications SQL :**
+```sql
+-- Comparer les counts par opération
+SELECT o.id, o.libelle, COUNT(p.id) as nb_passages
+FROM geo_app.operations o
+LEFT JOIN geo_app.ope_pass p ON p.fk_operation = o.id
+WHERE o.fk_entite = 45
+GROUP BY o.id, o.libelle
+ORDER BY o.id;
+
+-- Vérifier les montants totaux
+SELECT SUM(montant) FROM geosector.ope_pass
+WHERE fk_operation IN (SELECT rowid FROM geosector.operations WHERE fk_entite = 45);
+
+SELECT SUM(montant) FROM geo_app.ope_pass
+WHERE fk_operation IN (SELECT id FROM geo_app.operations WHERE fk_entite = 45);
+```
+
+**⚠️ TRÈS IMPORTANT :**
+- Cette phase peut prendre du temps (nombreux passages)
+- Utiliser la migration par lots (5000 par 5000)
+- Afficher une progression dans l'interface
+- Vérifier que les emails sont valides avant chiffrement
+
+---
+
+#### Phase 9️⃣ : Historique et médias
+
+**Tables concernées :**
+- `ope_pass_histo`
+- `medias`
+
+**Endpoint :**
+```http
+POST /api/migrations/entity/step
+{
+ "entity_id": 45,
+ "step": "ope_pass_histo"
+}
+```
+
+**Mappings :**
+- [ ] `ope_pass_histo` : Suppression du champ `fk_user` (n'existe plus)
+- [ ] `medias` : Mapping `support_rowid` → `support_id`
+
+**Vérifications SQL :**
+```sql
+-- Historique des passages
+SELECT COUNT(*) FROM geo_app.ope_pass_histo h
+JOIN geo_app.ope_pass p ON h.fk_pass = p.id
+JOIN geo_app.operations o ON p.fk_operation = o.id
+WHERE o.fk_entite = 45;
+
+-- Médias de l'entité
+SELECT COUNT(*) FROM geo_app.medias
+WHERE support = 'entite' AND support_id = 45;
+```
+
+---
+
+### Endpoints de gestion et monitoring
+
+#### Endpoints de vérification
+```http
+# Statut de la migration d'une entité
+GET /api/migrations/entity/{entity_id}/status
+
+# Logs de migration
+GET /api/migrations/entity/{entity_id}/logs
+
+# Rapport de migration
+GET /api/migrations/entity/{entity_id}/report
+```
+
+#### Endpoints de rollback
+```http
+# Annuler la migration d'une entité
+DELETE /api/migrations/entity/{entity_id}
+
+# Supprimer uniquement une étape
+DELETE /api/migrations/entity/{entity_id}/step/{step_name}
+```
+
+#### Endpoints de comparaison
+```http
+# Comparer les données source vs cible
+GET /api/migrations/entity/{entity_id}/compare
+
+# Vérifier l'intégrité des données
+GET /api/migrations/entity/{entity_id}/verify
+```
+
+---
+
+### Checklist finale de validation
+
+#### ✅ Données de base
+- [ ] L'entité existe dans `geo_app.entites` avec toutes les données chiffrées
+- [ ] Le déchiffrement fonctionne (appel `GET /api/entites/{id}`)
+- [ ] Le logo de l'entité est présent (si applicable)
+
+#### ✅ Utilisateurs
+- [ ] Tous les utilisateurs de l'entité sont présents
+- [ ] Le login fonctionne avec les anciens identifiants
+- [ ] Les données chiffrées sont déchiffrables
+- [ ] Les rôles sont corrects (`fk_role`)
+
+#### ✅ Opérations
+- [ ] Toutes les opérations sont migrées (pas de limite à 3)
+- [ ] Les dates `date_deb` et `date_fin` sont correctes
+- [ ] Les secteurs associés sont présents
+
+#### ✅ Passages
+- [ ] Le nombre total de passages correspond
+- [ ] Les montants totaux correspondent
+- [ ] Les emails chiffrés sont valides et déchiffrables
+- [ ] Les `fk_type` sont corrects (vérifier transformations 8→5, 9→6)
+
+#### ✅ Intégrité référentielle
+- [ ] Aucune contrainte FK violée
+- [ ] Tous les `fk_user` existent dans `users`
+- [ ] Tous les `fk_operation` existent dans `operations`
+- [ ] Tous les `fk_sector` existent dans `ope_sectors`
+
+#### ✅ Tests fonctionnels
+- [ ] Login avec un utilisateur de l'entité
+- [ ] Affichage des opérations dans l'interface
+- [ ] Affichage des passages dans une opération
+- [ ] Création d'un nouveau passage (test post-migration)
+- [ ] Génération d'un reçu fiscal (si applicable)
+
+---
+
+### Scripts SQL utiles
+
+#### Comparer les totaux globaux
+```sql
+-- Script à exécuter après migration complète
+SELECT
+ 'Entité' as type,
+ (SELECT COUNT(*) FROM geosector.users_entites WHERE rowid = 45) as source,
+ (SELECT COUNT(*) FROM geo_app.entites WHERE id = 45) as cible
+UNION ALL
+SELECT
+ 'Users',
+ (SELECT COUNT(*) FROM geosector.users WHERE fk_entite = 45),
+ (SELECT COUNT(*) FROM geo_app.users WHERE fk_entite = 45)
+UNION ALL
+SELECT
+ 'Operations',
+ (SELECT COUNT(*) FROM geosector.operations WHERE fk_entite = 45),
+ (SELECT COUNT(*) FROM geo_app.operations WHERE fk_entite = 45)
+UNION ALL
+SELECT
+ 'Passages',
+ (SELECT COUNT(*) FROM geosector.ope_pass WHERE fk_operation IN
+ (SELECT rowid FROM geosector.operations WHERE fk_entite = 45)),
+ (SELECT COUNT(*) FROM geo_app.ope_pass WHERE fk_operation IN
+ (SELECT id FROM geo_app.operations WHERE fk_entite = 45));
+```
+
+#### Vérifier l'intégrité des montants
+```sql
+-- Totaux des passages par type
+SELECT p.fk_type, COUNT(*) as nb, SUM(p.montant) as total
+FROM geo_app.ope_pass p
+JOIN geo_app.operations o ON p.fk_operation = o.id
+WHERE o.fk_entite = 45
+GROUP BY p.fk_type;
+```
+
+---
+
+## 📋 TODO List
+
+### 🔴 Priorité CRITIQUE (À faire IMMÉDIATEMENT)
+
+- [ ] **Créer l'endpoint API de migration**
+ - Fichier : `src/Controllers/MigrationController.php`
+ - Service : `src/Services/MigrationService.php`
+ - Routes : Ajouter dans `index.php`
+ - Action : Implémenter la logique de migration par étape
+
+- [ ] **Configurer la connexion à la base source**
+ - Fichier : `src/Config/AppConfig.php`
+ - Action : Ajouter les paramètres de connexion à `geosector` via tunnel SSH
+
+### ⚠️ Priorité HAUTE (Cette semaine)
+
+- [ ] **Externaliser configuration sensible**
+ - Créer `scripts/.env.example`
+ - Parser `.env` dans `config.php`
+ - Ajouter `scripts/.env` au `.gitignore`
+ - Mettre à jour documentation
+
+- [ ] **Paramétrer limite opérations**
+ - Fichier : `scripts/php/migrate_operations.php`
+ - Ajouter argument CLI `--limit-operations=N`
+ - Documenter dans `--help`
+ - Valeur par défaut : 0 (toutes)
+
+- [ ] **Tests de chiffrement optionnels**
+ - Fichier : `scripts/php/migrate_users.php`
+ - Créer flag `--test-encryption`
+ - Désactiver par défaut
+
+### 📊 Priorité MOYENNE (Ce mois-ci)
+
+- [ ] **Uniformiser logging**
+ - Remplacer tous les `echo` par `logOperation()`
+ - Ajouter niveaux : DEBUG, INFO, WARNING, ERROR
+ - Format uniforme
+
+- [ ] **Script de vérification post-migration**
+ - Créer `scripts/php/verify_migration.php`
+ - Comparer counts source vs cible
+ - Vérifier intégrité référentielle
+ - Générer rapport HTML
+
+- [ ] **Remplacer dates `0000-00-00` par `NULL`**
+ - Fichiers concernés : `migrate_operations.php`, `sectors_adresses.php`
+ - Remplacer par `NULL` pour compatibilité MariaDB strict mode
+
+- [ ] **Documentation utilisateur**
+ - Guide pas-à-pas migration complète
+ - Prérequis (tunnel SSH, accès bases)
+ - Procédure de rollback
+ - FAQ troubleshooting
+
+### 💡 Priorité BASSE (Nice to have)
+
+- [ ] **Option transaction globale**
+ - Flag `--transactional` dans `migrate.php`
+ - Englober toutes migrations dans 1 transaction
+ - Warning si échec = rollback complet
+
+- [ ] **Barre de progression**
+ - Utiliser library CLI (ex: `symfony/console`)
+ - Afficher progression en temps réel
+ - ETA par table
+
+- [ ] **Mode dry-run**
+ - Flag `--dry-run` pour simuler sans écrire
+ - Afficher ce qui serait fait
+ - Utile pour tests
+
+- [ ] **Export/Import rapide**
+ - Alternative : dump SQL + sed pour mapping IDs
+ - Plus rapide pour gros volumes
+ - Comparer perf vs PHP
+
+- [ ] **Tests automatisés**
+ - PHPUnit tests pour chaque script
+ - Mock des connexions DB
+ - CI/CD avec GitHub Actions
+
+---
+
+## 📚 Références
+
+### Fichiers clés
+- **Schéma DB** : `docs/geo_app.sql`
+- **Configuration** : `scripts/config.php`
+- **Orchestrateur** : `scripts/php/migrate.php`
+- **Logs** : `scripts/logs/migration_YYYY-MM-DD.log`
+
+### Ordre d'exécution (défini dans `migrate.php:64-85`)
+```
+x_devises → x_entites_types → x_types_passages → x_types_reglements →
+x_users_roles → x_pays → x_regions → x_departements → x_villes →
+entites → users → operations → ope_sectors → sectors_adresses →
+ope_users → ope_users_sectors → ope_pass → ope_pass_histo → medias
+```
+
+### Commandes utiles
+```bash
+# Migration complète
+php scripts/php/migrate.php
+
+# Migration d'une table spécifique
+php scripts/php/migrate.php users
+
+# Migration avec truncate
+php scripts/php/migrate.php users --truncate
+
+# Afficher l'aide
+php scripts/php/migrate.php --help
+
+# Consulter les logs
+tail -f scripts/logs/migration_$(date +%Y-%m-%d).log
+```
+
+---
+
+## 🔧 Corrections Critiques Appliquées (10/10/2025)
+
+### Correction #15 : Ajout de contraintes UNIQUE pour éviter les doublons
+
+**Problème identifié** : Les tables `ope_users` et `ope_users_sectors` n'avaient PAS de contrainte UNIQUE sur leurs combinaisons de FK, permettant des doublons massifs lors de la migration.
+
+**Diagnostic** :
+- Table `ope_users` : Possibilité de centaines de doublons pour la même paire (fk_operation, fk_user)
+- Table `ope_users_sectors` : Risque de doublons sur (fk_operation, fk_user, fk_sector)
+- Le `ON DUPLICATE KEY UPDATE` dans le script PHP ne fonctionnait pas car aucune contrainte UNIQUE n'existait
+
+**Solution appliquée** :
+
+1. **Mise à jour de `geo_app_structure.sql`** (structure de référence) :
+ - Ajout de `UNIQUE KEY idx_operation_user (fk_operation, fk_user)` sur `ope_users` (ligne 403)
+ - Ajout de `UNIQUE KEY idx_operation_user_sector (fk_operation, fk_user, fk_sector)` sur `ope_users_sectors` (ligne 430)
+
+2. **Correction du code PHP** dans `migrate_from_backup.php` :
+ ```php
+ // AVANT (INCORRECT - créait des doublons)
+ $sql = "SELECT ou.rowid, ou.fk_operation, ou.fk_user, ...
+ INSERT INTO ope_users (id, fk_operation, fk_user, ...) VALUES (:id, ...
+
+ // APRÈS (CORRECT - évite les doublons)
+ $sql = "SELECT DISTINCT ou.fk_operation, ou.fk_user, ... // Pas de rowid
+ INSERT INTO ope_users (fk_operation, fk_user, ...) VALUES (... // Pas d'id, auto-increment
+ ```
+
+3. **Application sur les bases de données** :
+ ```sql
+ -- Vérifier les doublons existants
+ SELECT fk_operation, fk_user, COUNT(*) as count
+ FROM ope_users GROUP BY fk_operation, fk_user HAVING count > 1;
+
+ -- Ajouter les contraintes UNIQUE
+ ALTER TABLE ope_users
+ ADD UNIQUE KEY `idx_operation_user` (`fk_operation`, `fk_user`);
+
+ ALTER TABLE ope_users_sectors
+ ADD UNIQUE KEY `idx_operation_user_sector` (`fk_operation`, `fk_user`, `fk_sector`);
+
+ -- Vérifier que les contraintes sont bien créées
+ SHOW INDEX FROM ope_users WHERE Key_name = 'idx_operation_user';
+ SHOW INDEX FROM ope_users_sectors WHERE Key_name = 'idx_operation_user_sector';
+ ```
+
+**Impact** :
+- ✅ Empêche définitivement les doublons dans `ope_users` et `ope_users_sectors`
+- ✅ Le `ON DUPLICATE KEY UPDATE` fonctionne désormais correctement
+- ✅ Migration idempotente (peut être relancée sans créer de duplicatas)
+
+**Fichiers modifiés** :
+- `scripts/php/geo_app_structure.sql` - Structure de référence mise à jour (lignes 403, 430)
+- `scripts/php/migrate_from_backup.php` - Code corrigé avec SELECT DISTINCT (lignes 1125-1191)
+
+**À appliquer sur les environnements** :
+```bash
+# DEV (dva_geo sur maria3/IN3)
+mysql -h 13.23.33.4 -u dva_geo_user -p'CBq9tKHj6PGPZuTmAHV7' dva_geo
+# Puis exécuter les ALTER TABLE ci-dessus
+
+# REC (rca_geo sur maria3/IN3)
+mysql -h 13.23.33.4 -u rca_geo_user -p'UPf3C0cQ805LypyM71iW' rca_geo
+# Puis exécuter les ALTER TABLE ci-dessus
+
+# PROD (pra_geo sur maria4/IN4)
+mysql -h 13.23.33.4 -u pra_geo_user -p'd2jAAGGWi8fxFrWgXjOA' pra_geo
+# Puis exécuter les ALTER TABLE ci-dessus
+```
+
+**Pour re-migrer après correction** :
+```bash
+# Supprimer les données de l'entité avant re-migration
+php scripts/php/migrate_from_backup.php \
+ --source-db=geosector_20251008 \
+ --mode=entity \
+ --entity-id=5 \
+ --delete-before
+```
+
+---
+
+**Dernière mise à jour** : 2025-10-10
+**Auteur de l'analyse** : Claude Code
+**Version** : 1.1
diff --git a/api/scripts/cron/CRON.md b/api/scripts/cron/CRON.md
new file mode 100644
index 00000000..e85c5441
--- /dev/null
+++ b/api/scripts/cron/CRON.md
@@ -0,0 +1,273 @@
+# Documentation des tâches CRON - API Geosector
+
+Ce dossier contient les scripts automatisés de maintenance et de traitement pour l'API Geosector.
+
+## Scripts disponibles
+
+### 1. `process_email_queue.php`
+
+**Fonction** : Traite la queue d'emails en attente (reçus fiscaux, notifications)
+
+**Caractéristiques** :
+
+- Traite 50 emails maximum par exécution
+- 3 tentatives maximum par email
+- Lock file pour éviter l'exécution simultanée
+- Nettoyage automatique des emails envoyés de plus de 30 jours
+
+**Fréquence recommandée** : Toutes les 5 minutes
+
+**Ligne crontab** :
+
+```bash
+*/5 * * * * /usr/bin/php /var/www/geosector/api/scripts/cron/process_email_queue.php >> /var/www/geosector/api/logs/email_queue.log 2>&1
+```
+
+---
+
+### 2. `cleanup_security_data.php`
+
+**Fonction** : Purge les données de sécurité obsolètes selon la politique de rétention
+
+**Données nettoyées** :
+
+- Métriques de performance : 30 jours
+- Tentatives de login échouées : 7 jours
+- Alertes résolues : 90 jours
+- IPs expirées : Déblocage immédiat
+
+**Fréquence recommandée** : Quotidien à 2h du matin
+
+**Ligne crontab** :
+
+```bash
+0 2 * * * /usr/bin/php /var/www/geosector/api/scripts/cron/cleanup_security_data.php >> /var/www/geosector/api/logs/cleanup_security.log 2>&1
+```
+
+---
+
+### 3. `cleanup_logs.php`
+
+**Fonction** : Supprime les fichiers de logs de plus de 10 jours
+
+**Caractéristiques** :
+
+- Cible tous les fichiers `*.log` dans `/api/logs/`
+- Exclut le dossier `/logs/events/` (rétention 15 mois)
+- Rétention : 10 jours
+- Logs détaillés des fichiers supprimés et taille libérée
+- Lock file pour éviter l'exécution simultanée
+
+**Fréquence recommandée** : Quotidien à 3h du matin
+
+**Ligne crontab** :
+
+```bash
+0 3 * * * /usr/bin/php /var/www/geosector/api/scripts/cron/cleanup_logs.php >> /var/www/geosector/api/logs/cleanup_logs.log 2>&1
+```
+
+---
+
+### 4. `rotate_event_logs.php`
+
+**Fonction** : Rotation des logs d'événements JSONL (système EventLogService)
+
+**Politique de rétention (15 mois)** :
+
+- 0-15 mois : fichiers `.jsonl` conservés (non compressés pour accès API)
+- > 15 mois : suppression automatique
+
+**Caractéristiques** :
+
+- Suppression des fichiers > 15 mois
+- Pas de compression (fichiers accessibles par l'API)
+- Logs détaillés des suppressions
+- Lock file pour éviter l'exécution simultanée
+
+**Fréquence recommandée** : Mensuel le 1er à 3h du matin
+
+**Ligne crontab** :
+
+```bash
+0 3 1 * * /usr/bin/php /var/www/geosector/api/scripts/cron/rotate_event_logs.php >> /var/www/geosector/api/logs/rotation_events.log 2>&1
+```
+
+---
+
+### 5. `update_stripe_devices.php`
+
+**Fonction** : Met à jour la liste des appareils Android certifiés pour Tap to Pay
+
+**Caractéristiques** :
+
+- Liste de 95+ devices intégrée
+- Ajoute les nouveaux appareils certifiés
+- Met à jour les versions Android minimales
+- Désactive les appareils obsolètes
+- Notification email si changements importants
+- Possibilité de personnaliser via `/data/stripe_certified_devices.json`
+
+**Fréquence recommandée** : Hebdomadaire le dimanche à 3h
+
+**Ligne crontab** :
+
+```bash
+0 3 * * 0 /usr/bin/php /var/www/geosector/api/scripts/cron/update_stripe_devices.php >> /var/www/geosector/api/logs/stripe_devices.log 2>&1
+```
+
+---
+
+### 6. `sync_databases.php`
+
+**Fonction** : Synchronise les bases de données entre environnements
+
+**Note** : Ce script est spécifique à un cas d'usage particulier. Vérifier son utilité avant activation.
+
+**Fréquence recommandée** : À définir selon les besoins
+
+**Ligne crontab** :
+
+```bash
+# À configurer selon les besoins
+# 0 4 * * * /usr/bin/php /var/www/geosector/api/scripts/cron/sync_databases.php >> /var/www/geosector/api/logs/sync_databases.log 2>&1
+```
+
+---
+
+## Installation sur les containers Incus
+
+### 1. Déployer les scripts sur les environnements
+
+```bash
+# DEV (dva-geo sur IN3)
+./deploy-api.sh
+
+# RECETTE (rca-geo sur IN3)
+./deploy-api.sh rca
+
+# PRODUCTION (pra-geo sur IN4)
+./deploy-api.sh pra
+```
+
+### 2. Configurer le crontab sur chaque container
+
+```bash
+# Se connecter au container
+incus exec dva-geo -- sh # ou rca-geo, pra-geo
+
+# Éditer le crontab
+crontab -e
+
+# Ajouter les lignes ci-dessous (adapter les chemins si nécessaire)
+```
+
+### 3. Configuration complète recommandée
+
+```bash
+# Traitement de la queue d'emails (toutes les 5 minutes)
+*/5 * * * * /usr/bin/php /var/www/geosector/api/scripts/cron/process_email_queue.php >> /var/www/geosector/api/logs/email_queue.log 2>&1
+
+# Nettoyage des données de sécurité (quotidien à 2h)
+0 2 * * * /usr/bin/php /var/www/geosector/api/scripts/cron/cleanup_security_data.php >> /var/www/geosector/api/logs/cleanup_security.log 2>&1
+
+# Nettoyage des anciens logs (quotidien à 3h)
+0 3 * * * /usr/bin/php /var/www/geosector/api/scripts/cron/cleanup_logs.php >> /var/www/geosector/api/logs/cleanup_logs.log 2>&1
+
+# Rotation des logs événements (mensuel le 1er à 3h)
+0 3 1 * * /usr/bin/php /var/www/geosector/api/scripts/cron/rotate_event_logs.php >> /var/www/geosector/api/logs/rotation_events.log 2>&1
+
+# Mise à jour des devices Stripe (hebdomadaire dimanche à 3h)
+0 3 * * 0 /usr/bin/php /var/www/geosector/api/scripts/cron/update_stripe_devices.php >> /var/www/geosector/api/logs/stripe_devices.log 2>&1
+```
+
+### 4. Vérifier que les CRONs sont actifs
+
+```bash
+# Lister les CRONs configurés
+crontab -l
+
+# Vérifier les logs pour s'assurer qu'ils s'exécutent
+tail -f /var/www/geosector/api/logs/email_queue.log
+tail -f /var/www/geosector/api/logs/cleanup_logs.log
+```
+
+---
+
+## Surveillance et monitoring
+
+### Emplacement des logs
+
+Tous les logs CRON sont stockés dans `/var/www/geosector/api/logs/` :
+
+- `email_queue.log` : Traitement de la queue d'emails
+- `cleanup_security.log` : Nettoyage des données de sécurité
+- `cleanup_logs.log` : Nettoyage des anciens fichiers logs
+- `rotation_events.log` : Rotation des logs événements JSONL
+- `stripe_devices.log` : Mise à jour des devices Tap to Pay
+
+### Vérification de l'exécution
+
+```bash
+# Voir les dernières exécutions du processeur d'emails
+tail -n 50 /var/www/geosector/api/logs/email_queue.log
+
+# Voir les derniers nettoyages de logs
+tail -n 50 /var/www/geosector/api/logs/cleanup_logs.log
+
+# Voir les dernières rotations des logs événements
+tail -n 50 /var/www/geosector/api/logs/rotation_events.log
+
+# Voir les dernières mises à jour Stripe
+tail -n 50 /var/www/geosector/api/logs/stripe_devices.log
+```
+
+---
+
+## Notes importantes
+
+1. **Détection d'environnement** : Tous les scripts détectent automatiquement l'environnement via `gethostname()` :
+
+ - `pra-geo` → Production (app3.geosector.fr)
+ - `rca-geo` → Recette (rapp.geosector.fr)
+ - `dva-geo` → Développement (dapp.geosector.fr)
+
+2. **Lock files** : Les scripts critiques utilisent des fichiers de lock dans `/tmp/` pour éviter l'exécution simultanée
+
+3. **Permissions** : Les scripts doivent être exécutables (`chmod +x script.php`)
+
+4. **Logs** : Tous les scripts loggent via `LogService` pour traçabilité complète
+
+---
+
+## Dépannage
+
+### Le CRON ne s'exécute pas
+
+```bash
+# Vérifier que le service cron est actif
+rc-service crond status # Alpine Linux
+
+# Relancer le service si nécessaire
+rc-service crond restart
+```
+
+### Erreur de permissions
+
+```bash
+# Vérifier les permissions du script
+ls -l /var/www/geosector/api/scripts/cron/
+
+# Rendre exécutable si nécessaire
+chmod +x /var/www/geosector/api/scripts/cron/*.php
+
+# Vérifier les permissions du dossier logs
+ls -ld /var/www/geosector/api/logs/
+```
+
+### Lock file bloqué
+
+```bash
+# Si un script semble bloqué, supprimer le lock file
+rm /tmp/process_email_queue.lock
+rm /tmp/cleanup_logs.lock
+```
diff --git a/api/scripts/cron/cleanup_logs.php b/api/scripts/cron/cleanup_logs.php
new file mode 100755
index 00000000..cb5ec1de
--- /dev/null
+++ b/api/scripts/cron/cleanup_logs.php
@@ -0,0 +1,165 @@
+#!/usr/bin/env php
+> /var/www/geosector/api/logs/cleanup_logs.log 2>&1
+ */
+
+declare(strict_types=1);
+
+// Configuration
+define('LOG_RETENTION_DAYS', 10);
+define('LOCK_FILE', '/tmp/cleanup_logs.lock');
+
+// Empêcher l'exécution multiple simultanée
+if (file_exists(LOCK_FILE)) {
+ $lockTime = filemtime(LOCK_FILE);
+ // Si le lock a plus de 30 minutes, on le supprime (processus probablement bloqué)
+ if (time() - $lockTime > 1800) {
+ unlink(LOCK_FILE);
+ } else {
+ die("Le processus est déjà en cours d'exécution\n");
+ }
+}
+
+// Créer le fichier de lock
+file_put_contents(LOCK_FILE, getmypid());
+
+// Enregistrer un handler pour supprimer le lock en cas d'arrêt
+register_shutdown_function(function() {
+ if (file_exists(LOCK_FILE)) {
+ unlink(LOCK_FILE);
+ }
+});
+
+// Simuler l'environnement web pour AppConfig en CLI
+if (php_sapi_name() === 'cli') {
+ // Détecter l'environnement basé sur le hostname
+ $hostname = gethostname();
+ if (strpos($hostname, 'pra') !== false) {
+ $_SERVER['SERVER_NAME'] = 'app3.geosector.fr';
+ } elseif (strpos($hostname, 'rca') !== false) {
+ $_SERVER['SERVER_NAME'] = 'rapp.geosector.fr';
+ } else {
+ $_SERVER['SERVER_NAME'] = 'dapp.geosector.fr'; // DVA par défaut
+ }
+
+ $_SERVER['HTTP_HOST'] = $_SERVER['HTTP_HOST'] ?? $_SERVER['SERVER_NAME'];
+ $_SERVER['REMOTE_ADDR'] = $_SERVER['REMOTE_ADDR'] ?? '127.0.0.1';
+
+ // Définir getallheaders si elle n'existe pas (CLI)
+ if (!function_exists('getallheaders')) {
+ function getallheaders() {
+ return [];
+ }
+ }
+}
+
+// Chargement de l'environnement
+require_once __DIR__ . '/../../vendor/autoload.php';
+require_once __DIR__ . '/../../src/Config/AppConfig.php';
+require_once __DIR__ . '/../../src/Services/LogService.php';
+
+try {
+ // Initialisation de la configuration
+ $appConfig = AppConfig::getInstance();
+ $environment = $appConfig->getEnvironment();
+
+ // Définir le chemin du dossier logs
+ $logDir = __DIR__ . '/../../logs';
+
+ if (!is_dir($logDir)) {
+ echo "Le dossier de logs n'existe pas : {$logDir}\n";
+ exit(0);
+ }
+
+ // Date limite (10 jours en arrière)
+ $cutoffDate = time() - (LOG_RETENTION_DAYS * 24 * 60 * 60);
+
+ // Lister tous les fichiers .log (exclure le dossier events/)
+ $logFiles = glob($logDir . '/*.log');
+
+ // Exclure explicitement les logs du sous-dossier events/
+ $logFiles = array_filter($logFiles, function($file) {
+ return strpos($file, '/events/') === false;
+ });
+
+ if (empty($logFiles)) {
+ echo "Aucun fichier .log trouvé dans {$logDir}\n";
+ exit(0);
+ }
+
+ $deletedCount = 0;
+ $deletedSize = 0;
+ $deletedFiles = [];
+
+ foreach ($logFiles as $file) {
+ $fileTime = filemtime($file);
+
+ // Vérifier si le fichier est plus vieux que la date limite
+ if ($fileTime < $cutoffDate) {
+ $fileSize = filesize($file);
+ $fileName = basename($file);
+
+ if (unlink($file)) {
+ $deletedCount++;
+ $deletedSize += $fileSize;
+ $deletedFiles[] = $fileName;
+ echo "Supprimé : {$fileName} (" . number_format($fileSize / 1024, 2) . " KB)\n";
+ } else {
+ echo "ERREUR : Impossible de supprimer {$fileName}\n";
+ }
+ }
+ }
+
+ // Logger le résumé
+ if ($deletedCount > 0) {
+ $message = sprintf(
+ "Nettoyage des logs terminé - %d fichier(s) supprimé(s) - %.2f MB libérés",
+ $deletedCount,
+ $deletedSize / (1024 * 1024)
+ );
+
+ LogService::log($message, [
+ 'level' => 'info',
+ 'script' => 'cleanup_logs.php',
+ 'environment' => $environment,
+ 'deleted_count' => $deletedCount,
+ 'deleted_size_mb' => round($deletedSize / (1024 * 1024), 2),
+ 'deleted_files' => $deletedFiles
+ ]);
+
+ echo "\n" . $message . "\n";
+ } else {
+ echo "Aucun fichier à supprimer (tous les logs ont moins de " . LOG_RETENTION_DAYS . " jours)\n";
+ }
+
+} catch (Exception $e) {
+ $errorMsg = 'Erreur lors du nettoyage des logs : ' . $e->getMessage();
+
+ LogService::log($errorMsg, [
+ 'level' => 'error',
+ 'script' => 'cleanup_logs.php',
+ 'trace' => $e->getTraceAsString()
+ ]);
+
+ echo $errorMsg . "\n";
+
+ // Supprimer le lock en cas d'erreur
+ if (file_exists(LOCK_FILE)) {
+ unlink(LOCK_FILE);
+ }
+
+ exit(1);
+}
+
+// Supprimer le lock
+if (file_exists(LOCK_FILE)) {
+ unlink(LOCK_FILE);
+}
+
+exit(0);
diff --git a/api/scripts/cron/process_email_queue.php b/api/scripts/cron/process_email_queue.php
index 219ddb8a..288cd08f 100755
--- a/api/scripts/cron/process_email_queue.php
+++ b/api/scripts/cron/process_email_queue.php
@@ -41,14 +41,14 @@ register_shutdown_function(function() {
if (php_sapi_name() === 'cli') {
// Détecter l'environnement basé sur le hostname ou un paramètre
$hostname = gethostname();
- if (strpos($hostname, 'prod') !== false) {
- $_SERVER['SERVER_NAME'] = 'app.geosector.fr';
- } elseif (strpos($hostname, 'rec') !== false || strpos($hostname, 'rapp') !== false) {
+ if (strpos($hostname, 'pra') !== false) {
+ $_SERVER['SERVER_NAME'] = 'app3.geosector.fr';
+ } elseif (strpos($hostname, 'rca') !== false) {
$_SERVER['SERVER_NAME'] = 'rapp.geosector.fr';
} else {
- $_SERVER['SERVER_NAME'] = 'app.geo.dev'; // DVA par défaut
+ $_SERVER['SERVER_NAME'] = 'dapp.geosector.fr'; // DVA par défaut
}
-
+
$_SERVER['HTTP_HOST'] = $_SERVER['HTTP_HOST'] ?? $_SERVER['SERVER_NAME'];
$_SERVER['REMOTE_ADDR'] = $_SERVER['REMOTE_ADDR'] ?? '127.0.0.1';
@@ -69,6 +69,7 @@ require_once __DIR__ . '/../../src/Services/LogService.php';
use PHPMailer\PHPMailer\PHPMailer;
use PHPMailer\PHPMailer\SMTP;
use PHPMailer\PHPMailer\Exception;
+use App\Services\LogService;
try {
// Initialisation de la configuration
diff --git a/api/scripts/cron/rotate_event_logs.php b/api/scripts/cron/rotate_event_logs.php
new file mode 100644
index 00000000..ff6f6f22
--- /dev/null
+++ b/api/scripts/cron/rotate_event_logs.php
@@ -0,0 +1,169 @@
+#!/usr/bin/env php
+ 15 mois : suppression
+ *
+ * À exécuter mensuellement via crontab (1er du mois à 3h) :
+ * 0 3 1 * * /usr/bin/php /var/www/geosector/api/scripts/cron/rotate_event_logs.php >> /var/www/geosector/api/logs/rotation_events.log 2>&1
+ */
+
+declare(strict_types=1);
+
+// Configuration
+define('RETENTION_MONTHS', 15); // Conserver 15 mois
+define('LOCK_FILE', '/tmp/rotate_event_logs.lock');
+
+// Empêcher l'exécution multiple simultanée
+if (file_exists(LOCK_FILE)) {
+ $lockTime = filemtime(LOCK_FILE);
+ // Si le lock a plus de 2 heures, on le supprime (processus probablement bloqué)
+ if (time() - $lockTime > 7200) {
+ unlink(LOCK_FILE);
+ } else {
+ die("Le processus est déjà en cours d'exécution\n");
+ }
+}
+
+// Créer le fichier de lock
+file_put_contents(LOCK_FILE, getmypid());
+
+// Enregistrer un handler pour supprimer le lock en cas d'arrêt
+register_shutdown_function(function() {
+ if (file_exists(LOCK_FILE)) {
+ unlink(LOCK_FILE);
+ }
+});
+
+// Simuler l'environnement web pour AppConfig en CLI
+if (php_sapi_name() === 'cli') {
+ // Détecter l'environnement basé sur le hostname
+ $hostname = gethostname();
+ if (strpos($hostname, 'pra') !== false) {
+ $_SERVER['SERVER_NAME'] = 'app3.geosector.fr';
+ } elseif (strpos($hostname, 'rca') !== false) {
+ $_SERVER['SERVER_NAME'] = 'rapp.geosector.fr';
+ } else {
+ $_SERVER['SERVER_NAME'] = 'dapp.geosector.fr'; // DVA par défaut
+ }
+
+ $_SERVER['HTTP_HOST'] = $_SERVER['HTTP_HOST'] ?? $_SERVER['SERVER_NAME'];
+ $_SERVER['REMOTE_ADDR'] = $_SERVER['REMOTE_ADDR'] ?? '127.0.0.1';
+
+ // Définir getallheaders si elle n'existe pas (CLI)
+ if (!function_exists('getallheaders')) {
+ function getallheaders() {
+ return [];
+ }
+ }
+}
+
+// Chargement de l'environnement
+require_once __DIR__ . '/../../vendor/autoload.php';
+require_once __DIR__ . '/../../src/Config/AppConfig.php';
+require_once __DIR__ . '/../../src/Services/LogService.php';
+
+try {
+ // Initialisation de la configuration
+ $appConfig = AppConfig::getInstance();
+ $environment = $appConfig->getEnvironment();
+
+ // Définir le chemin du dossier des logs événements
+ $eventLogDir = __DIR__ . '/../../logs/events';
+
+ if (!is_dir($eventLogDir)) {
+ echo "Le dossier de logs événements n'existe pas : {$eventLogDir}\n";
+ exit(0);
+ }
+
+ // Date limite de suppression
+ $deletionDate = strtotime('-' . RETENTION_MONTHS . ' months');
+
+ // Lister tous les fichiers .jsonl
+ $jsonlFiles = glob($eventLogDir . '/*.jsonl');
+
+ if (empty($jsonlFiles)) {
+ echo "Aucun fichier .jsonl trouvé dans {$eventLogDir}\n";
+ exit(0);
+ }
+
+ $deletedCount = 0;
+ $deletedSize = 0;
+ $deletedFiles = [];
+
+ // ========================================
+ // Suppression des fichiers > 15 mois
+ // ========================================
+ foreach ($jsonlFiles as $file) {
+ $fileTime = filemtime($file);
+
+ // Vérifier si le fichier est plus vieux que la date de rétention
+ if ($fileTime < $deletionDate) {
+ $fileSize = filesize($file);
+ $fileName = basename($file);
+
+ if (unlink($file)) {
+ $deletedCount++;
+ $deletedSize += $fileSize;
+ $deletedFiles[] = $fileName;
+ echo "Supprimé : {$fileName} (> " . RETENTION_MONTHS . " mois, " .
+ number_format($fileSize / 1024, 2) . " KB)\n";
+ } else {
+ echo "ERREUR : Impossible de supprimer {$fileName}\n";
+ }
+ }
+ }
+
+ // ========================================
+ // RÉSUMÉ ET LOGGING
+ // ========================================
+ if ($deletedCount > 0) {
+ $message = sprintf(
+ "Rotation des logs événements terminée - %d fichier(s) supprimé(s) - %.2f MB libérés",
+ $deletedCount,
+ $deletedSize / (1024 * 1024)
+ );
+
+ LogService::log($message, [
+ 'level' => 'info',
+ 'script' => 'rotate_event_logs.php',
+ 'environment' => $environment,
+ 'deleted_count' => $deletedCount,
+ 'deleted_size_mb' => round($deletedSize / (1024 * 1024), 2),
+ 'deleted_files' => $deletedFiles
+ ]);
+
+ echo "\n" . $message . "\n";
+ } else {
+ echo "Aucune rotation nécessaire - Tous les fichiers .jsonl ont moins de " . RETENTION_MONTHS . " mois\n";
+ }
+
+} catch (Exception $e) {
+ $errorMsg = 'Erreur lors de la rotation des logs événements : ' . $e->getMessage();
+
+ LogService::log($errorMsg, [
+ 'level' => 'error',
+ 'script' => 'rotate_event_logs.php',
+ 'trace' => $e->getTraceAsString()
+ ]);
+
+ echo $errorMsg . "\n";
+
+ // Supprimer le lock en cas d'erreur
+ if (file_exists(LOCK_FILE)) {
+ unlink(LOCK_FILE);
+ }
+
+ exit(1);
+}
+
+// Supprimer le lock
+if (file_exists(LOCK_FILE)) {
+ unlink(LOCK_FILE);
+}
+
+exit(0);
diff --git a/api/scripts/cron/test_email_queue.php b/api/scripts/cron/test_email_queue.php
deleted file mode 100755
index 2491e98d..00000000
--- a/api/scripts/cron/test_email_queue.php
+++ /dev/null
@@ -1,186 +0,0 @@
-#!/usr/bin/env php
-getDatabaseConfig();
-
- // Initialiser la base de données avec la configuration
- Database::init($dbConfig);
- $db = Database::getInstance();
-
- echo "=== TEST DE LA QUEUE D'EMAILS ===\n\n";
-
- // Statistiques générales
- $stmt = $db->query('
- SELECT
- status,
- COUNT(*) as count,
- MIN(created_at) as oldest,
- MAX(created_at) as newest
- FROM email_queue
- GROUP BY status
- ');
-
- $stats = $stmt->fetchAll(PDO::FETCH_ASSOC);
-
- echo "STATISTIQUES:\n";
- echo "-------------\n";
- foreach ($stats as $stat) {
- echo sprintf(
- "Status: %s - Nombre: %d (Plus ancien: %s, Plus récent: %s)\n",
- $stat['status'],
- $stat['count'],
- $stat['oldest'] ?? 'N/A',
- $stat['newest'] ?? 'N/A'
- );
- }
-
- echo "\n";
-
- // Emails en attente
- $stmt = $db->prepare('
- SELECT
- eq.id,
- eq.fk_pass,
- eq.to_email,
- eq.subject,
- eq.created_at,
- eq.attempts,
- eq.status,
- p.fk_type,
- p.montant,
- p.nom_recu
- FROM email_queue eq
- LEFT JOIN ope_pass p ON eq.fk_pass = p.id
- WHERE eq.status = ?
- ORDER BY eq.created_at DESC
- LIMIT 10
- ');
-
- $stmt->execute(['pending']);
- $pendingEmails = $stmt->fetchAll(PDO::FETCH_ASSOC);
-
- if (empty($pendingEmails)) {
- echo "Aucun email en attente.\n";
- } else {
- echo "EMAILS EN ATTENTE (10 plus récents):\n";
- echo "------------------------------------\n";
- foreach ($pendingEmails as $email) {
- echo sprintf(
- "ID: %d | Passage: %d | Destinataire: %s\n",
- $email['id'],
- $email['fk_pass'],
- $email['to_email']
- );
- echo sprintf(
- " Sujet: %s\n",
- $email['subject']
- );
- echo sprintf(
- " Créé le: %s | Tentatives: %d\n",
- $email['created_at'],
- $email['attempts']
- );
- if ($email['fk_pass'] > 0) {
- echo sprintf(
- " Passage - Type: %s | Montant: %.2f€ | Reçu: %s\n",
- $email['fk_type'] == 1 ? 'DON' : 'Autre',
- $email['montant'] ?? 0,
- $email['nom_recu'] ?? 'Non généré'
- );
- }
- echo "---\n";
- }
- }
-
- // Emails échoués
- $stmt = $db->prepare('
- SELECT
- id,
- fk_pass,
- to_email,
- subject,
- created_at,
- attempts,
- error_message
- FROM email_queue
- WHERE status = ?
- ORDER BY created_at DESC
- LIMIT 5
- ');
-
- $stmt->execute(['failed']);
- $failedEmails = $stmt->fetchAll(PDO::FETCH_ASSOC);
-
- if (!empty($failedEmails)) {
- echo "\nEMAILS ÉCHOUÉS (5 plus récents):\n";
- echo "--------------------------------\n";
- foreach ($failedEmails as $email) {
- echo sprintf(
- "ID: %d | Passage: %d | Destinataire: %s\n",
- $email['id'],
- $email['fk_pass'],
- $email['to_email']
- );
- echo sprintf(
- " Sujet: %s\n",
- $email['subject']
- );
- echo sprintf(
- " Tentatives: %d | Erreur: %s\n",
- $email['attempts'],
- $email['error_message'] ?? 'Non spécifiée'
- );
- echo "---\n";
- }
- }
-
- // Vérifier la configuration SMTP
- echo "\nCONFIGURATION SMTP:\n";
- echo "-------------------\n";
-
- $smtpConfig = $appConfig->getSmtpConfig();
- $emailConfig = $appConfig->getEmailConfig();
-
- echo "Host: " . ($smtpConfig['host'] ?? 'Non configuré') . "\n";
- echo "Port: " . ($smtpConfig['port'] ?? 'Non configuré') . "\n";
- echo "Username: " . ($smtpConfig['user'] ?? 'Non configuré') . "\n";
- echo "Password: " . (isset($smtpConfig['pass']) ? '***' : 'Non configuré') . "\n";
- echo "Encryption: " . ($smtpConfig['secure'] ?? 'Non configuré') . "\n";
- echo "From Email: " . ($emailConfig['from'] ?? 'Non configuré') . "\n";
- echo "Contact Email: " . ($emailConfig['contact'] ?? 'Non configuré') . "\n";
-
- echo "\n=== FIN DU TEST ===\n";
-
-} catch (Exception $e) {
- echo "ERREUR: " . $e->getMessage() . "\n";
- exit(1);
-}
-
-exit(0);
\ No newline at end of file
diff --git a/api/scripts/cron/update_stripe_devices.php b/api/scripts/cron/update_stripe_devices.php
index eca39b13..82098d7f 100644
--- a/api/scripts/cron/update_stripe_devices.php
+++ b/api/scripts/cron/update_stripe_devices.php
@@ -42,7 +42,7 @@ register_shutdown_function(function() {
if (php_sapi_name() === 'cli') {
$hostname = gethostname();
if (strpos($hostname, 'prod') !== false || strpos($hostname, 'pra') !== false) {
- $_SERVER['SERVER_NAME'] = 'app.geosector.fr';
+ $_SERVER['SERVER_NAME'] = 'app3.geosector.fr';
} elseif (strpos($hostname, 'rec') !== false || strpos($hostname, 'rca') !== false) {
$_SERVER['SERVER_NAME'] = 'rapp.geosector.fr';
} else {
@@ -67,6 +67,8 @@ require_once dirname(dirname(__DIR__)) . '/src/Config/AppConfig.php';
require_once dirname(dirname(__DIR__)) . '/src/Core/Database.php';
require_once dirname(dirname(__DIR__)) . '/src/Services/LogService.php';
+use App\Services\LogService;
+
try {
echo "[" . date('Y-m-d H:i:s') . "] Début de la mise à jour des devices Stripe certifiés\n";
diff --git a/api/scripts/migrate_batch.sh b/api/scripts/migrate_batch.sh
new file mode 100755
index 00000000..64947fba
--- /dev/null
+++ b/api/scripts/migrate_batch.sh
@@ -0,0 +1,467 @@
+#!/bin/bash
+
+###############################################################################
+# Script de migration en batch des entités depuis geosector_20251008
+#
+# Usage: ./migrate_batch.sh [options]
+#
+# Options:
+# --start N Commencer à partir de l'entité N (défaut: 1)
+# --limit N Migrer seulement N entités (défaut: toutes)
+# --dry-run Simuler sans exécuter
+# --continue Continuer après une erreur (défaut: s'arrêter)
+# --interactive Mode interactif (défaut si aucune option)
+#
+# Exemple:
+# ./migrate_batch.sh --start 10 --limit 5
+# ./migrate_batch.sh --continue
+# ./migrate_batch.sh --interactive
+###############################################################################
+
+# Configuration
+SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
+JSON_FILE="${SCRIPT_DIR}/migrations_entites.json"
+LOG_DIR="/var/www/geosector/api/logs/migrations"
+MIGRATION_SCRIPT="${SCRIPT_DIR}/php/migrate_from_backup.php"
+SOURCE_DB="geosector_20251013_13"
+TARGET_DB="pra_geo"
+
+# Paramètres par défaut
+START_INDEX=1
+LIMIT=0
+DRY_RUN=0
+CONTINUE_ON_ERROR=0
+INTERACTIVE_MODE=0
+SPECIFIC_ENTITY_ID=""
+SPECIFIC_CP=""
+
+# Couleurs
+RED='\033[0;31m'
+GREEN='\033[0;32m'
+YELLOW='\033[1;33m'
+BLUE='\033[0;34m'
+CYAN='\033[0;36m'
+NC='\033[0m' # No Color
+
+# Sauvegarder le nombre d'arguments avant le parsing
+INITIAL_ARGS=$#
+
+# Parse des arguments
+while [[ $# -gt 0 ]]; do
+ case $1 in
+ --start)
+ START_INDEX="$2"
+ shift 2
+ ;;
+ --limit)
+ LIMIT="$2"
+ shift 2
+ ;;
+ --dry-run)
+ DRY_RUN=1
+ shift
+ ;;
+ --continue)
+ CONTINUE_ON_ERROR=1
+ shift
+ ;;
+ --interactive|-i)
+ INTERACTIVE_MODE=1
+ shift
+ ;;
+ --help)
+ grep "^#" "$0" | grep -v "^#!/bin/bash" | sed 's/^# //'
+ exit 0
+ ;;
+ *)
+ echo "Option inconnue: $1"
+ echo "Utilisez --help pour l'aide"
+ exit 1
+ ;;
+ esac
+done
+
+# Activer le mode interactif si aucun argument n'a été fourni
+if [ $INITIAL_ARGS -eq 0 ]; then
+ INTERACTIVE_MODE=1
+fi
+
+# Vérifications préalables
+if [ ! -f "$JSON_FILE" ]; then
+ echo -e "${RED}❌ Fichier JSON introuvable: $JSON_FILE${NC}"
+ exit 1
+fi
+
+if [ ! -f "$MIGRATION_SCRIPT" ]; then
+ echo -e "${RED}❌ Script de migration introuvable: $MIGRATION_SCRIPT${NC}"
+ exit 1
+fi
+
+# Créer le répertoire de logs
+mkdir -p "$LOG_DIR"
+
+# Fichiers de log
+BATCH_LOG="${LOG_DIR}/batch_$(date +%Y%m%d_%H%M%S).log"
+SUCCESS_LOG="${LOG_DIR}/success.log"
+ERROR_LOG="${LOG_DIR}/errors.log"
+
+# MODE INTERACTIF
+if [ $INTERACTIVE_MODE -eq 1 ]; then
+ echo ""
+ echo -e "${CYAN}═══════════════════════════════════════════════════════════${NC}"
+ echo -e "${CYAN} 🔧 Mode interactif - Migration d'entités GeoSector${NC}"
+ echo -e "${CYAN}═══════════════════════════════════════════════════════════${NC}"
+ echo ""
+
+ # Question 1: Migration globale ou ciblée ?
+ echo -e "${YELLOW}1️⃣ Type de migration :${NC}"
+ echo -e " ${CYAN}a)${NC} Migration globale (toutes les entités)"
+ echo -e " ${CYAN}b)${NC} Migration par lot (plage d'entités)"
+ echo -e " ${CYAN}c)${NC} Migration par code postal"
+ echo -e " ${CYAN}d)${NC} Migration d'une entité spécifique (ID)"
+ echo ""
+ echo -ne "${YELLOW}Votre choix (a/b/c/d) : ${NC}"
+ read -r MIGRATION_TYPE
+ echo ""
+
+ case $MIGRATION_TYPE in
+ a|A)
+ # Migration globale - garder les valeurs par défaut
+ START_INDEX=1
+ LIMIT=0
+ echo -e "${GREEN}✓${NC} Migration globale sélectionnée"
+ ;;
+ b|B)
+ # Migration par lot
+ echo -e "${YELLOW}2️⃣ Configuration du lot :${NC}"
+ echo -ne " Première entité (index, défaut=1) : "
+ read -r USER_START
+ if [ -n "$USER_START" ]; then
+ START_INDEX=$USER_START
+ fi
+
+ echo -ne " Limite (nombre d'entités, défaut=toutes) : "
+ read -r USER_LIMIT
+ if [ -n "$USER_LIMIT" ]; then
+ LIMIT=$USER_LIMIT
+ fi
+ echo ""
+ echo -e "${GREEN}✓${NC} Migration par lot : de l'index $START_INDEX, limite de $LIMIT entités"
+ ;;
+ c|C)
+ # Migration par code postal
+ echo -ne "${YELLOW}2️⃣ Code postal à migrer : ${NC}"
+ read -r SPECIFIC_CP
+ echo ""
+ if [ -z "$SPECIFIC_CP" ]; then
+ echo -e "${RED}❌ Code postal requis${NC}"
+ exit 1
+ fi
+ echo -e "${GREEN}✓${NC} Migration pour le code postal : $SPECIFIC_CP"
+ ;;
+ d|D)
+ # Migration d'une entité spécifique - bypass complet du JSON
+ echo -ne "${YELLOW}2️⃣ ID de l'entité à migrer : ${NC}"
+ read -r SPECIFIC_ENTITY_ID
+ echo ""
+ if [ -z "$SPECIFIC_ENTITY_ID" ]; then
+ echo -e "${RED}❌ ID d'entité requis${NC}"
+ exit 1
+ fi
+ echo -e "${GREEN}✓${NC} Migration de l'entité ID : $SPECIFIC_ENTITY_ID"
+ echo ""
+
+ # Demander si suppression des données de l'entité avant migration
+ echo -ne "${YELLOW}3️⃣ Supprimer les données existantes de cette entité dans la TARGET avant migration ? (y/N): ${NC}"
+ read -r DELETE_BEFORE
+ DELETE_FLAG=""
+ if [[ $DELETE_BEFORE =~ ^[Yy]$ ]]; then
+ echo -e "${GREEN}✓${NC} Les données seront supprimées avant migration"
+ DELETE_FLAG="--delete-before"
+ else
+ echo -e "${BLUE}ℹ${NC} Les données seront conservées (ON DUPLICATE KEY UPDATE)"
+ fi
+ echo ""
+
+ # Confirmer la migration
+ echo -ne "${YELLOW}⚠️ Confirmer la migration de l'entité #${SPECIFIC_ENTITY_ID} ? (y/N): ${NC}"
+ read -r CONFIRM
+ if [[ ! $CONFIRM =~ ^[Yy]$ ]]; then
+ echo -e "${RED}❌ Migration annulée${NC}"
+ exit 0
+ fi
+
+ # Exécuter directement la migration sans passer par le JSON
+ ENTITY_LOG="${LOG_DIR}/entity_${SPECIFIC_ENTITY_ID}_$(date +%Y%m%d_%H%M%S).log"
+
+ echo ""
+ echo -e "${BLUE}⏳ Migration de l'entité #${SPECIFIC_ENTITY_ID} en cours...${NC}"
+
+ php "$MIGRATION_SCRIPT" \
+ --source-db="$SOURCE_DB" \
+ --target-db="$TARGET_DB" \
+ --mode=entity \
+ --entity-id="$SPECIFIC_ENTITY_ID" \
+ --log="$ENTITY_LOG" \
+ $DELETE_FLAG
+
+ EXIT_CODE=$?
+
+ if [ $EXIT_CODE -eq 0 ]; then
+ echo -e "${GREEN}✅ Entité #${SPECIFIC_ENTITY_ID} migrée avec succès${NC}"
+ echo -e "${BLUE}📋 Log détaillé : $ENTITY_LOG${NC}"
+ else
+ echo -e "${RED}❌ Erreur lors de la migration de l'entité #${SPECIFIC_ENTITY_ID}${NC}"
+ echo -e "${RED}📋 Voir le log : $ENTITY_LOG${NC}"
+ exit 1
+ fi
+
+ exit 0
+ ;;
+ *)
+ echo -e "${RED}❌ Choix invalide${NC}"
+ exit 1
+ ;;
+ esac
+
+ echo ""
+fi
+
+# Fonctions utilitaires
+log() {
+ echo -e "$1" | tee -a "$BATCH_LOG"
+}
+
+log_success() {
+ echo "$1" >> "$SUCCESS_LOG"
+ log "${GREEN}✓${NC} $1"
+}
+
+log_error() {
+ echo "$1" >> "$ERROR_LOG"
+ log "${RED}✗${NC} $1"
+}
+
+# Extraire les entity_id du JSON (compatible sans jq)
+get_entity_ids() {
+ if [ -n "$SPECIFIC_ENTITY_ID" ]; then
+ # Entité spécifique par ID - chercher exactement "entity_id" : ID,
+ grep "\"entity_id\" : ${SPECIFIC_ENTITY_ID}," "$JSON_FILE" | sed 's/.*: \([0-9]*\).*/\1/'
+ elif [ -n "$SPECIFIC_CP" ]; then
+ # Entités par code postal
+ grep -B 2 "\"code_postal\" : \"$SPECIFIC_CP\"" "$JSON_FILE" | grep '"entity_id"' | sed 's/.*: \([0-9]*\).*/\1/'
+ else
+ # Toutes les entités
+ grep '"entity_id"' "$JSON_FILE" | sed 's/.*: \([0-9]*\).*/\1/'
+ fi
+}
+
+# Compter le nombre total d'entités
+TOTAL_ENTITIES=$(get_entity_ids | wc -l)
+
+# Vérifier si des entités ont été trouvées
+if [ $TOTAL_ENTITIES -eq 0 ]; then
+ if [ -n "$SPECIFIC_ENTITY_ID" ]; then
+ echo -e "${RED}❌ Entité #${SPECIFIC_ENTITY_ID} introuvable dans le fichier JSON${NC}"
+ elif [ -n "$SPECIFIC_CP" ]; then
+ echo -e "${RED}❌ Aucune entité trouvée pour le code postal ${SPECIFIC_CP}${NC}"
+ else
+ echo -e "${RED}❌ Aucune entité trouvée${NC}"
+ fi
+ exit 1
+fi
+
+# Calculer le nombre d'entités à migrer
+if [ $LIMIT -gt 0 ]; then
+ END_INDEX=$((START_INDEX + LIMIT - 1))
+ if [ $END_INDEX -gt $TOTAL_ENTITIES ]; then
+ END_INDEX=$TOTAL_ENTITIES
+ fi
+else
+ END_INDEX=$TOTAL_ENTITIES
+fi
+
+# Bannière de démarrage
+echo ""
+log "${BLUE}═══════════════════════════════════════════════════════════${NC}"
+log "${BLUE} Migration en batch des entités GeoSector${NC}"
+log "${BLUE}═══════════════════════════════════════════════════════════${NC}"
+log "📅 Date: $(date '+%Y-%m-%d %H:%M:%S')"
+log "📁 Source: $SOURCE_DB"
+log "📁 Cible: $TARGET_DB"
+
+# Afficher les informations selon le mode
+if [ -n "$SPECIFIC_ENTITY_ID" ]; then
+ log "🎯 Mode: Migration d'une entité spécifique"
+ log "📊 Entité ID: $SPECIFIC_ENTITY_ID"
+elif [ -n "$SPECIFIC_CP" ]; then
+ log "🎯 Mode: Migration par code postal"
+ log "📮 Code postal: $SPECIFIC_CP"
+ log "📊 Entités trouvées: $TOTAL_ENTITIES"
+else
+ TOTAL_AVAILABLE=$(grep '"entity_id"' "$JSON_FILE" | wc -l)
+ log "📊 Total entités disponibles: $TOTAL_AVAILABLE"
+ log "🎯 Entités à migrer: $START_INDEX à $END_INDEX"
+fi
+
+if [ $DRY_RUN -eq 1 ]; then
+ log "${YELLOW}🔍 Mode DRY-RUN (simulation)${NC}"
+fi
+log "${BLUE}═══════════════════════════════════════════════════════════${NC}"
+echo ""
+
+# Confirmation utilisateur
+if [ $DRY_RUN -eq 0 ]; then
+ echo -ne "${YELLOW}⚠️ Confirmer la migration ? (y/N): ${NC}"
+ read -r CONFIRM
+ if [[ ! $CONFIRM =~ ^[Yy]$ ]]; then
+ log "❌ Migration annulée par l'utilisateur"
+ exit 0
+ fi
+ echo ""
+fi
+
+# Compteurs
+SUCCESS_COUNT=0
+ERROR_COUNT=0
+SKIPPED_COUNT=0
+CURRENT_INDEX=0
+
+# Début de la migration
+START_TIME=$(date +%s)
+
+# Lire les entity_id et migrer
+get_entity_ids | while read -r ENTITY_ID; do
+ CURRENT_INDEX=$((CURRENT_INDEX + 1))
+
+ # Filtrer par index
+ if [ $CURRENT_INDEX -lt $START_INDEX ]; then
+ continue
+ fi
+
+ if [ $CURRENT_INDEX -gt $END_INDEX ]; then
+ break
+ fi
+
+ # Récupérer les détails de l'entité depuis le JSON (match exact avec la virgule)
+ ENTITY_INFO=$(grep -A 8 "\"entity_id\" : ${ENTITY_ID}," "$JSON_FILE")
+ ENTITY_NAME=$(echo "$ENTITY_INFO" | grep '"nom"' | sed 's/.*: "\(.*\)".*/\1/')
+ ENTITY_CP=$(echo "$ENTITY_INFO" | grep '"code_postal"' | sed 's/.*: "\(.*\)".*/\1/')
+ NB_USERS=$(echo "$ENTITY_INFO" | grep '"nb_users"' | sed 's/.*: \([0-9]*\).*/\1/')
+ NB_PASSAGES=$(echo "$ENTITY_INFO" | grep '"nb_passages"' | sed 's/.*: \([0-9]*\).*/\1/')
+
+ # Afficher la progression
+ PROGRESS=$((CURRENT_INDEX - START_INDEX + 1))
+ TOTAL=$((END_INDEX - START_INDEX + 1))
+ PERCENT=$((PROGRESS * 100 / TOTAL))
+
+ log ""
+ log "${BLUE}━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━${NC}"
+ log "${BLUE}[$PROGRESS/$TOTAL - ${PERCENT}%]${NC} Entité #${ENTITY_ID}: ${ENTITY_NAME} (${ENTITY_CP})"
+ log " 👥 Users: ${NB_USERS} | 📍 Passages: ${NB_PASSAGES}"
+
+ # Mode dry-run
+ if [ $DRY_RUN -eq 1 ]; then
+ log "${YELLOW} 🔍 [DRY-RUN] Simulation de la migration${NC}"
+ SKIPPED_COUNT=$((SKIPPED_COUNT + 1))
+ continue
+ fi
+
+ # Exécuter la migration
+ ENTITY_LOG="${LOG_DIR}/entity_${ENTITY_ID}_$(date +%Y%m%d_%H%M%S).log"
+
+ log " ⏳ Migration en cours..."
+ php "$MIGRATION_SCRIPT" \
+ --source-db="$SOURCE_DB" \
+ --target-db="$TARGET_DB" \
+ --mode=entity \
+ --entity-id="$ENTITY_ID" \
+ --log="$ENTITY_LOG" > /tmp/migration_output_$$.txt 2>&1
+
+ EXIT_CODE=$?
+
+ if [ $EXIT_CODE -eq 0 ]; then
+ # Succès
+ SUCCESS_COUNT=$((SUCCESS_COUNT + 1))
+ log_success "Entité #${ENTITY_ID} (${ENTITY_NAME}) migrée avec succès"
+
+ # Afficher un résumé du log avec détails
+ if [ -f "$ENTITY_LOG" ]; then
+ # Chercher la ligne avec les marqueurs #STATS#
+ STATS_LINE=$(grep "#STATS#" "$ENTITY_LOG" 2>/dev/null)
+
+ if [ -n "$STATS_LINE" ]; then
+ # Extraire chaque compteur
+ OPE=$(echo "$STATS_LINE" | grep -oE 'OPE:[0-9]+' | cut -d: -f2)
+ USERS=$(echo "$STATS_LINE" | grep -oE 'USER:[0-9]+' | cut -d: -f2)
+ SECTORS=$(echo "$STATS_LINE" | grep -oE 'SECTOR:[0-9]+' | cut -d: -f2)
+ PASSAGES=$(echo "$STATS_LINE" | grep -oE 'PASS:[0-9]+' | cut -d: -f2)
+
+ # Valeurs par défaut si extraction échoue
+ OPE=${OPE:-0}
+ USERS=${USERS:-0}
+ SECTORS=${SECTORS:-0}
+ PASSAGES=${PASSAGES:-0}
+
+ log " 📊 ope: ${OPE} | users: ${USERS} | sectors: ${SECTORS} | passages: ${PASSAGES}"
+ else
+ log " 📊 Statistiques non disponibles"
+ fi
+ fi
+ else
+ # Erreur
+ ERROR_COUNT=$((ERROR_COUNT + 1))
+ log_error "Entité #${ENTITY_ID} (${ENTITY_NAME}) - Erreur code $EXIT_CODE"
+
+ # Afficher les dernières lignes du log d'erreur
+ if [ -f "/tmp/migration_output_$$.txt" ]; then
+ log "${RED} 📋 Dernières erreurs:${NC}"
+ tail -5 "/tmp/migration_output_$$.txt" | sed 's/^/ /' | tee -a "$BATCH_LOG"
+ fi
+
+ # Arrêter ou continuer ?
+ if [ $CONTINUE_ON_ERROR -eq 0 ]; then
+ log ""
+ log "${RED}❌ Migration interrompue suite à une erreur${NC}"
+ log " Utilisez --continue pour continuer malgré les erreurs"
+ exit 1
+ fi
+ fi
+
+ # Nettoyage
+ rm -f "/tmp/migration_output_$$.txt"
+
+ # Pause entre les migrations (pour éviter de surcharger)
+ sleep 1
+done
+
+# Fin de la migration
+END_TIME=$(date +%s)
+DURATION=$((END_TIME - START_TIME))
+HOURS=$((DURATION / 3600))
+MINUTES=$(((DURATION % 3600) / 60))
+SECONDS=$((DURATION % 60))
+
+# Résumé final
+log ""
+log "${BLUE}═══════════════════════════════════════════════════════════${NC}"
+log "${BLUE} Résumé de la migration${NC}"
+log "${BLUE}═══════════════════════════════════════════════════════════${NC}"
+log "✅ Succès: ${GREEN}${SUCCESS_COUNT}${NC}"
+log "❌ Erreurs: ${RED}${ERROR_COUNT}${NC}"
+log "⏭️ Ignorées: ${YELLOW}${SKIPPED_COUNT}${NC}"
+log "⏱️ Durée: ${HOURS}h ${MINUTES}m ${SECONDS}s"
+log ""
+log "📋 Logs détaillés:"
+log " - Batch: $BATCH_LOG"
+log " - Succès: $SUCCESS_LOG"
+log " - Erreurs: $ERROR_LOG"
+log " - Individuels: $LOG_DIR/entity_*.log"
+log "${BLUE}═══════════════════════════════════════════════════════════${NC}"
+
+# Code de sortie
+if [ $ERROR_COUNT -gt 0 ]; then
+ exit 1
+else
+ exit 0
+fi
diff --git a/api/scripts/migration2/README.md b/api/scripts/migration2/README.md
new file mode 100644
index 00000000..bdaf852c
--- /dev/null
+++ b/api/scripts/migration2/README.md
@@ -0,0 +1,410 @@
+# Migration v2 - Architecture modulaire
+
+## Vue d'ensemble
+
+Cette nouvelle architecture simplifie la migration en utilisant :
+- **Source fixe** : `geosector` (synchronisée 2x/jour par PM7 depuis nx4)
+- **Multi-environnement** : `--env=dva` (développement), `--env=rca` (recette) ou `--env=pra` (production)
+- **Auto-détection** : L'environnement est détecté automatiquement selon le serveur
+- **Classes réutilisables** : Configuration, Logger, Connexion
+
+## Structure modulaire
+
+```
+migration2/
+├── README.md # Ce fichier
+├── logs/ # Logs de migration (auto-créé)
+│ └── .gitignore
+├── php/
+│ ├── migrate_from_backup.php # Script principal orchestrateur
+│ └── lib/
+│ ├── DatabaseConfig.php # Configuration multi-env
+│ ├── MigrationLogger.php # Gestion des logs
+│ ├── DatabaseConnection.php # Connexions PDO
+│ ├── OperationMigrator.php # Migration des opérations
+│ ├── UserMigrator.php # Migration des ope_users
+│ ├── SectorMigrator.php # Migration des secteurs
+│ └── PassageMigrator.php # Migration des passages
+```
+
+**Architecture modulaire** : Chaque type de données a son propre migrator spécialisé, orchestré par le script principal.
+
+## ⚠️ AVERTISSEMENT IMPORTANT
+
+**Par défaut, le script SUPPRIME toutes les données de l'entité dans la base cible avant la migration.**
+
+Cela inclut :
+- ✅ Toutes les opérations de l'entité
+- ✅ Tous les utilisateurs de l'entité
+- ✅ Tous les secteurs et passages
+- ✅ Tous les médias associés
+- ℹ️ L'entité elle-même est conservée (seules les données liées sont supprimées)
+
+Pour **désactiver** la suppression et conserver les données existantes :
+```bash
+php php/migrate_from_backup.php --mode=entity --entity-id=45 --delete-before=false
+```
+
+⚠️ **Attention** : Sans suppression préalable, risque de doublons si les données existent déjà.
+
+---
+
+## Utilisation
+
+### Migration d'une entité spécifique
+
+#### Sur dva-geo (IN3)
+```bash
+# Auto-détection de l'environnement (recommandé)
+php php/migrate_from_backup.php --mode=entity --entity-id=45
+
+# Ou avec environnement explicite
+php php/migrate_from_backup.php --env=dva --mode=entity --entity-id=45
+```
+
+#### Sur rca-geo (IN3)
+```bash
+# Auto-détection de l'environnement (recommandé)
+php php/migrate_from_backup.php --mode=entity --entity-id=45
+
+# Ou avec environnement explicite
+php php/migrate_from_backup.php --env=rca --mode=entity --entity-id=45
+```
+
+#### Sur pra-geo (IN4)
+```bash
+# Auto-détection de l'environnement (recommandé)
+php php/migrate_from_backup.php --mode=entity --entity-id=45
+
+# Ou avec environnement explicite
+php php/migrate_from_backup.php --env=pra --mode=entity --entity-id=45
+```
+
+### Migration globale (toutes les entités)
+
+```bash
+# Sur dva-geo, rca-geo ou pra-geo
+php php/migrate_from_backup.php --mode=global
+```
+
+### Options disponibles
+
+```bash
+--env=ENV # 'dva' (développement), 'rca' (recette) ou 'pra' (production)
+ # Par défaut : auto-détection selon le hostname
+--mode=MODE # 'global' ou 'entity' (défaut: global)
+--entity-id=ID # ID de l'entité à migrer (requis si mode=entity)
+--log=PATH # Fichier de log personnalisé
+ # Par défaut : logs/migration_YYYYMMDD_HHMMSS.log
+--delete-before # Supprimer les données existantes avant migration (défaut: true)
+--help # Afficher l'aide complète
+```
+
+### Exemples d'utilisation
+
+```bash
+# Migration STANDARD (avec suppression des données existantes - recommandé)
+php php/migrate_from_backup.php --mode=entity --entity-id=45
+
+# Migration SANS suppression (pour ajout/mise à jour uniquement - risque de doublons)
+php php/migrate_from_backup.php --mode=entity --entity-id=45 --delete-before=false
+
+# Migration avec log personnalisé
+php php/migrate_from_backup.php --mode=entity --entity-id=45 --log=/custom/path/entity_45.log
+
+# Afficher l'aide complète
+php php/migrate_from_backup.php --help
+```
+
+## Différences avec l'ancienne version
+
+| Aspect | Ancien | Nouveau |
+|--------|--------|---------|
+| **Source** | `--source-db=geosector_YYYYMMDD_HH` | Toujours `geosector` (fixe) |
+| **Cible** | `--target-db=pra_geo` | Déduite de `--env` ou auto-détectée (dva_geo, rca_geo, pra_geo) |
+| **Config** | Constantes hardcodées | Classes configurables |
+| **Environnement** | Manuel | Auto-détection par hostname (dva-geo, rca-geo, pra-geo) |
+| **Arguments** | 2 arguments DB requis | 1 seul `--env` (optionnel) |
+
+## Avantages
+
+✅ **Plus simple** : Plus besoin de spécifier les noms de bases
+✅ **Plus sûr** : Moins de risques d'erreurs de saisie
+✅ **Plus flexible** : Fonctionne sur dva-geo, rca-geo et pra-geo sans modification
+✅ **Plus maintenable** : Configuration centralisée dans DatabaseConfig
+✅ **Meilleurs logs** : Séparateurs, niveaux (info/warning/error/success)
+
+## Déploiement
+
+### Copier vers dva-geo (IN3)
+```bash
+scp -r migration2 root@195.154.80.116:/tmp/
+ssh root@195.154.80.116 "incus file push -r /tmp/migration2 dva-geo/var/www/geosector/api/scripts/"
+```
+
+### Copier vers rca-geo (IN3)
+```bash
+scp -r migration2 root@195.154.80.116:/tmp/
+ssh root@195.154.80.116 "incus file push -r /tmp/migration2 rca-geo/var/www/geosector/api/scripts/"
+```
+
+### Copier vers pra-geo (IN4)
+```bash
+scp -r migration2 root@51.159.7.190:/tmp/
+ssh root@51.159.7.190 "incus file push -r /tmp/migration2 pra-geo/var/www/geosector/api/scripts/"
+```
+
+### Test après déploiement
+
+```bash
+# Se connecter au container
+incus exec dva-geo -- bash # ou rca-geo, ou pra-geo
+
+# Tester avec une entité
+cd /var/www/geosector/api/scripts/migration2
+php php/migrate_from_backup.php --mode=entity --entity-id=45
+```
+
+## Logs
+
+Les logs sont enregistrés par défaut dans :
+```
+scripts/migration2/logs/migration_[MODE]_YYYYMMDD_HHMMSS.log
+```
+
+**Nommage automatique selon le mode :**
+- Migration globale : `migration_global_20251021_143045.log`
+- Migration d'une entité : `migration_entite_45_20251021_143045.log`
+
+Format des logs :
+- `[INFO]` : Informations générales
+- `[SUCCESS]` : Opérations réussies
+- `[WARNING]` : Avertissements
+- `[ERROR]` : Erreurs
+
+Le dossier `logs/` est créé automatiquement si nécessaire.
+
+**Note :** Vous pouvez toujours spécifier un fichier de log personnalisé avec l'option `--log=PATH`.
+
+## Récapitulatif de migration
+
+À la fin de chaque migration, un **récapitulatif détaillé** est automatiquement affiché et enregistré dans le fichier de log.
+
+### Format du récapitulatif
+
+```
+========================================
+📊 RÉCAPITULATIF DE LA MIGRATION
+========================================
+Entité: Nom de l'entité (ID: XX)
+Date: YYYY-MM-DD HH:MM:SS
+
+Opérations migrées: 3
+
+Opération #1: "Adhésions 2024" (ID: 850)
+ ├─ Utilisateurs: 12
+ ├─ Secteurs: 5
+ ├─ Passages totaux: 245
+ └─ Détail par secteur:
+ ├─ Centre-ville (ID: 5400)
+ │ ├─ Utilisateurs affectés: 3
+ │ └─ Passages: 67
+ ├─ Quartier Est (ID: 5401)
+ │ ├─ Utilisateurs affectés: 5
+ │ └─ Passages: 98
+ └─ Nord (ID: 5402)
+ ├─ Utilisateurs affectés: 4
+ └─ Passages: 80
+
+Opération #2: "Collecte Printemps" (ID: 851)
+ ├─ Utilisateurs: 8
+ ├─ Secteurs: 3
+ ├─ Passages totaux: 156
+ └─ Détail par secteur:
+ [...]
+
+========================================
+```
+
+### Informations fournies
+
+Le récapitulatif inclut pour chaque migration :
+
+**Au niveau de l'entité :**
+- Nom et ID de l'entité
+- Date et heure de la migration
+- Nombre total d'opérations migrées
+
+**Pour chaque opération :**
+- Nom et nouvel ID
+- Nombre d'utilisateurs migrés
+- Nombre de secteurs migrés
+- Nombre total de passages migrés
+
+**Pour chaque secteur :**
+- Nom et nouvel ID
+- Nombre d'utilisateurs affectés au secteur
+- Nombre de passages effectués dans le secteur
+
+### Utilisation du récapitulatif
+
+Ce récapitulatif permet de :
+- ✅ Vérifier rapidement que toutes les données ont été migrées
+- ✅ Comparer avec les données source pour validation
+- ✅ Identifier d'éventuelles anomalies (secteurs vides, passages manquants)
+- ✅ Documenter précisément ce qui a été migré
+- ✅ Tracer les migrations pour audit
+
+Le récapitulatif est présent à la fois :
+- **À l'écran** (stdout) en temps réel
+- **Dans le fichier de log** pour conservation
+
+## Dépannage
+
+### Erreur "env doit être 'dva', 'rca' ou 'pra'"
+L'auto-détection a échoué. Spécifiez manuellement :
+```bash
+php php/migrate_from_backup.php --env=dva --mode=entity --entity-id=45
+```
+
+### Erreur de connexion
+Vérifiez que vous êtes bien dans le bon container (dva-geo, rca-geo ou pra-geo).
+
+### Données dupliquées après migration
+Si vous avez des doublons, c'est que vous avez utilisé `--delete-before=false` sur des données existantes.
+
+**Solution** : Refaire la migration avec suppression (défaut) :
+```bash
+php php/migrate_from_backup.php --mode=entity --entity-id=45
+```
+
+### Vérifier ce qui sera supprimé avant migration
+Consultez la section "Ordre de suppression" ci-dessous pour voir exactement quelles tables seront affectées.
+
+### Logs non créés
+Vérifiez les permissions du dossier `logs/` :
+```bash
+ls -la scripts/migration2/logs/
+```
+
+## Détails techniques
+
+### Architecture hiérarchique de migration
+
+La migration fonctionne par **opération** avec une hiérarchie complète :
+
+```
+Pour chaque opération de l'entité:
+ migrateOperation($oldOperationId)
+ ├── Créer operation
+ ├── Migrer ope_users (DISTINCT depuis ope_users_sectors)
+ │ └── Mapper oldUserId → newOpeUserId
+ ├── Pour chaque secteur DISTINCT de l'opération:
+ │ └── migrateSector($oldOperationId, $newOperationId, $oldSectorId)
+ │ ├── Créer ope_sectors
+ │ ├── Mapper "opId_sectId" → newOpeSectorId
+ │ ├── Migrer sectors_adresses (fk_sector = newOpeSectorId)
+ │ ├── Migrer ope_users_sectors (avec mappings users + sector)
+ │ ├── Migrer ope_pass (avec mappings users + sector)
+ │ │ └── Pour chaque passage:
+ │ │ └── migratePassageHisto($oldPassId, $newPassId)
+ │ └── Migrer médias des passages
+ └── Migrer médias de l'opération
+```
+
+### Changement d'organisation des données : Exemple concret
+
+#### Contexte : Opération de collecte des adhésions 2024
+
+**Ancienne organisation** (base geosector - partagée) :
+- 1 opération "Adhésions 2024" avec ID 450
+- 3 utilisateurs affectés : Jean (ID 100), Marie (ID 101), Paul (ID 102)
+- 2 secteurs utilisés : Centre-ville (ID 1004) et Quartier Est (ID 1005)
+- Jean travaille sur Centre-ville, Marie et Paul sur Quartier Est
+
+Dans l'ancienne base :
+- Les 3 users existent UNE SEULE FOIS dans la table centrale `users`
+- Les 2 secteurs existent UNE SEULE FOIS dans la table centrale `sectors`
+- Les liens entre users et secteurs sont dans `ope_users_sectors`
+- Les passages font référence directement aux users (ID 100, 101, 102)
+
+**Nouvelle organisation** (base rca_geo/pra_geo - isolée par opération) :
+
+Après migration, **CHAQUE opération devient autonome** :
+- L'opération "Adhésions 2024" reçoit un nouvel ID (exemple : 850)
+- Les 3 utilisateurs sont **dupliqués** dans `ope_users` avec de nouveaux IDs :
+ - Jean → ope_users.id = 2500 (avec fk_user = 100 et fk_operation = 850)
+ - Marie → ope_users.id = 2501 (avec fk_user = 101 et fk_operation = 850)
+ - Paul → ope_users.id = 2502 (avec fk_user = 102 et fk_operation = 850)
+- Les 2 secteurs sont **dupliqués** dans `ope_sectors` :
+ - Centre-ville → ope_sectors.id = 5400 (avec fk_operation = 850)
+ - Quartier Est → ope_sectors.id = 5401 (avec fk_operation = 850)
+- Tous les passages sont mis à jour pour référencer les NOUVEAUX IDs (2500, 2501, 2502)
+
+**Pourquoi cette duplication ?**
+
+✅ **Isolation complète** : Si l'opération est supprimée, tout part avec (secteurs, users, passages)
+✅ **Performance** : Pas de jointures complexes entre opérations
+✅ **Historique** : Les données de l'opération restent figées dans le temps
+✅ **Simplicité** : Chaque opération est indépendante
+
+**Impact pour un utilisateur qui travaille sur 3 opérations différentes** :
+- Il existera 1 seule fois dans la table centrale `users` (ID 100)
+- Il existera 3 fois dans `ope_users` (1 enregistrement par opération)
+- Chaque enregistrement `ope_users` garde la référence vers `users.id = 100`
+
+Cette architecture permet de **fermer** une opération complètement sans impacter les autres.
+
+### Sélection des opérations à migrer
+
+Pour chaque entité, **maximum 3 opérations** sont migrées :
+1. **1 opération active** (`active = 1`)
+2. **2 dernières opérations inactives** (`active = 0`) ayant au moins **10 passages effectués** (`fk_type = 1`)
+
+### Ordre de suppression (si --delete-before=true)
+
+Les données sont supprimées dans cet ordre pour respecter les contraintes de clés étrangères :
+
+1. `medias` - Médias associés à l'entité ou aux opérations
+2. `ope_pass_histo` - Historique des passages
+3. `ope_pass` - Passages
+4. `ope_users_sectors` - Associations utilisateurs/secteurs
+5. `ope_users` - Utilisateurs d'opérations
+6. `sectors_adresses` - Adresses de secteurs
+7. `ope_sectors` - Secteurs d'opérations
+8. `operations` - Opérations
+9. `users` - Utilisateurs de l'entité
+
+⚠️ **L'entité elle-même** (`entites`) **n'est jamais supprimée**.
+
+### Tables de référence non migrées
+
+Les tables suivantes ne sont **pas** migrées car déjà remplies dans la cible :
+- `x_*` - Tables de référence (secteurs, adresses, etc.)
+
+## Notes importantes
+
+1. **Configuration centralisée** : Les paramètres de connexion DB sont récupérés depuis `AppConfig.php` - pas de duplication
+2. **Chiffrement** : ApiService est toujours utilisé pour les mots de passe
+3. **Logique métier** : Inchangée (migrateEntites, migrateUsers, etc.)
+4. **Mappings** : Secteurs et adresses sont toujours mappés automatiquement
+5. **Backup** : Un backup de l'ancien script est disponible dans `migrate_from_backup.php.backup`
+6. **Suppression par défaut** : Activée pour éviter les doublons et garantir une migration propre
+
+## Statut
+
+**Architecture modulaire v2** :
+- ✅ DatabaseConfig.php - Configuration multi-environnement
+- ✅ MigrationLogger.php - Gestion des logs
+- ✅ DatabaseConnection.php - Connexions PDO
+- ✅ OperationMigrator.php - Migration hiérarchique des opérations
+- ✅ UserMigrator.php - Migration des utilisateurs par opération
+- ✅ SectorMigrator.php - Migration des secteurs par opération
+- ✅ PassageMigrator.php - Migration des passages et historiques
+- ✅ migrate_from_backup.php - Script principal orchestrateur
+- ⏳ Tests sur rca-geo
+- ⏳ Tests sur pra-geo
+
+## Support
+
+En cas de problème, consulter les logs détaillés ou contacter l'équipe technique.
diff --git a/api/scripts/migration2/geo_app_structure.sql b/api/scripts/migration2/geo_app_structure.sql
new file mode 100644
index 00000000..8461e1ac
--- /dev/null
+++ b/api/scripts/migration2/geo_app_structure.sql
@@ -0,0 +1,1199 @@
+/*M!999999\- enable the sandbox mode */
+-- MariaDB dump 10.19-11.4.5-MariaDB, for Linux (x86_64)
+--
+-- Host: localhost Database: geo_app
+-- ------------------------------------------------------
+-- Server version 11.4.5-MariaDB
+
+/*!40101 SET @OLD_CHARACTER_SET_CLIENT=@@CHARACTER_SET_CLIENT */;
+/*!40101 SET @OLD_CHARACTER_SET_RESULTS=@@CHARACTER_SET_RESULTS */;
+/*!40101 SET @OLD_COLLATION_CONNECTION=@@COLLATION_CONNECTION */;
+/*!40101 SET NAMES utf8mb4 */;
+/*!40103 SET @OLD_TIME_ZONE=@@TIME_ZONE */;
+/*!40103 SET TIME_ZONE='+00:00' */;
+/*!40014 SET @OLD_UNIQUE_CHECKS=@@UNIQUE_CHECKS, UNIQUE_CHECKS=0 */;
+/*!40014 SET @OLD_FOREIGN_KEY_CHECKS=@@FOREIGN_KEY_CHECKS, FOREIGN_KEY_CHECKS=0 */;
+/*!40101 SET @OLD_SQL_MODE=@@SQL_MODE, SQL_MODE='NO_AUTO_VALUE_ON_ZERO' */;
+/*M!100616 SET @OLD_NOTE_VERBOSITY=@@NOTE_VERBOSITY, NOTE_VERBOSITY=0 */;
+
+--
+-- Table structure for table `chat_messages`
+--
+
+DROP TABLE IF EXISTS `chat_messages`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `chat_messages` (
+ `id` varchar(36) NOT NULL COMMENT 'UUID du message',
+ `room_id` varchar(36) NOT NULL COMMENT 'ID de la salle',
+ `content` text NOT NULL COMMENT 'Contenu du message',
+ `sender_id` int(11) unsigned NOT NULL COMMENT 'ID de l''expéditeur',
+ `sent_at` timestamp NOT NULL DEFAULT current_timestamp() COMMENT 'Date d''envoi',
+ `edited_at` timestamp NULL DEFAULT NULL COMMENT 'Date de modification',
+ `is_deleted` tinyint(1) NOT NULL DEFAULT 0 COMMENT 'Message supprimé',
+ PRIMARY KEY (`id`),
+ KEY `idx_room_id` (`room_id`),
+ KEY `idx_sender_id` (`sender_id`),
+ KEY `idx_sent_at` (`sent_at`),
+ KEY `idx_room_sent` (`room_id`,`sent_at`),
+ CONSTRAINT `fk_chat_messages_room` FOREIGN KEY (`room_id`) REFERENCES `chat_rooms` (`id`) ON DELETE CASCADE,
+ CONSTRAINT `fk_chat_messages_sender` FOREIGN KEY (`sender_id`) REFERENCES `users` (`id`) ON DELETE CASCADE
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci COMMENT='Messages du chat' `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `chat_participants`
+--
+
+DROP TABLE IF EXISTS `chat_participants`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `chat_participants` (
+ `room_id` varchar(36) NOT NULL COMMENT 'ID de la salle',
+ `user_id` int(11) unsigned NOT NULL COMMENT 'ID de l''utilisateur',
+ `role` int(11) DEFAULT NULL COMMENT 'Rôle de l''utilisateur (fk_role)',
+ `entite_id` int(11) unsigned DEFAULT NULL COMMENT 'ID de l''entité',
+ `joined_at` timestamp NOT NULL DEFAULT current_timestamp() COMMENT 'Date d''adhésion',
+ `left_at` timestamp NULL DEFAULT NULL COMMENT 'Date de départ',
+ `is_admin` tinyint(1) NOT NULL DEFAULT 0 COMMENT 'Admin de la salle',
+ `can_write` tinyint(1) DEFAULT 1 COMMENT 'Permission d''écrire dans la room (FALSE pour les destinataires de broadcast)',
+ `last_read_at` timestamp NULL DEFAULT NULL COMMENT 'Dernière lecture',
+ PRIMARY KEY (`room_id`,`user_id`),
+ KEY `idx_user_id` (`user_id`),
+ KEY `idx_entite_id` (`entite_id`),
+ KEY `idx_joined_at` (`joined_at`),
+ KEY `idx_chat_user_rooms` (`user_id`,`left_at`,`joined_at` DESC),
+ KEY `idx_can_write` (`can_write`),
+ CONSTRAINT `fk_chat_participants_entite` FOREIGN KEY (`entite_id`) REFERENCES `entites` (`id`) ON DELETE SET NULL,
+ CONSTRAINT `fk_chat_participants_room` FOREIGN KEY (`room_id`) REFERENCES `chat_rooms` (`id`) ON DELETE CASCADE,
+ CONSTRAINT `fk_chat_participants_user` FOREIGN KEY (`user_id`) REFERENCES `users` (`id`) ON DELETE CASCADE
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci COMMENT='Participants aux conversations' `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `chat_read_receipts`
+--
+
+DROP TABLE IF EXISTS `chat_read_receipts`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `chat_read_receipts` (
+ `message_id` varchar(36) NOT NULL COMMENT 'ID du message',
+ `user_id` int(11) unsigned NOT NULL COMMENT 'ID de l''utilisateur',
+ `read_at` timestamp NOT NULL DEFAULT current_timestamp() COMMENT 'Date de lecture',
+ PRIMARY KEY (`message_id`,`user_id`),
+ KEY `idx_user_id` (`user_id`),
+ KEY `idx_read_at` (`read_at`),
+ CONSTRAINT `fk_chat_read_message` FOREIGN KEY (`message_id`) REFERENCES `chat_messages` (`id`) ON DELETE CASCADE,
+ CONSTRAINT `fk_chat_read_user` FOREIGN KEY (`user_id`) REFERENCES `users` (`id`) ON DELETE CASCADE
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci COMMENT='Accusés de lecture' `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `chat_rooms`
+--
+
+DROP TABLE IF EXISTS `chat_rooms`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `chat_rooms` (
+ `id` varchar(36) NOT NULL COMMENT 'UUID de la salle',
+ `title` varchar(255) DEFAULT NULL COMMENT 'Titre de la conversation',
+ `type` enum('private','group','broadcast') NOT NULL DEFAULT 'private' COMMENT 'Type de conversation',
+ `created_at` timestamp NOT NULL DEFAULT current_timestamp() COMMENT 'Date de création',
+ `created_by` int(11) unsigned NOT NULL COMMENT 'ID du créateur',
+ `updated_at` timestamp NULL DEFAULT NULL ON UPDATE current_timestamp() COMMENT 'Dernière modification',
+ `is_active` tinyint(1) NOT NULL DEFAULT 1 COMMENT 'Conversation active',
+ PRIMARY KEY (`id`),
+ KEY `idx_created_by` (`created_by`),
+ KEY `idx_type` (`type`),
+ KEY `idx_created_at` (`created_at`),
+ KEY `idx_chat_active_rooms` (`is_active`,`created_at` DESC),
+ CONSTRAINT `fk_chat_rooms_creator` FOREIGN KEY (`created_by`) REFERENCES `users` (`id`) ON DELETE CASCADE
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci COMMENT='Salles de conversation' `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Temporary table structure for view `chat_rooms_with_last_message`
+--
+
+DROP TABLE IF EXISTS `chat_rooms_with_last_message`;
+/*!50001 DROP VIEW IF EXISTS `chat_rooms_with_last_message`*/;
+SET @saved_cs_client = @@character_set_client;
+SET character_set_client = utf8mb4;
+/*!50001 CREATE VIEW `chat_rooms_with_last_message` AS SELECT
+ 1 AS `id`,
+ 1 AS `title`,
+ 1 AS `type`,
+ 1 AS `created_at`,
+ 1 AS `created_by`,
+ 1 AS `updated_at`,
+ 1 AS `is_active`,
+ 1 AS `last_message_content`,
+ 1 AS `last_message_sender`,
+ 1 AS `last_message_at`,
+ 1 AS `last_message_sender_name` */;
+SET character_set_client = @saved_cs_client;
+
+--
+-- Table structure for table `email_counter`
+--
+
+DROP TABLE IF EXISTS `email_counter`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `email_counter` (
+ `id` int(10) unsigned NOT NULL DEFAULT 1,
+ `hour_start` timestamp NULL DEFAULT NULL,
+ `count` int(10) unsigned DEFAULT 0,
+ PRIMARY KEY (`id`)
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `email_queue`
+--
+
+DROP TABLE IF EXISTS `email_queue`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `email_queue` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `fk_pass` int(10) unsigned NOT NULL DEFAULT 0,
+ `to_email` varchar(255) DEFAULT NULL,
+ `subject` varchar(255) DEFAULT NULL,
+ `body` text DEFAULT NULL,
+ `headers` text DEFAULT NULL,
+ `created_at` timestamp NULL DEFAULT current_timestamp(),
+ `status` enum('pending','sent','failed') DEFAULT 'pending',
+ `sent_at` timestamp NULL DEFAULT NULL COMMENT 'Date/heure envoi effectif de l email',
+ `attempts` int(10) unsigned DEFAULT 0,
+ `error_message` text DEFAULT NULL COMMENT 'Message erreur en cas d échec',
+ PRIMARY KEY (`id`),
+ KEY `idx_status_attempts` (`status`,`attempts`),
+ KEY `idx_sent_at` (`sent_at`)
+) ENGINE=InnoDB AUTO_INCREMENT=5 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `entites`
+--
+
+DROP TABLE IF EXISTS `entites`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `entites` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `encrypted_name` varchar(255) DEFAULT NULL,
+ `adresse1` varchar(45) DEFAULT '',
+ `adresse2` varchar(45) DEFAULT '',
+ `code_postal` varchar(5) DEFAULT '',
+ `ville` varchar(45) DEFAULT '',
+ `fk_region` int(10) unsigned DEFAULT NULL,
+ `fk_type` int(10) unsigned DEFAULT 1,
+ `encrypted_phone` varchar(128) DEFAULT '',
+ `encrypted_mobile` varchar(128) DEFAULT '',
+ `encrypted_email` varchar(255) DEFAULT '',
+ `gps_lat` varchar(20) NOT NULL DEFAULT '',
+ `gps_lng` varchar(20) NOT NULL DEFAULT '',
+ `chk_stripe` tinyint(1) unsigned DEFAULT 0,
+ `encrypted_stripe_id` varchar(255) DEFAULT '',
+ `encrypted_iban` varchar(255) DEFAULT '',
+ `encrypted_bic` varchar(128) DEFAULT '',
+ `chk_demo` tinyint(1) unsigned DEFAULT 1,
+ `chk_mdp_manuel` tinyint(1) unsigned NOT NULL DEFAULT 0 COMMENT 'Gestion des mots de passe manuelle (1) ou automatique (0)',
+ `chk_username_manuel` tinyint(1) unsigned NOT NULL DEFAULT 0 COMMENT 'Gestion des usernames manuelle (1) ou automatique (0)',
+ `chk_user_delete_pass` tinyint(1) unsigned NOT NULL DEFAULT 0 COMMENT 'Autoriser les membres à supprimer des passages (1) ou non (0)',
+ `chk_copie_mail_recu` tinyint(1) unsigned NOT NULL DEFAULT 0,
+ `chk_accept_sms` tinyint(1) unsigned NOT NULL DEFAULT 0,
+ `chk_lot_actif` tinyint(1) unsigned NOT NULL DEFAULT 0,
+ `created_at` timestamp NOT NULL DEFAULT current_timestamp() COMMENT 'Date de création',
+ `fk_user_creat` int(10) unsigned DEFAULT NULL,
+ `updated_at` timestamp NULL DEFAULT NULL ON UPDATE current_timestamp() COMMENT 'Date de modification',
+ `fk_user_modif` int(10) unsigned DEFAULT NULL,
+ `chk_active` tinyint(1) unsigned DEFAULT 1,
+ PRIMARY KEY (`id`),
+ KEY `entites_ibfk_1` (`fk_region`),
+ KEY `entites_ibfk_2` (`fk_type`),
+ CONSTRAINT `entites_ibfk_1` FOREIGN KEY (`fk_region`) REFERENCES `x_regions` (`id`) ON UPDATE CASCADE,
+ CONSTRAINT `entites_ibfk_2` FOREIGN KEY (`fk_type`) REFERENCES `x_entites_types` (`id`) ON UPDATE CASCADE
+) ENGINE=InnoDB AUTO_INCREMENT=1230 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `medias`
+--
+
+DROP TABLE IF EXISTS `medias`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `medias` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `support` varchar(45) NOT NULL DEFAULT '',
+ `support_id` int(10) unsigned NOT NULL DEFAULT 0,
+ `fk_entite` int(10) unsigned DEFAULT NULL COMMENT 'ID de entité propriétaire',
+ `fk_operation` int(10) unsigned DEFAULT NULL COMMENT 'ID de opération (pour passages)',
+ `fichier` varchar(250) NOT NULL DEFAULT '',
+ `file_type` varchar(50) DEFAULT NULL COMMENT 'Extension du fichier (pdf, jpg, xlsx, etc.)',
+ `file_category` varchar(50) DEFAULT NULL COMMENT 'export, logo, carte',
+ `file_size` int(10) unsigned DEFAULT NULL COMMENT 'Taille du fichier en octets',
+ `mime_type` varchar(100) DEFAULT NULL COMMENT 'Type MIME du fichier',
+ `original_name` varchar(255) DEFAULT NULL COMMENT 'Nom original du fichier uploadé',
+ `file_path` varchar(500) DEFAULT NULL COMMENT 'Chemin complet du fichier',
+ `original_width` int(10) unsigned DEFAULT NULL COMMENT 'Largeur originale de image',
+ `original_height` int(10) unsigned DEFAULT NULL COMMENT 'Hauteur originale de image',
+ `processed_width` int(10) unsigned DEFAULT NULL COMMENT 'Largeur après traitement',
+ `processed_height` int(10) unsigned DEFAULT NULL COMMENT 'Hauteur après traitement',
+ `is_processed` tinyint(1) unsigned DEFAULT 0 COMMENT 'Image redimensionnée (1) ou originale (0)',
+ `description` varchar(100) NOT NULL DEFAULT '',
+ `created_at` timestamp NOT NULL DEFAULT current_timestamp(),
+ `fk_user_creat` int(10) unsigned NOT NULL DEFAULT 0,
+ `updated_at` timestamp NULL DEFAULT NULL ON UPDATE current_timestamp(),
+ `fk_user_modif` int(10) unsigned NOT NULL DEFAULT 0,
+ PRIMARY KEY (`id`),
+ UNIQUE KEY `id_UNIQUE` (`id`),
+ KEY `idx_entite` (`fk_entite`),
+ KEY `idx_operation` (`fk_operation`),
+ KEY `idx_support_type` (`support`,`support_id`),
+ KEY `idx_file_type` (`file_type`),
+ KEY `idx_file_category` (`file_category`),
+ CONSTRAINT `fk_medias_entite` FOREIGN KEY (`fk_entite`) REFERENCES `entites` (`id`) ON DELETE CASCADE ON UPDATE CASCADE,
+ CONSTRAINT `fk_medias_operation` FOREIGN KEY (`fk_operation`) REFERENCES `operations` (`id`) ON DELETE CASCADE ON UPDATE CASCADE
+) ENGINE=InnoDB AUTO_INCREMENT=184 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `ope_pass`
+--
+
+DROP TABLE IF EXISTS `ope_pass`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `ope_pass` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `fk_operation` int(10) unsigned NOT NULL DEFAULT 0,
+ `fk_sector` int(10) unsigned DEFAULT NULL,
+ `fk_user` int(10) unsigned NOT NULL DEFAULT 0,
+ `fk_adresse` varchar(25) DEFAULT '' COMMENT 'adresses.cp??.id',
+ `passed_at` timestamp NULL DEFAULT NULL COMMENT 'Date du passage',
+ `fk_type` int(10) unsigned DEFAULT 0,
+ `numero` varchar(10) NOT NULL DEFAULT '',
+ `rue` varchar(75) NOT NULL DEFAULT '',
+ `rue_bis` varchar(1) NOT NULL DEFAULT '',
+ `ville` varchar(75) NOT NULL DEFAULT '',
+ `fk_habitat` int(10) unsigned DEFAULT 1,
+ `appt` varchar(5) DEFAULT '',
+ `niveau` varchar(5) DEFAULT '',
+ `residence` varchar(75) DEFAULT '',
+ `gps_lat` varchar(20) NOT NULL DEFAULT '',
+ `gps_lng` varchar(20) NOT NULL DEFAULT '',
+ `encrypted_name` varchar(255) NOT NULL DEFAULT '',
+ `montant` decimal(7,2) NOT NULL DEFAULT 0.00,
+ `fk_type_reglement` int(10) unsigned DEFAULT 4,
+ `remarque` text DEFAULT '',
+ `encrypted_email` varchar(255) DEFAULT '',
+ `nom_recu` varchar(50) DEFAULT NULL,
+ `date_recu` timestamp NULL DEFAULT NULL COMMENT 'Date de réception',
+ `date_creat_recu` timestamp NULL DEFAULT NULL COMMENT 'Date de création du reçu',
+ `date_sent_recu` timestamp NULL DEFAULT NULL COMMENT 'Date envoi du reçu',
+ `email_erreur` varchar(30) DEFAULT '',
+ `chk_email_sent` tinyint(1) unsigned NOT NULL DEFAULT 0,
+ `encrypted_phone` varchar(128) NOT NULL DEFAULT '',
+ `docremis` tinyint(1) unsigned DEFAULT 0,
+ `date_repasser` timestamp NULL DEFAULT NULL COMMENT 'Date prévue pour repasser',
+ `nb_passages` int(11) DEFAULT 1 COMMENT 'Nb passages pour les a repasser',
+ `chk_gps_maj` tinyint(1) unsigned DEFAULT 0,
+ `chk_map_create` tinyint(1) unsigned DEFAULT 0,
+ `chk_mobile` tinyint(1) unsigned DEFAULT 0,
+ `chk_synchro` tinyint(1) unsigned DEFAULT 1 COMMENT 'chk synchro entre web et appli',
+ `chk_api_adresse` tinyint(1) unsigned DEFAULT 0,
+ `chk_maj_adresse` tinyint(1) unsigned DEFAULT 0,
+ `anomalie` tinyint(1) unsigned DEFAULT 0,
+ `created_at` timestamp NOT NULL DEFAULT current_timestamp() COMMENT 'Date de création',
+ `fk_user_creat` int(10) unsigned DEFAULT NULL,
+ `updated_at` timestamp NULL DEFAULT NULL ON UPDATE current_timestamp() COMMENT 'Date de modification',
+ `fk_user_modif` int(10) unsigned DEFAULT NULL,
+ `chk_active` tinyint(1) unsigned NOT NULL DEFAULT 1,
+ `stripe_payment_id` varchar(50) DEFAULT NULL COMMENT 'ID du PaymentIntent Stripe (pi_xxx)',
+ PRIMARY KEY (`id`),
+ KEY `fk_operation` (`fk_operation`),
+ KEY `fk_sector` (`fk_sector`),
+ KEY `fk_user` (`fk_user`),
+ KEY `fk_type` (`fk_type`),
+ KEY `fk_type_reglement` (`fk_type_reglement`),
+ KEY `email` (`encrypted_email`),
+ KEY `idx_fk_adresse` (`fk_adresse`),
+ KEY `idx_address_lookup` (`fk_operation`,`numero`,`rue`,`ville`),
+ KEY `idx_stripe_payment` (`stripe_payment_id`),
+ CONSTRAINT `ope_pass_ibfk_1` FOREIGN KEY (`fk_operation`) REFERENCES `operations` (`id`) ON UPDATE CASCADE,
+ CONSTRAINT `ope_pass_ibfk_2` FOREIGN KEY (`fk_sector`) REFERENCES `ope_sectors` (`id`) ON UPDATE CASCADE,
+ CONSTRAINT `ope_pass_ibfk_3` FOREIGN KEY (`fk_user`) REFERENCES `users` (`id`) ON UPDATE CASCADE,
+ CONSTRAINT `ope_pass_ibfk_4` FOREIGN KEY (`fk_type_reglement`) REFERENCES `x_types_reglements` (`id`) ON UPDATE CASCADE
+) ENGINE=InnoDB AUTO_INCREMENT=19502730 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `ope_pass_histo`
+--
+
+DROP TABLE IF EXISTS `ope_pass_histo`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `ope_pass_histo` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `fk_pass` int(10) unsigned NOT NULL DEFAULT 0,
+ `date_histo` timestamp NOT NULL DEFAULT current_timestamp() COMMENT 'Date historique',
+ `sujet` varchar(50) DEFAULT NULL,
+ `remarque` varchar(250) NOT NULL DEFAULT '',
+ PRIMARY KEY (`id`),
+ KEY `ope_pass_histo_fk_pass_IDX` (`fk_pass`) USING BTREE,
+ KEY `ope_pass_histo_date_histo_IDX` (`date_histo`) USING BTREE,
+ CONSTRAINT `ope_pass_histo_ibfk_1` FOREIGN KEY (`fk_pass`) REFERENCES `ope_pass` (`id`) ON DELETE CASCADE ON UPDATE CASCADE
+) ENGINE=InnoDB AUTO_INCREMENT=6752 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `ope_sectors`
+--
+
+DROP TABLE IF EXISTS `ope_sectors`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `ope_sectors` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `fk_operation` int(10) unsigned NOT NULL DEFAULT 0,
+ `fk_old_sector` int(10) unsigned DEFAULT NULL,
+ `libelle` varchar(75) NOT NULL DEFAULT '',
+ `sector` text NOT NULL DEFAULT '',
+ `color` varchar(7) NOT NULL DEFAULT '#4B77BE',
+ `created_at` timestamp NOT NULL DEFAULT current_timestamp() COMMENT 'Date de création',
+ `fk_user_creat` int(10) unsigned NOT NULL DEFAULT 0,
+ `updated_at` timestamp NULL DEFAULT NULL ON UPDATE current_timestamp() COMMENT 'Date de modification',
+ `fk_user_modif` int(10) unsigned NOT NULL DEFAULT 0,
+ `chk_active` tinyint(1) unsigned NOT NULL DEFAULT 1,
+ PRIMARY KEY (`id`),
+ UNIQUE KEY `id` (`id`),
+ KEY `fk_operation` (`fk_operation`),
+ CONSTRAINT `ope_sectors_ibfk_1` FOREIGN KEY (`fk_operation`) REFERENCES `operations` (`id`) ON UPDATE CASCADE
+) ENGINE=InnoDB AUTO_INCREMENT=27697 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `ope_users`
+--
+
+DROP TABLE IF EXISTS `ope_users`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `ope_users` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `fk_operation` int(10) unsigned NOT NULL DEFAULT 0,
+ `fk_user` int(10) unsigned NOT NULL DEFAULT 0,
+ `fk_role` int(10) unsigned DEFAULT 1,
+ `first_name` varchar(45) DEFAULT '',
+ `encrypted_name` varchar(255) DEFAULT '',
+ `sect_name` varchar(60) DEFAULT '',
+ `created_at` timestamp NOT NULL DEFAULT current_timestamp() COMMENT 'Date de création',
+ `fk_user_creat` int(10) unsigned DEFAULT NULL,
+ `updated_at` timestamp NULL DEFAULT NULL ON UPDATE current_timestamp() COMMENT 'Date de modification',
+ `fk_user_modif` int(10) unsigned DEFAULT NULL,
+ `chk_active` tinyint(1) unsigned NOT NULL DEFAULT 1,
+ PRIMARY KEY (`id`),
+ UNIQUE KEY `id_UNIQUE` (`id`),
+ UNIQUE KEY `idx_operation_user` (`fk_operation`,`fk_user`),
+ KEY `ope_users_ibfk_1` (`fk_operation`),
+ KEY `ope_users_ibfk_2` (`fk_user`),
+ CONSTRAINT `ope_users_ibfk_1` FOREIGN KEY (`fk_operation`) REFERENCES `operations` (`id`) ON UPDATE CASCADE,
+ CONSTRAINT `ope_users_ibfk_2` FOREIGN KEY (`fk_user`) REFERENCES `users` (`id`) ON UPDATE CASCADE
+) ENGINE=InnoDB AUTO_INCREMENT=199016 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `ope_users_sectors`
+--
+
+DROP TABLE IF EXISTS `ope_users_sectors`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `ope_users_sectors` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `fk_operation` int(10) unsigned NOT NULL DEFAULT 0,
+ `fk_user` int(10) unsigned NOT NULL DEFAULT 0,
+ `fk_sector` int(10) unsigned NOT NULL DEFAULT 0,
+ `created_at` timestamp NOT NULL DEFAULT current_timestamp() COMMENT 'Date de création',
+ `fk_user_creat` int(10) unsigned NOT NULL DEFAULT 0,
+ `updated_at` timestamp NULL DEFAULT NULL ON UPDATE current_timestamp() COMMENT 'Date de modification',
+ `fk_user_modif` int(10) unsigned DEFAULT NULL,
+ `chk_active` tinyint(1) unsigned DEFAULT 1,
+ PRIMARY KEY (`id`),
+ UNIQUE KEY `id` (`id`),
+ UNIQUE KEY `idx_operation_user_sector` (`fk_operation`,`fk_user`,`fk_sector`),
+ KEY `fk_operation` (`fk_operation`),
+ KEY `fk_user` (`fk_user`),
+ KEY `fk_sector` (`fk_sector`),
+ CONSTRAINT `ope_users_sectors_ibfk_1` FOREIGN KEY (`fk_operation`) REFERENCES `operations` (`id`) ON UPDATE CASCADE,
+ CONSTRAINT `ope_users_sectors_ibfk_2` FOREIGN KEY (`fk_user`) REFERENCES `users` (`id`) ON UPDATE CASCADE,
+ CONSTRAINT `ope_users_sectors_ibfk_3` FOREIGN KEY (`fk_sector`) REFERENCES `ope_sectors` (`id`) ON UPDATE CASCADE
+) ENGINE=InnoDB AUTO_INCREMENT=48141 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `operations`
+--
+
+DROP TABLE IF EXISTS `operations`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `operations` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `fk_entite` int(10) unsigned NOT NULL DEFAULT 1,
+ `libelle` varchar(75) NOT NULL DEFAULT '',
+ `date_deb` date NOT NULL DEFAULT '0000-00-00',
+ `date_fin` date NOT NULL DEFAULT '0000-00-00',
+ `chk_distinct_sectors` tinyint(1) unsigned NOT NULL DEFAULT 0,
+ `created_at` timestamp NOT NULL DEFAULT current_timestamp() COMMENT 'Date de création',
+ `fk_user_creat` int(10) unsigned NOT NULL DEFAULT 0,
+ `updated_at` timestamp NULL DEFAULT NULL ON UPDATE current_timestamp() COMMENT 'Date de modification',
+ `fk_user_modif` int(10) unsigned NOT NULL DEFAULT 0,
+ `chk_active` tinyint(1) unsigned NOT NULL DEFAULT 1,
+ PRIMARY KEY (`id`),
+ KEY `fk_entite` (`fk_entite`),
+ KEY `date_deb` (`date_deb`),
+ CONSTRAINT `operations_ibfk_1` FOREIGN KEY (`fk_entite`) REFERENCES `entites` (`id`) ON UPDATE CASCADE
+) ENGINE=InnoDB AUTO_INCREMENT=3122 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `params`
+--
+
+DROP TABLE IF EXISTS `params`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `params` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `libelle` varchar(35) NOT NULL DEFAULT '',
+ `valeur` varchar(255) NOT NULL DEFAULT '',
+ `aide` varchar(150) NOT NULL DEFAULT '',
+ PRIMARY KEY (`id`)
+) ENGINE=InnoDB AUTO_INCREMENT=2 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `sec_alerts`
+--
+
+DROP TABLE IF EXISTS `sec_alerts`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `sec_alerts` (
+ `id` int(11) unsigned NOT NULL AUTO_INCREMENT,
+ `alert_type` varchar(50) NOT NULL COMMENT 'Type d''alerte (BRUTE_FORCE, SQL_ERROR, etc.)',
+ `alert_level` enum('INFO','WARNING','ERROR','CRITICAL','SECURITY') NOT NULL DEFAULT 'INFO',
+ `ip_address` varchar(45) DEFAULT NULL COMMENT 'Adresse IP source',
+ `user_id` int(11) unsigned DEFAULT NULL COMMENT 'ID utilisateur si connecté',
+ `username` varchar(255) DEFAULT NULL COMMENT 'Username tenté ou utilisé',
+ `endpoint` varchar(255) DEFAULT NULL COMMENT 'Endpoint API concerné',
+ `method` varchar(10) DEFAULT NULL COMMENT 'Méthode HTTP',
+ `details` longtext CHARACTER SET utf8mb4 COLLATE utf8mb4_bin DEFAULT NULL COMMENT 'Détails additionnels en JSON' CHECK (json_valid(`details`)),
+ `occurrences` int(11) DEFAULT 1 COMMENT 'Nombre d''occurrences',
+ `first_seen` timestamp NOT NULL DEFAULT current_timestamp(),
+ `last_seen` timestamp NOT NULL DEFAULT current_timestamp() ON UPDATE current_timestamp(),
+ `email_sent` tinyint(1) DEFAULT 0 COMMENT 'Email d''alerte envoyé',
+ `email_sent_at` timestamp NULL DEFAULT NULL,
+ `resolved` tinyint(1) DEFAULT 0 COMMENT 'Alerte résolue',
+ `resolved_at` timestamp NULL DEFAULT NULL,
+ `resolved_by` int(11) unsigned DEFAULT NULL COMMENT 'ID admin qui a résolu',
+ `notes` text DEFAULT NULL COMMENT 'Notes de résolution',
+ PRIMARY KEY (`id`),
+ KEY `idx_ip` (`ip_address`),
+ KEY `idx_type_time` (`alert_type`,`last_seen`),
+ KEY `idx_level` (`alert_level`),
+ KEY `idx_resolved` (`resolved`),
+ KEY `idx_user` (`user_id`),
+ KEY `fk_sec_alerts_resolver` (`resolved_by`),
+ KEY `idx_sec_alerts_recent` (`last_seen` DESC,`alert_level`),
+ CONSTRAINT `fk_sec_alerts_resolver` FOREIGN KEY (`resolved_by`) REFERENCES `users` (`id`) ON DELETE SET NULL,
+ CONSTRAINT `fk_sec_alerts_user` FOREIGN KEY (`user_id`) REFERENCES `users` (`id`) ON DELETE SET NULL
+) ENGINE=InnoDB AUTO_INCREMENT=234 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci COMMENT='Alertes de sécurité et monitoring' `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `sec_blocked_ips`
+--
+
+DROP TABLE IF EXISTS `sec_blocked_ips`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `sec_blocked_ips` (
+ `ip_address` varchar(45) NOT NULL COMMENT 'Adresse IP bloquée',
+ `reason` varchar(255) NOT NULL COMMENT 'Raison du blocage',
+ `details` longtext CHARACTER SET utf8mb4 COLLATE utf8mb4_bin DEFAULT NULL COMMENT 'Détails additionnels' CHECK (json_valid(`details`)),
+ `blocked_at` timestamp NOT NULL DEFAULT current_timestamp(),
+ `blocked_until` timestamp NOT NULL COMMENT 'Bloqué jusqu''à',
+ `blocked_by` varchar(50) DEFAULT 'system' COMMENT 'Qui a bloqué (system ou user ID)',
+ `permanent` tinyint(1) DEFAULT 0 COMMENT 'Blocage permanent',
+ `unblocked_at` timestamp NULL DEFAULT NULL COMMENT 'Date de déblocage effectif',
+ `unblocked_by` int(11) unsigned DEFAULT NULL COMMENT 'Qui a débloqué',
+ `block_count` int(11) DEFAULT 1 COMMENT 'Nombre de fois bloquée',
+ PRIMARY KEY (`ip_address`),
+ KEY `idx_blocked_until` (`blocked_until`),
+ KEY `idx_permanent` (`permanent`),
+ KEY `fk_sec_blocked_unblocked_by` (`unblocked_by`),
+ CONSTRAINT `fk_sec_blocked_unblocked_by` FOREIGN KEY (`unblocked_by`) REFERENCES `users` (`id`) ON DELETE SET NULL
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci COMMENT='IPs bloquées temporairement ou définitivement' `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `sec_failed_login_attempts`
+--
+
+DROP TABLE IF EXISTS `sec_failed_login_attempts`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `sec_failed_login_attempts` (
+ `id` int(11) unsigned NOT NULL AUTO_INCREMENT,
+ `username` varchar(255) DEFAULT NULL COMMENT 'Username tenté',
+ `encrypted_username` varchar(255) DEFAULT NULL COMMENT 'Username chiffré si trouvé',
+ `ip_address` varchar(45) NOT NULL COMMENT 'Adresse IP',
+ `user_agent` text DEFAULT NULL COMMENT 'User Agent',
+ `attempt_time` timestamp NOT NULL DEFAULT current_timestamp(),
+ `error_type` varchar(50) DEFAULT NULL COMMENT 'Type d''erreur (invalid_password, user_not_found, etc.)',
+ `country_code` varchar(2) DEFAULT NULL COMMENT 'Code pays de l''IP (si géoloc activée)',
+ PRIMARY KEY (`id`),
+ KEY `idx_ip_time` (`ip_address`,`attempt_time`),
+ KEY `idx_username` (`username`),
+ KEY `idx_encrypted_username` (`encrypted_username`),
+ KEY `idx_time` (`attempt_time`),
+ KEY `idx_sec_failed_recent` (`attempt_time` DESC,`ip_address`)
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci COMMENT='Tentatives de connexion échouées' `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `sec_performance_metrics`
+--
+
+DROP TABLE IF EXISTS `sec_performance_metrics`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `sec_performance_metrics` (
+ `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
+ `endpoint` varchar(255) NOT NULL COMMENT 'Endpoint API',
+ `method` varchar(10) NOT NULL COMMENT 'Méthode HTTP',
+ `response_time_ms` int(11) NOT NULL COMMENT 'Temps de réponse total en ms',
+ `db_time_ms` int(11) DEFAULT 0 COMMENT 'Temps cumulé des requêtes DB en ms',
+ `db_queries_count` int(11) DEFAULT 0 COMMENT 'Nombre de requêtes DB',
+ `memory_peak_mb` float DEFAULT NULL COMMENT 'Pic mémoire en MB',
+ `memory_start_mb` float DEFAULT NULL COMMENT 'Mémoire au début en MB',
+ `http_status` int(11) DEFAULT NULL COMMENT 'Code HTTP de réponse',
+ `user_id` int(11) unsigned DEFAULT NULL COMMENT 'ID utilisateur si connecté',
+ `ip_address` varchar(45) DEFAULT NULL COMMENT 'Adresse IP',
+ `user_agent` text DEFAULT NULL COMMENT 'User Agent complet',
+ `request_size` int(11) DEFAULT NULL COMMENT 'Taille de la requête en octets',
+ `response_size` int(11) DEFAULT NULL COMMENT 'Taille de la réponse en octets',
+ `created_at` timestamp NOT NULL DEFAULT current_timestamp(),
+ PRIMARY KEY (`id`),
+ KEY `idx_endpoint_time` (`endpoint`,`created_at`),
+ KEY `idx_response_time` (`response_time_ms`),
+ KEY `idx_created` (`created_at`),
+ KEY `idx_status` (`http_status`),
+ KEY `idx_user` (`user_id`),
+ KEY `idx_date_endpoint` (`created_at`,`endpoint`),
+ KEY `idx_sec_metrics_recent` (`created_at` DESC,`endpoint`)
+) ENGINE=InnoDB AUTO_INCREMENT=12348 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci COMMENT='Métriques de performance des requêtes' `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `sectors_adresses`
+--
+
+DROP TABLE IF EXISTS `sectors_adresses`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `sectors_adresses` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `fk_adresse` varchar(25) DEFAULT NULL COMMENT 'adresses.cp??.id',
+ `osm_id` int(10) unsigned NOT NULL DEFAULT 0,
+ `fk_sector` int(10) unsigned NOT NULL DEFAULT 0,
+ `osm_name` varchar(50) NOT NULL DEFAULT '',
+ `numero` varchar(5) NOT NULL DEFAULT '',
+ `rue_bis` varchar(5) NOT NULL DEFAULT '',
+ `rue` varchar(60) NOT NULL DEFAULT '',
+ `cp` varchar(5) NOT NULL DEFAULT '',
+ `ville` varchar(60) NOT NULL DEFAULT '',
+ `gps_lat` varchar(20) NOT NULL DEFAULT '',
+ `gps_lng` varchar(20) NOT NULL DEFAULT '',
+ `osm_date_creat` timestamp NOT NULL DEFAULT '0000-00-00 00:00:00',
+ `created_at` timestamp NOT NULL DEFAULT current_timestamp() COMMENT 'Date de création',
+ `updated_at` timestamp NULL DEFAULT NULL ON UPDATE current_timestamp() COMMENT 'Date de modification',
+ PRIMARY KEY (`id`),
+ KEY `sectors_adresses_fk_sector_index` (`fk_sector`),
+ KEY `sectors_adresses_numero_index` (`numero`),
+ KEY `sectors_adresses_rue_index` (`rue`),
+ KEY `sectors_adresses_ville_index` (`ville`),
+ KEY `idx_fk_adresse` (`fk_adresse`),
+ CONSTRAINT `sectors_adresses_ibfk_1` FOREIGN KEY (`fk_sector`) REFERENCES `ope_sectors` (`id`) ON UPDATE CASCADE
+) ENGINE=InnoDB AUTO_INCREMENT=1570831 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `stripe_accounts`
+--
+
+DROP TABLE IF EXISTS `stripe_accounts`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `stripe_accounts` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `fk_entite` int(10) unsigned NOT NULL,
+ `stripe_account_id` varchar(255) DEFAULT NULL,
+ `stripe_location_id` varchar(255) DEFAULT NULL,
+ `charges_enabled` tinyint(1) DEFAULT 0,
+ `payouts_enabled` tinyint(1) DEFAULT 0,
+ `onboarding_completed` tinyint(1) DEFAULT 0,
+ `created_at` timestamp NULL DEFAULT current_timestamp(),
+ `updated_at` timestamp NULL DEFAULT current_timestamp() ON UPDATE current_timestamp(),
+ PRIMARY KEY (`id`),
+ UNIQUE KEY `stripe_account_id` (`stripe_account_id`),
+ KEY `idx_fk_entite` (`fk_entite`),
+ KEY `idx_stripe_account_id` (`stripe_account_id`),
+ CONSTRAINT `stripe_accounts_ibfk_1` FOREIGN KEY (`fk_entite`) REFERENCES `entites` (`id`) ON DELETE CASCADE
+) ENGINE=InnoDB AUTO_INCREMENT=2 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `stripe_android_certified_devices`
+--
+
+DROP TABLE IF EXISTS `stripe_android_certified_devices`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `stripe_android_certified_devices` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `manufacturer` varchar(100) DEFAULT NULL,
+ `model` varchar(200) DEFAULT NULL,
+ `model_identifier` varchar(200) DEFAULT NULL,
+ `tap_to_pay_certified` tinyint(1) DEFAULT 0,
+ `certification_date` date DEFAULT NULL,
+ `min_android_version` int(11) DEFAULT NULL,
+ `country` varchar(2) DEFAULT 'FR',
+ `notes` text DEFAULT NULL,
+ `last_verified` timestamp NULL DEFAULT current_timestamp(),
+ `created_at` timestamp NULL DEFAULT current_timestamp(),
+ `updated_at` timestamp NULL DEFAULT current_timestamp() ON UPDATE current_timestamp(),
+ PRIMARY KEY (`id`),
+ UNIQUE KEY `unique_device` (`manufacturer`,`model`,`model_identifier`),
+ KEY `idx_manufacturer_model` (`manufacturer`,`model`),
+ KEY `idx_certified` (`tap_to_pay_certified`,`country`)
+) ENGINE=InnoDB AUTO_INCREMENT=78 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `stripe_payment_history`
+--
+
+DROP TABLE IF EXISTS `stripe_payment_history`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `stripe_payment_history` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `stripe_payment_intent_id` varchar(255) DEFAULT NULL COMMENT 'ID du PaymentIntent Stripe',
+ `event_type` varchar(50) DEFAULT NULL COMMENT 'created, processing, succeeded, failed, refunded',
+ `event_data` longtext CHARACTER SET utf8mb4 COLLATE utf8mb4_bin DEFAULT NULL CHECK (json_valid(`event_data`)),
+ `webhook_id` varchar(255) DEFAULT NULL,
+ `created_at` timestamp NULL DEFAULT current_timestamp(),
+ PRIMARY KEY (`id`),
+ KEY `idx_event_type` (`event_type`),
+ KEY `idx_created_at` (`created_at`),
+ KEY `idx_stripe_payment_intent_id` (`stripe_payment_intent_id`)
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `stripe_refunds`
+--
+
+DROP TABLE IF EXISTS `stripe_refunds`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `stripe_refunds` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `stripe_refund_id` varchar(255) DEFAULT NULL,
+ `stripe_payment_intent_id` varchar(255) NOT NULL COMMENT 'ID du PaymentIntent Stripe',
+ `amount` int(11) NOT NULL COMMENT 'Montant remboursé en centimes',
+ `reason` varchar(100) DEFAULT NULL COMMENT 'duplicate, fraudulent, requested_by_customer',
+ `status` varchar(50) DEFAULT NULL,
+ `metadata` longtext CHARACTER SET utf8mb4 COLLATE utf8mb4_bin DEFAULT NULL CHECK (json_valid(`metadata`)),
+ `created_at` timestamp NULL DEFAULT current_timestamp(),
+ `updated_at` timestamp NULL DEFAULT current_timestamp() ON UPDATE current_timestamp(),
+ PRIMARY KEY (`id`),
+ UNIQUE KEY `stripe_refund_id` (`stripe_refund_id`),
+ KEY `idx_status` (`status`),
+ KEY `idx_stripe_payment_intent_id` (`stripe_payment_intent_id`)
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `stripe_terminal_readers`
+--
+
+DROP TABLE IF EXISTS `stripe_terminal_readers`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `stripe_terminal_readers` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `stripe_reader_id` varchar(255) DEFAULT NULL,
+ `fk_entite` int(10) unsigned NOT NULL,
+ `label` varchar(255) DEFAULT NULL,
+ `location` varchar(255) DEFAULT NULL,
+ `status` varchar(50) DEFAULT NULL,
+ `device_type` varchar(50) DEFAULT NULL COMMENT 'ios_tap_to_pay, android_tap_to_pay',
+ `device_info` longtext CHARACTER SET utf8mb4 COLLATE utf8mb4_bin DEFAULT NULL COMMENT 'Infos sur le device (modèle, OS, etc)' CHECK (json_valid(`device_info`)),
+ `last_seen_at` timestamp NULL DEFAULT NULL,
+ `created_at` timestamp NULL DEFAULT current_timestamp(),
+ `updated_at` timestamp NULL DEFAULT current_timestamp() ON UPDATE current_timestamp(),
+ PRIMARY KEY (`id`),
+ UNIQUE KEY `stripe_reader_id` (`stripe_reader_id`),
+ KEY `idx_fk_entite` (`fk_entite`),
+ KEY `idx_device_type` (`device_type`),
+ CONSTRAINT `stripe_terminal_readers_ibfk_1` FOREIGN KEY (`fk_entite`) REFERENCES `entites` (`id`) ON DELETE CASCADE
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `stripe_webhooks`
+--
+
+DROP TABLE IF EXISTS `stripe_webhooks`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `stripe_webhooks` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `stripe_event_id` varchar(255) DEFAULT NULL,
+ `event_type` varchar(100) DEFAULT NULL,
+ `livemode` tinyint(1) DEFAULT 0,
+ `payload` longtext CHARACTER SET utf8mb4 COLLATE utf8mb4_bin DEFAULT NULL CHECK (json_valid(`payload`)),
+ `processed` tinyint(1) DEFAULT 0,
+ `error_message` text DEFAULT NULL,
+ `created_at` timestamp NULL DEFAULT current_timestamp(),
+ `processed_at` timestamp NULL DEFAULT NULL,
+ PRIMARY KEY (`id`),
+ UNIQUE KEY `stripe_event_id` (`stripe_event_id`),
+ KEY `idx_event_type` (`event_type`),
+ KEY `idx_processed` (`processed`),
+ KEY `idx_created_at` (`created_at`)
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `user_devices`
+--
+
+DROP TABLE IF EXISTS `user_devices`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `user_devices` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `fk_user` int(10) unsigned NOT NULL COMMENT 'Référence vers la table users',
+ `platform` varchar(20) NOT NULL COMMENT 'Plateforme: iOS, Android, etc.',
+ `device_model` varchar(100) DEFAULT NULL COMMENT 'Modèle du device (ex: iPhone13,2)',
+ `device_name` varchar(255) DEFAULT NULL COMMENT 'Nom personnalisé du device',
+ `device_manufacturer` varchar(100) DEFAULT NULL COMMENT 'Fabricant (Apple, Samsung, etc.)',
+ `device_identifier` varchar(100) DEFAULT NULL COMMENT 'Identifiant unique du device',
+ `device_ip_local` varchar(15) DEFAULT NULL COMMENT 'Adresse IP locale IPv4',
+ `device_ip_public` varchar(15) DEFAULT NULL COMMENT 'Adresse IP publique IPv4',
+ `device_wifi_name` varchar(255) DEFAULT NULL COMMENT 'Nom du réseau WiFi (SSID)',
+ `device_wifi_bssid` varchar(17) DEFAULT NULL COMMENT 'BSSID du point d''accès (format\nXX:XX:XX:XX:XX:XX)',
+ `ios_version` varchar(20) DEFAULT NULL COMMENT 'Version iOS/Android OS',
+ `device_nfc_capable` tinyint(1) DEFAULT NULL COMMENT 'Support NFC (1=oui, 0=non)',
+ `device_supports_tap_to_pay` tinyint(1) DEFAULT NULL COMMENT 'Support Tap to Pay (1=oui, 0=non)',
+ `battery_level` tinyint(3) unsigned DEFAULT NULL COMMENT 'Niveau batterie en pourcentage (0-100)',
+ `battery_charging` tinyint(1) DEFAULT NULL COMMENT 'En charge (1=oui, 0=non)',
+ `battery_state` varchar(20) DEFAULT NULL COMMENT 'État batterie (charging, discharging, full)',
+ `app_version` varchar(20) DEFAULT NULL COMMENT 'Version de l''application (ex: 3.2.8)',
+ `app_build` varchar(20) DEFAULT NULL COMMENT 'Numéro de build (ex: 328)',
+ `last_device_info_check` timestamp NULL DEFAULT NULL COMMENT 'Dernier check des infos device côté\napp',
+ `created_at` timestamp NOT NULL DEFAULT current_timestamp() COMMENT 'Date de création de\nl''enregistrement',
+ `updated_at` timestamp NOT NULL DEFAULT current_timestamp() ON UPDATE current_timestamp() COMMENT 'Date de dernière modification',
+ PRIMARY KEY (`id`),
+ UNIQUE KEY `unique_user_device` (`fk_user`,`device_identifier`) COMMENT 'Un seul enregistrement\npar device/user',
+ KEY `idx_fk_user` (`fk_user`) COMMENT 'Index pour recherche par utilisateur',
+ KEY `idx_updated_at` (`updated_at`) COMMENT 'Index pour tri par date de mise à jour',
+ KEY `idx_last_check` (`last_device_info_check`) COMMENT 'Index pour recherche par dernière\nvérification',
+ CONSTRAINT `fk_user_devices_user` FOREIGN KEY (`fk_user`) REFERENCES `users` (`id`) ON DELETE CASCADE
+) ENGINE=InnoDB AUTO_INCREMENT=56 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci COMMENT='Informations des devices\nutilisateurs' `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `users`
+--
+
+DROP TABLE IF EXISTS `users`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `users` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `fk_entite` int(10) unsigned DEFAULT 1,
+ `fk_role` int(10) unsigned DEFAULT 1,
+ `fk_titre` int(10) unsigned DEFAULT 1,
+ `encrypted_name` varchar(255) DEFAULT NULL,
+ `first_name` varchar(45) DEFAULT NULL,
+ `sect_name` varchar(60) DEFAULT '',
+ `encrypted_user_name` varchar(255) DEFAULT '' COMMENT 'Username chiffré - Supporte UTF-8 30 caractères maximum',
+ `user_pass_hash` varchar(60) DEFAULT NULL,
+ `encrypted_phone` varchar(128) DEFAULT NULL,
+ `encrypted_mobile` varchar(128) DEFAULT NULL,
+ `encrypted_email` varchar(255) DEFAULT '',
+ `chk_alert_email` tinyint(1) unsigned DEFAULT 1,
+ `chk_suivi` tinyint(1) unsigned DEFAULT 0,
+ `date_naissance` date DEFAULT NULL,
+ `date_embauche` date DEFAULT NULL,
+ `created_at` timestamp NOT NULL DEFAULT current_timestamp() COMMENT 'Date de création',
+ `fk_user_creat` int(10) unsigned DEFAULT NULL,
+ `updated_at` timestamp NULL DEFAULT NULL ON UPDATE current_timestamp() COMMENT 'Date de modification',
+ `fk_user_modif` int(10) unsigned DEFAULT NULL,
+ `chk_active` tinyint(1) unsigned DEFAULT 1,
+ PRIMARY KEY (`id`),
+ KEY `fk_entite` (`fk_entite`),
+ KEY `username` (`encrypted_user_name`),
+ KEY `users_ibfk_2` (`fk_role`),
+ KEY `users_ibfk_3` (`fk_titre`),
+ KEY `idx_encrypted_user_name` (`encrypted_user_name`),
+ CONSTRAINT `users_ibfk_1` FOREIGN KEY (`fk_entite`) REFERENCES `entites` (`id`) ON UPDATE CASCADE,
+ CONSTRAINT `users_ibfk_2` FOREIGN KEY (`fk_role`) REFERENCES `x_users_roles` (`id`) ON UPDATE CASCADE,
+ CONSTRAINT `users_ibfk_3` FOREIGN KEY (`fk_titre`) REFERENCES `x_users_titres` (`id`) ON UPDATE CASCADE
+) ENGINE=InnoDB AUTO_INCREMENT=10027766 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Temporary table structure for view `v_stripe_entite_stats`
+--
+
+DROP TABLE IF EXISTS `v_stripe_entite_stats`;
+/*!50001 DROP VIEW IF EXISTS `v_stripe_entite_stats`*/;
+SET @saved_cs_client = @@character_set_client;
+SET character_set_client = utf8mb4;
+/*!50001 CREATE VIEW `v_stripe_entite_stats` AS SELECT
+ 1 AS `entite_id`,
+ 1 AS `entite_name`,
+ 1 AS `stripe_account_id`,
+ 1 AS `charges_enabled`,
+ 1 AS `payouts_enabled`,
+ 1 AS `total_passages`,
+ 1 AS `passages_stripe`,
+ 1 AS `revenue_stripe`,
+ 1 AS `revenue_total` */;
+SET character_set_client = @saved_cs_client;
+
+--
+-- Temporary table structure for view `v_stripe_payment_stats`
+--
+
+DROP TABLE IF EXISTS `v_stripe_payment_stats`;
+/*!50001 DROP VIEW IF EXISTS `v_stripe_payment_stats`*/;
+SET @saved_cs_client = @@character_set_client;
+SET character_set_client = utf8mb4;
+/*!50001 CREATE VIEW `v_stripe_payment_stats` AS SELECT
+ 1 AS `fk_entite`,
+ 1 AS `entite_name`,
+ 1 AS `fk_user`,
+ 1 AS `user_name`,
+ 1 AS `total_ventes`,
+ 1 AS `ventes_stripe`,
+ 1 AS `montant_stripe`,
+ 1 AS `montant_autres`,
+ 1 AS `date_vente` */;
+SET character_set_client = @saved_cs_client;
+
+--
+-- Table structure for table `x_departements`
+--
+
+DROP TABLE IF EXISTS `x_departements`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `x_departements` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `code` varchar(3) DEFAULT NULL,
+ `fk_region` int(10) unsigned DEFAULT 1,
+ `libelle` varchar(45) DEFAULT NULL,
+ `dept_limitrophes` varchar(100) DEFAULT NULL COMMENT 'Liste des codes départements limitrophes séparés par des virgules',
+ `contour` geometry DEFAULT NULL COMMENT 'Contour géographique du département',
+ `chk_active` tinyint(1) unsigned DEFAULT 1,
+ PRIMARY KEY (`id`),
+ UNIQUE KEY `id_UNIQUE` (`id`),
+ KEY `x_departements_ibfk_1` (`fk_region`),
+ CONSTRAINT `x_departements_ibfk_1` FOREIGN KEY (`fk_region`) REFERENCES `x_regions` (`id`) ON UPDATE CASCADE
+) ENGINE=InnoDB AUTO_INCREMENT=105 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `x_departements_contours`
+--
+
+DROP TABLE IF EXISTS `x_departements_contours`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `x_departements_contours` (
+ `id` int(11) NOT NULL AUTO_INCREMENT,
+ `code_dept` varchar(3) NOT NULL COMMENT 'Code département (22, 2A, 971...)',
+ `nom_dept` varchar(100) NOT NULL,
+ `contour` geometry NOT NULL COMMENT 'Géométrie du contour du département (Polygon ou MultiPolygon)',
+ `bbox_min_lat` decimal(10,0) DEFAULT NULL COMMENT 'Latitude min de la bounding box',
+ `bbox_max_lat` decimal(10,0) DEFAULT NULL COMMENT 'Latitude max de la bounding box',
+ `bbox_min_lng` decimal(11,0) DEFAULT NULL COMMENT 'Longitude min de la bounding box',
+ `bbox_max_lng` decimal(11,0) DEFAULT NULL COMMENT 'Longitude max de la bounding box',
+ `created_at` timestamp NOT NULL DEFAULT current_timestamp(),
+ `updated_at` timestamp NOT NULL DEFAULT current_timestamp() ON UPDATE current_timestamp(),
+ PRIMARY KEY (`id`),
+ UNIQUE KEY `idx_code_dept` (`code_dept`),
+ SPATIAL KEY `idx_contour` (`contour`),
+ KEY `idx_dept_bbox` (`bbox_min_lat`,`bbox_max_lat`,`bbox_min_lng`,`bbox_max_lng`)
+) ENGINE=InnoDB AUTO_INCREMENT=169 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci COMMENT='Contours géographiques des départements français' `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `x_devises`
+--
+
+DROP TABLE IF EXISTS `x_devises`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `x_devises` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `code` varchar(3) DEFAULT NULL,
+ `symbole` varchar(6) DEFAULT NULL,
+ `libelle` varchar(45) DEFAULT NULL,
+ `chk_active` tinyint(1) unsigned DEFAULT 1,
+ PRIMARY KEY (`id`),
+ UNIQUE KEY `id_UNIQUE` (`id`)
+) ENGINE=InnoDB AUTO_INCREMENT=2 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `x_entites_types`
+--
+
+DROP TABLE IF EXISTS `x_entites_types`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `x_entites_types` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `libelle` varchar(45) DEFAULT NULL,
+ `chk_active` tinyint(1) unsigned DEFAULT NULL,
+ PRIMARY KEY (`id`),
+ UNIQUE KEY `id_UNIQUE` (`id`)
+) ENGINE=InnoDB AUTO_INCREMENT=3 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `x_pays`
+--
+
+DROP TABLE IF EXISTS `x_pays`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `x_pays` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `code` varchar(3) DEFAULT NULL,
+ `fk_continent` int(10) unsigned DEFAULT NULL,
+ `fk_devise` int(10) unsigned DEFAULT 1,
+ `libelle` varchar(45) DEFAULT NULL,
+ `chk_active` tinyint(1) unsigned DEFAULT 1,
+ PRIMARY KEY (`id`),
+ UNIQUE KEY `id_UNIQUE` (`id`),
+ KEY `x_pays_ibfk_1` (`fk_devise`),
+ CONSTRAINT `x_pays_ibfk_1` FOREIGN KEY (`fk_devise`) REFERENCES `x_devises` (`id`) ON UPDATE CASCADE
+) ENGINE=InnoDB AUTO_INCREMENT=10 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci COMMENT='Table des pays avec leurs codes' `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `x_regions`
+--
+
+DROP TABLE IF EXISTS `x_regions`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `x_regions` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `fk_pays` int(10) unsigned DEFAULT 1,
+ `libelle` varchar(45) DEFAULT NULL,
+ `libelle_long` varchar(45) DEFAULT NULL,
+ `table_osm` varchar(45) DEFAULT NULL,
+ `departements` varchar(45) DEFAULT NULL,
+ `chk_active` tinyint(1) unsigned DEFAULT 1,
+ PRIMARY KEY (`id`),
+ UNIQUE KEY `id_UNIQUE` (`id`),
+ KEY `x_regions_ibfk_1` (`fk_pays`),
+ CONSTRAINT `x_regions_ibfk_1` FOREIGN KEY (`fk_pays`) REFERENCES `x_pays` (`id`) ON UPDATE CASCADE
+) ENGINE=InnoDB AUTO_INCREMENT=29 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `x_types_passages`
+--
+
+DROP TABLE IF EXISTS `x_types_passages`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `x_types_passages` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `libelle` varchar(10) DEFAULT NULL,
+ `color_button` varchar(15) DEFAULT NULL,
+ `color_mark` varchar(15) DEFAULT NULL,
+ `color_table` varchar(15) DEFAULT NULL,
+ `chk_active` tinyint(1) unsigned NOT NULL DEFAULT 1,
+ PRIMARY KEY (`id`)
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `x_types_reglements`
+--
+
+DROP TABLE IF EXISTS `x_types_reglements`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `x_types_reglements` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `libelle` varchar(45) DEFAULT NULL,
+ `chk_active` tinyint(1) unsigned DEFAULT 1,
+ PRIMARY KEY (`id`)
+) ENGINE=InnoDB AUTO_INCREMENT=5 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `x_users_roles`
+--
+
+DROP TABLE IF EXISTS `x_users_roles`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `x_users_roles` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `libelle` varchar(45) DEFAULT NULL,
+ `chk_active` tinyint(1) unsigned DEFAULT 1,
+ PRIMARY KEY (`id`),
+ UNIQUE KEY `id_UNIQUE` (`id`)
+) ENGINE=InnoDB AUTO_INCREMENT=10 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci COMMENT='Les différents rôles des utilisateurs' `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `x_users_titres`
+--
+
+DROP TABLE IF EXISTS `x_users_titres`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `x_users_titres` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `libelle` varchar(45) DEFAULT NULL,
+ `chk_active` tinyint(1) unsigned DEFAULT 1,
+ PRIMARY KEY (`id`),
+ UNIQUE KEY `id_UNIQUE` (`id`)
+) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci COMMENT='Les différents titres des utilisateurs' `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `x_villes`
+--
+
+DROP TABLE IF EXISTS `x_villes`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `x_villes` (
+ `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `fk_departement` int(10) unsigned DEFAULT 1,
+ `libelle` varchar(65) DEFAULT NULL,
+ `code_postal` varchar(5) DEFAULT NULL,
+ `code_insee` varchar(5) DEFAULT NULL,
+ `chk_active` tinyint(1) unsigned DEFAULT 1,
+ PRIMARY KEY (`id`),
+ UNIQUE KEY `id_UNIQUE` (`id`),
+ KEY `x_villes_ibfk_1` (`fk_departement`),
+ CONSTRAINT `x_villes_ibfk_1` FOREIGN KEY (`fk_departement`) REFERENCES `x_departements` (`id`) ON UPDATE CASCADE
+) ENGINE=InnoDB AUTO_INCREMENT=38950 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `z_sessions`
+--
+
+DROP TABLE IF EXISTS `z_sessions`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8mb4 */;
+CREATE TABLE `z_sessions` (
+ `sid` text NOT NULL,
+ `fk_user` int(11) NOT NULL,
+ `role` varchar(10) DEFAULT NULL,
+ `date_modified` timestamp NOT NULL DEFAULT current_timestamp() ON UPDATE current_timestamp(),
+ `ip` varchar(50) NOT NULL,
+ `browser` varchar(150) NOT NULL,
+ `data` mediumtext DEFAULT NULL
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Final view structure for view `chat_rooms_with_last_message`
+--
+
+/*!50001 DROP VIEW IF EXISTS `chat_rooms_with_last_message`*/;
+/*!50001 SET @saved_cs_client = @@character_set_client */;
+/*!50001 SET @saved_cs_results = @@character_set_results */;
+/*!50001 SET @saved_col_connection = @@collation_connection */;
+/*!50001 SET character_set_client = utf8mb4 */;
+/*!50001 SET character_set_results = utf8mb4 */;
+/*!50001 SET collation_connection = utf8mb4_general_ci */;
+/*!50001 CREATE ALGORITHM=UNDEFINED */
+/*!50013 DEFINER=`admin`@`13.23.33.1` SQL SECURITY DEFINER */
+/*!50001 VIEW `chat_rooms_with_last_message` AS select `r`.`id` AS `id`,`r`.`title` AS `title`,`r`.`type` AS `type`,`r`.`created_at` AS `created_at`,`r`.`created_by` AS `created_by`,`r`.`updated_at` AS `updated_at`,`r`.`is_active` AS `is_active`,`m`.`content` AS `last_message_content`,`m`.`sender_id` AS `last_message_sender`,`m`.`sent_at` AS `last_message_at`,`u`.`encrypted_name` AS `last_message_sender_name` from ((`chat_rooms` `r` left join (select `m1`.`id` AS `id`,`m1`.`room_id` AS `room_id`,`m1`.`content` AS `content`,`m1`.`sender_id` AS `sender_id`,`m1`.`sent_at` AS `sent_at`,`m1`.`edited_at` AS `edited_at`,`m1`.`is_deleted` AS `is_deleted` from (`chat_messages` `m1` join (select `chat_messages`.`room_id` AS `room_id`,max(`chat_messages`.`sent_at`) AS `max_sent_at` from `chat_messages` where `chat_messages`.`is_deleted` = 0 group by `chat_messages`.`room_id`) `m2` on(`m1`.`room_id` = `m2`.`room_id` and `m1`.`sent_at` = `m2`.`max_sent_at`))) `m` on(`r`.`id` = `m`.`room_id`)) left join `users` `u` on(`m`.`sender_id` = `u`.`id`)) where `r`.`is_active` = 1 */;
+/*!50001 SET character_set_client = @saved_cs_client */;
+/*!50001 SET character_set_results = @saved_cs_results */;
+/*!50001 SET collation_connection = @saved_col_connection */;
+
+--
+-- Final view structure for view `v_stripe_entite_stats`
+--
+
+/*!50001 DROP VIEW IF EXISTS `v_stripe_entite_stats`*/;
+/*!50001 SET @saved_cs_client = @@character_set_client */;
+/*!50001 SET @saved_cs_results = @@character_set_results */;
+/*!50001 SET @saved_col_connection = @@collation_connection */;
+/*!50001 SET character_set_client = utf8mb3 */;
+/*!50001 SET character_set_results = utf8mb3 */;
+/*!50001 SET collation_connection = utf8mb3_general_ci */;
+/*!50001 CREATE ALGORITHM=UNDEFINED */
+/*!50013 DEFINER=`root`@`localhost` SQL SECURITY DEFINER */
+/*!50001 VIEW `v_stripe_entite_stats` AS select `e`.`id` AS `entite_id`,`e`.`encrypted_name` AS `entite_name`,`sa`.`stripe_account_id` AS `stripe_account_id`,`sa`.`charges_enabled` AS `charges_enabled`,`sa`.`payouts_enabled` AS `payouts_enabled`,count(distinct `p`.`id`) AS `total_passages`,count(distinct case when `p`.`stripe_payment_id` is not null then `p`.`id` end) AS `passages_stripe`,sum(case when `p`.`stripe_payment_id` is not null then `p`.`montant` else 0 end) AS `revenue_stripe`,sum(`p`.`montant`) AS `revenue_total` from (((`entites` `e` left join `stripe_accounts` `sa` on(`e`.`id` = `sa`.`fk_entite`)) left join `operations` `o` on(`e`.`id` = `o`.`fk_entite`)) left join `ope_pass` `p` on(`o`.`id` = `p`.`fk_operation`)) group by `e`.`id`,`e`.`encrypted_name`,`sa`.`stripe_account_id` */;
+/*!50001 SET character_set_client = @saved_cs_client */;
+/*!50001 SET character_set_results = @saved_cs_results */;
+/*!50001 SET collation_connection = @saved_col_connection */;
+
+--
+-- Final view structure for view `v_stripe_payment_stats`
+--
+
+/*!50001 DROP VIEW IF EXISTS `v_stripe_payment_stats`*/;
+/*!50001 SET @saved_cs_client = @@character_set_client */;
+/*!50001 SET @saved_cs_results = @@character_set_results */;
+/*!50001 SET @saved_col_connection = @@collation_connection */;
+/*!50001 SET character_set_client = utf8mb3 */;
+/*!50001 SET character_set_results = utf8mb3 */;
+/*!50001 SET collation_connection = utf8mb3_general_ci */;
+/*!50001 CREATE ALGORITHM=UNDEFINED */
+/*!50013 DEFINER=`root`@`localhost` SQL SECURITY DEFINER */
+/*!50001 VIEW `v_stripe_payment_stats` AS select `o`.`fk_entite` AS `fk_entite`,`e`.`encrypted_name` AS `entite_name`,`p`.`fk_user` AS `fk_user`,concat(`u`.`first_name`,' ',`u`.`sect_name`) AS `user_name`,count(distinct `p`.`id`) AS `total_ventes`,count(distinct case when `p`.`stripe_payment_id` is not null then `p`.`id` end) AS `ventes_stripe`,sum(case when `p`.`stripe_payment_id` is not null then `p`.`montant` else 0 end) AS `montant_stripe`,sum(case when `p`.`stripe_payment_id` is null then `p`.`montant` else 0 end) AS `montant_autres`,cast(`p`.`created_at` as date) AS `date_vente` from (((`ope_pass` `p` left join `operations` `o` on(`p`.`fk_operation` = `o`.`id`)) left join `entites` `e` on(`o`.`fk_entite` = `e`.`id`)) left join `users` `u` on(`p`.`fk_user` = `u`.`id`)) where `p`.`fk_type` = 2 group by `o`.`fk_entite`,`p`.`fk_user`,cast(`p`.`created_at` as date) */;
+/*!50001 SET character_set_client = @saved_cs_client */;
+/*!50001 SET character_set_results = @saved_cs_results */;
+/*!50001 SET collation_connection = @saved_col_connection */;
+/*!40103 SET TIME_ZONE=@OLD_TIME_ZONE */;
+
+/*!40101 SET SQL_MODE=@OLD_SQL_MODE */;
+/*!40014 SET FOREIGN_KEY_CHECKS=@OLD_FOREIGN_KEY_CHECKS */;
+/*!40014 SET UNIQUE_CHECKS=@OLD_UNIQUE_CHECKS */;
+/*!40101 SET CHARACTER_SET_CLIENT=@OLD_CHARACTER_SET_CLIENT */;
+/*!40101 SET CHARACTER_SET_RESULTS=@OLD_CHARACTER_SET_RESULTS */;
+/*!40101 SET COLLATION_CONNECTION=@OLD_COLLATION_CONNECTION */;
+/*M!100616 SET NOTE_VERBOSITY=@OLD_NOTE_VERBOSITY */;
+
+-- Dump completed on 2025-10-06 16:59:27
diff --git a/api/scripts/migration2/geosector-structure.sql b/api/scripts/migration2/geosector-structure.sql
new file mode 100644
index 00000000..3cb56f08
--- /dev/null
+++ b/api/scripts/migration2/geosector-structure.sql
@@ -0,0 +1,1088 @@
+/*M!999999\- enable the sandbox mode */
+-- MariaDB dump 10.19 Distrib 10.11.9-MariaDB, for debian-linux-gnu (x86_64)
+--
+-- Host: localhost Database: geosector
+-- ------------------------------------------------------
+-- Server version 10.11.9-MariaDB-deb12
+
+/*!40101 SET @OLD_CHARACTER_SET_CLIENT=@@CHARACTER_SET_CLIENT */;
+/*!40101 SET @OLD_CHARACTER_SET_RESULTS=@@CHARACTER_SET_RESULTS */;
+/*!40101 SET @OLD_COLLATION_CONNECTION=@@COLLATION_CONNECTION */;
+/*!40101 SET NAMES utf8mb4 */;
+/*!40103 SET @OLD_TIME_ZONE=@@TIME_ZONE */;
+/*!40103 SET TIME_ZONE='+00:00' */;
+/*!40014 SET @OLD_UNIQUE_CHECKS=@@UNIQUE_CHECKS, UNIQUE_CHECKS=0 */;
+/*!40014 SET @OLD_FOREIGN_KEY_CHECKS=@@FOREIGN_KEY_CHECKS, FOREIGN_KEY_CHECKS=0 */;
+/*!40101 SET @OLD_SQL_MODE=@@SQL_MODE, SQL_MODE='NO_AUTO_VALUE_ON_ZERO' */;
+/*!40111 SET @OLD_SQL_NOTES=@@SQL_NOTES, SQL_NOTES=0 */;
+
+--
+-- Table structure for table `articles`
+--
+
+DROP TABLE IF EXISTS `articles`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `articles` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `fk_page` int(11) DEFAULT 0,
+ `row` int(11) DEFAULT 1,
+ `col` int(11) DEFAULT 1,
+ `libelle` varchar(45) DEFAULT NULL,
+ `ordre` int(11) DEFAULT 0,
+ `titre` varchar(200) DEFAULT NULL,
+ `chapeau` text DEFAULT NULL,
+ `texte` text DEFAULT NULL,
+ `icone` varchar(45) NOT NULL DEFAULT '',
+ `icone_position` int(11) DEFAULT 1,
+ `icone_size` int(11) DEFAULT 1,
+ `active` tinyint(1) DEFAULT 1,
+ PRIMARY KEY (`rowid`),
+ UNIQUE KEY `rowid_UNIQUE` (`rowid`)
+) ENGINE=InnoDB AUTO_INCREMENT=45 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `articles_pages`
+--
+
+DROP TABLE IF EXISTS `articles_pages`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `articles_pages` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `ordre` int(11) NOT NULL DEFAULT 0,
+ `libelle` varchar(75) NOT NULL DEFAULT '',
+ `script` varchar(45) NOT NULL DEFAULT '',
+ `admin` tinyint(1) DEFAULT 0,
+ `active` tinyint(1) DEFAULT 1,
+ PRIMARY KEY (`rowid`),
+ UNIQUE KEY `rowid_UNIQUE` (`rowid`)
+) ENGINE=InnoDB AUTO_INCREMENT=8 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `blog_articles`
+--
+
+DROP TABLE IF EXISTS `blog_articles`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `blog_articles` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `fk_entite` int(11) DEFAULT NULL,
+ `fk_page` int(11) DEFAULT NULL,
+ `lst_destinataires` text NOT NULL,
+ `libelle` varchar(75) NOT NULL,
+ `texte` text DEFAULT NULL,
+ `publie` tinyint(1) DEFAULT 1,
+ `date_publication` date DEFAULT NULL,
+ `date_fin_publication` date DEFAULT NULL,
+ `active` tinyint(1) DEFAULT 1,
+ PRIMARY KEY (`rowid`),
+ UNIQUE KEY `rowid_UNIQUE` (`rowid`)
+) ENGINE=InnoDB AUTO_INCREMENT=482 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `blog_pages`
+--
+
+DROP TABLE IF EXISTS `blog_pages`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `blog_pages` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `libelle` varchar(45) DEFAULT NULL,
+ `site` tinyint(1) DEFAULT 0,
+ `admin` tinyint(1) DEFAULT 0,
+ `script` varchar(20) DEFAULT 'accueil',
+ `active` tinyint(1) DEFAULT 1,
+ PRIMARY KEY (`rowid`),
+ UNIQUE KEY `rowid_UNIQUE` (`rowid`)
+) ENGINE=InnoDB AUTO_INCREMENT=3 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `email_counter`
+--
+
+DROP TABLE IF EXISTS `email_counter`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `email_counter` (
+ `id` int(11) NOT NULL DEFAULT 1,
+ `hour_start` timestamp NULL DEFAULT NULL,
+ `count` int(11) DEFAULT 0,
+ PRIMARY KEY (`id`)
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_general_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `email_queue`
+--
+
+DROP TABLE IF EXISTS `email_queue`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `email_queue` (
+ `id` int(11) NOT NULL AUTO_INCREMENT,
+ `rowid` int(11) NOT NULL DEFAULT 0,
+ `to_email` varchar(255) DEFAULT NULL,
+ `subject` varchar(255) DEFAULT NULL,
+ `body` text DEFAULT NULL,
+ `headers` text DEFAULT NULL,
+ `created_at` timestamp NULL DEFAULT current_timestamp(),
+ `status` enum('pending','sent','failed') DEFAULT 'pending',
+ `attempts` int(11) DEFAULT 0,
+ PRIMARY KEY (`id`)
+) ENGINE=InnoDB AUTO_INCREMENT=41279 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_general_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `medias`
+--
+
+DROP TABLE IF EXISTS `medias`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `medias` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `dir0` varchar(75) NOT NULL DEFAULT '',
+ `dir1` varchar(150) NOT NULL DEFAULT '',
+ `dir2` varchar(45) NOT NULL DEFAULT '',
+ `support` varchar(45) NOT NULL DEFAULT '',
+ `support_rowid` int(11) NOT NULL DEFAULT 0,
+ `fichier` varchar(250) NOT NULL DEFAULT '',
+ `type_fichier` varchar(5) NOT NULL DEFAULT 'pdf',
+ `description` varchar(100) NOT NULL DEFAULT '',
+ `position` char(1) NOT NULL DEFAULT 'd',
+ `hauteur` int(11) NOT NULL DEFAULT 0,
+ `largeur` int(11) NOT NULL DEFAULT 0,
+ `niveaugris` tinyint(1) NOT NULL DEFAULT 0,
+ `date_creat` datetime DEFAULT NULL,
+ `fk_user_creat` int(11) NOT NULL DEFAULT 0,
+ `date_modif` datetime DEFAULT NULL,
+ `fk_user_modif` int(11) NOT NULL DEFAULT 0,
+ PRIMARY KEY (`rowid`),
+ UNIQUE KEY `rowid_UNIQUE` (`rowid`)
+) ENGINE=InnoDB AUTO_INCREMENT=339 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `ope_pass`
+--
+
+DROP TABLE IF EXISTS `ope_pass`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `ope_pass` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `fk_operation` int(11) NOT NULL DEFAULT 0,
+ `fk_sector` int(11) DEFAULT 0,
+ `fk_user` int(11) NOT NULL DEFAULT 0,
+ `fk_adresse` varchar(25) DEFAULT '',
+ `date_eve` datetime DEFAULT NULL,
+ `fk_type` int(11) DEFAULT 0,
+ `numero` varchar(10) NOT NULL DEFAULT '',
+ `rue` varchar(75) NOT NULL DEFAULT '',
+ `rue_bis` varchar(1) NOT NULL DEFAULT '',
+ `ville` varchar(75) NOT NULL DEFAULT '',
+ `lieudit` varchar(75) DEFAULT '',
+ `fk_habitat` int(11) DEFAULT 1,
+ `appt` varchar(5) DEFAULT NULL,
+ `niveau` varchar(5) DEFAULT NULL,
+ `chk_habitat_vide` tinyint(1) NOT NULL DEFAULT 0,
+ `gps_lat` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL DEFAULT '',
+ `gps_lng` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL DEFAULT '',
+ `libelle` varchar(45) NOT NULL DEFAULT '',
+ `montant` decimal(7,2) NOT NULL DEFAULT 0.00,
+ `fk_type_reglement` int(11) DEFAULT 1,
+ `remarque` text DEFAULT NULL,
+ `recu` varchar(50) DEFAULT NULL,
+ `email` varchar(75) DEFAULT '',
+ `email_erreur` varchar(30) DEFAULT '',
+ `chk_email_sent` tinyint(1) NOT NULL DEFAULT 0,
+ `phone` varchar(15) NOT NULL DEFAULT '',
+ `docremis` tinyint(1) DEFAULT 0,
+ `date_repasser` datetime DEFAULT NULL,
+ `nb_passages` int(11) DEFAULT 1,
+ `lot_nb_passages` int(11) DEFAULT 1,
+ `chk_gps_maj` tinyint(1) DEFAULT 0,
+ `chk_map_create` tinyint(1) DEFAULT 0,
+ `chk_mobile` tinyint(1) DEFAULT 0,
+ `chk_synchro` tinyint(1) DEFAULT 1,
+ `chk_api_adresse` tinyint(1) DEFAULT 0,
+ `chk_maj_adresse` tinyint(1) DEFAULT 0,
+ `anomalie` tinyint(1) DEFAULT 0,
+ `date_creat` datetime DEFAULT NULL,
+ `fk_user_creat` int(11) DEFAULT NULL,
+ `date_modif` datetime DEFAULT NULL,
+ `fk_user_modif` int(11) DEFAULT NULL,
+ `active` tinyint(1) NOT NULL DEFAULT 1,
+ PRIMARY KEY (`rowid`),
+ KEY `fk_operation` (`fk_operation`),
+ KEY `fk_sector` (`fk_sector`),
+ KEY `fk_user` (`fk_user`),
+ KEY `fk_type` (`fk_type`),
+ KEY `fk_type_reglement` (`fk_type_reglement`),
+ KEY `email` (`email`)
+) ENGINE=InnoDB AUTO_INCREMENT=11583369 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `ope_pass_histo`
+--
+
+DROP TABLE IF EXISTS `ope_pass_histo`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `ope_pass_histo` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `fk_pass` int(11) DEFAULT NULL,
+ `fk_user` int(11) DEFAULT NULL,
+ `date_histo` datetime DEFAULT NULL,
+ `sujet` varchar(50) DEFAULT NULL,
+ `remarque` varchar(250) DEFAULT NULL,
+ PRIMARY KEY (`rowid`),
+ KEY `ope_pass_histo_fk_pass_IDX` (`fk_pass`) USING BTREE,
+ KEY `ope_pass_histo_date_histo_IDX` (`date_histo`) USING BTREE
+) ENGINE=InnoDB AUTO_INCREMENT=45125 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `ope_pass_recus`
+--
+
+DROP TABLE IF EXISTS `ope_pass_recus`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `ope_pass_recus` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `fk_pass` int(11) DEFAULT NULL,
+ `chemin` varchar(100) DEFAULT NULL,
+ `nom_recu` varchar(50) DEFAULT NULL,
+ `date_recu` datetime DEFAULT NULL,
+ `date_creat_recu` datetime DEFAULT NULL,
+ `date_sent_recu` datetime DEFAULT NULL,
+ PRIMARY KEY (`rowid`),
+ UNIQUE KEY `rowid_UNIQUE` (`rowid`),
+ KEY `ope_pass_recus_fk_pass_IDX` (`fk_pass`) USING BTREE
+) ENGINE=InnoDB AUTO_INCREMENT=140967 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_general_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `ope_users`
+--
+
+DROP TABLE IF EXISTS `ope_users`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `ope_users` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `fk_operation` int(11) NOT NULL DEFAULT 0,
+ `fk_user` int(11) NOT NULL DEFAULT 0,
+ `date_creat` datetime DEFAULT NULL,
+ `fk_user_creat` int(11) DEFAULT NULL,
+ `date_modif` datetime DEFAULT NULL,
+ `fk_user_modif` int(11) DEFAULT NULL,
+ `active` tinyint(1) NOT NULL DEFAULT 1,
+ PRIMARY KEY (`rowid`),
+ UNIQUE KEY `rowid_UNIQUE` (`rowid`)
+) ENGINE=InnoDB AUTO_INCREMENT=254796 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `ope_users_sectors`
+--
+
+DROP TABLE IF EXISTS `ope_users_sectors`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `ope_users_sectors` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `fk_operation` int(11) DEFAULT NULL,
+ `fk_user` int(11) DEFAULT NULL,
+ `fk_sector` int(11) DEFAULT NULL,
+ `active` tinyint(1) DEFAULT 1,
+ PRIMARY KEY (`rowid`),
+ UNIQUE KEY `rowid` (`rowid`),
+ KEY `fk_operation` (`fk_operation`),
+ KEY `fk_user` (`fk_user`),
+ KEY `fk_sector` (`fk_sector`)
+) ENGINE=InnoDB AUTO_INCREMENT=203919 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `ope_users_suivis`
+--
+
+DROP TABLE IF EXISTS `ope_users_suivis`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `ope_users_suivis` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `fk_operation` int(11) DEFAULT NULL,
+ `fk_user` int(11) DEFAULT NULL,
+ `date_suivi` datetime DEFAULT NULL,
+ `latitude` varchar(20) DEFAULT NULL,
+ `longitude` varchar(20) DEFAULT NULL,
+ `vitesse` varchar(20) DEFAULT NULL,
+ PRIMARY KEY (`rowid`),
+ UNIQUE KEY `rowid_UNIQUE` (`rowid`)
+) ENGINE=InnoDB AUTO_INCREMENT=2820230 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `operations`
+--
+
+DROP TABLE IF EXISTS `operations`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `operations` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `fk_entite` int(11) DEFAULT 1,
+ `libelle` varchar(75) NOT NULL DEFAULT '',
+ `date_deb` date DEFAULT NULL,
+ `date_fin` date DEFAULT NULL,
+ `chk_api_adresse` tinyint(1) DEFAULT 0,
+ `chk_distinct_sectors` tinyint(1) DEFAULT 0,
+ `date_creat` datetime DEFAULT NULL,
+ `fk_user_creat` int(11) DEFAULT NULL,
+ `date_modif` datetime DEFAULT NULL,
+ `fk_user_modif` int(11) DEFAULT NULL,
+ `active` tinyint(1) DEFAULT 1,
+ PRIMARY KEY (`rowid`),
+ KEY `fk_entite` (`fk_entite`),
+ KEY `date_deb` (`date_deb`)
+) ENGINE=InnoDB AUTO_INCREMENT=3507 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `operations_docs`
+--
+
+DROP TABLE IF EXISTS `operations_docs`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `operations_docs` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `fk_operation` int(11) NOT NULL DEFAULT 0,
+ `libelle` varchar(75) NOT NULL DEFAULT '',
+ `comment` text NOT NULL,
+ `active` tinyint(1) NOT NULL DEFAULT 1,
+ PRIMARY KEY (`rowid`)
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `operations_eve_docs`
+--
+
+DROP TABLE IF EXISTS `operations_eve_docs`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `operations_eve_docs` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `fk_evenement` int(11) NOT NULL DEFAULT 0,
+ `fk_document` int(11) NOT NULL DEFAULT 0,
+ `date_creat` datetime DEFAULT NULL,
+ `fk_user_creat` int(11) DEFAULT NULL,
+ `date_modif` datetime DEFAULT NULL,
+ `fk_user_modif` int(11) DEFAULT NULL,
+ `active` tinyint(1) DEFAULT 1,
+ PRIMARY KEY (`rowid`)
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `params`
+--
+
+DROP TABLE IF EXISTS `params`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `params` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `libelle` varchar(35) NOT NULL DEFAULT '',
+ `valeur` varchar(255) NOT NULL DEFAULT '',
+ `aide` varchar(150) NOT NULL DEFAULT '',
+ PRIMARY KEY (`rowid`)
+) ENGINE=InnoDB AUTO_INCREMENT=2 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `sectors`
+--
+
+DROP TABLE IF EXISTS `sectors`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `sectors` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `libelle` varchar(75) DEFAULT '',
+ `sector` text DEFAULT NULL,
+ `color` varchar(7) DEFAULT '#4B77BE',
+ `date_creat` datetime DEFAULT NULL,
+ `fk_user_creat` int(11) DEFAULT NULL,
+ `date_modif` datetime DEFAULT NULL,
+ `fk_user_modif` int(11) DEFAULT NULL,
+ `active` tinyint(1) DEFAULT 1,
+ PRIMARY KEY (`rowid`),
+ UNIQUE KEY `rowid_UNIQUE` (`rowid`)
+) ENGINE=InnoDB AUTO_INCREMENT=39675 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `sectors_adresses`
+--
+
+DROP TABLE IF EXISTS `sectors_adresses`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `sectors_adresses` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `fk_adresse` varchar(25) DEFAULT NULL,
+ `fk_sector` int(11) DEFAULT NULL,
+ `numero` varchar(5) DEFAULT NULL,
+ `rue_bis` varchar(5) DEFAULT NULL,
+ `rue` varchar(60) DEFAULT NULL,
+ `cp` varchar(5) DEFAULT NULL,
+ `ville` varchar(60) DEFAULT NULL,
+ `gps_lat` varchar(10) DEFAULT NULL,
+ `gps_lng` varchar(10) DEFAULT NULL,
+ PRIMARY KEY (`rowid`),
+ KEY `sectors_adresses_fk_sector_index` (`fk_sector`),
+ KEY `sectors_adresses_numero_index` (`numero`),
+ KEY `sectors_adresses_rue_index` (`rue`),
+ KEY `sectors_adresses_ville_index` (`ville`)
+) ENGINE=InnoDB AUTO_INCREMENT=29621480 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_general_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `sectors_streets`
+--
+
+DROP TABLE IF EXISTS `sectors_streets`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `sectors_streets` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `fk_sector` int(11) DEFAULT NULL,
+ `fk_adresse` varchar(25) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci DEFAULT '',
+ `osm_id` int(11) DEFAULT NULL,
+ `osm_lat` varchar(10) DEFAULT NULL,
+ `osm_lng` varchar(10) DEFAULT NULL,
+ `osm_name` varchar(50) DEFAULT NULL,
+ `osm_street` varchar(50) DEFAULT NULL,
+ `osm_number` varchar(10) DEFAULT NULL,
+ `osm_city` varchar(50) DEFAULT NULL,
+ `osm_date_creat` timestamp NULL DEFAULT NULL,
+ `date_creat` datetime DEFAULT NULL,
+ PRIMARY KEY (`rowid`),
+ UNIQUE KEY `rowid_UNIQUE` (`rowid`),
+ KEY `fk_sector` (`fk_sector`),
+ KEY `osm_lat` (`osm_lat`),
+ KEY `osm_lng` (`osm_lng`),
+ KEY `osm_name` (`osm_name`),
+ KEY `osm_city` (`osm_city`),
+ KEY `osm_street` (`osm_street`)
+) ENGINE=InnoDB AUTO_INCREMENT=47868262 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `users`
+--
+
+DROP TABLE IF EXISTS `users`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `users` (
+ `rowid` int(10) unsigned NOT NULL AUTO_INCREMENT,
+ `fk_entite` int(11) DEFAULT 1,
+ `fk_titre` int(11) DEFAULT 1,
+ `num_adherent` int(11) NOT NULL DEFAULT 0,
+ `libelle` varchar(91) DEFAULT NULL,
+ `libelle_naissance` varchar(90) NOT NULL DEFAULT '',
+ `prenom` varchar(45) DEFAULT NULL,
+ `nom_tournee` varchar(60) DEFAULT '',
+ `username` varchar(50) DEFAULT '',
+ `userpass` varchar(60) DEFAULT NULL,
+ `userpswd` varchar(60) DEFAULT NULL,
+ `josh` tinyint(1) NOT NULL DEFAULT 0,
+ `telephone` varchar(15) DEFAULT NULL,
+ `mobile` varchar(15) DEFAULT NULL,
+ `email` varchar(100) DEFAULT '',
+ `email_secondaire` varchar(100) NOT NULL DEFAULT '',
+ `fk_role` int(11) DEFAULT 1,
+ `infos` varchar(200) NOT NULL DEFAULT '',
+ `ltt` varchar(10) DEFAULT '48.08',
+ `lng` varchar(10) DEFAULT '-1.68',
+ `sector` text DEFAULT NULL,
+ `alert_email` tinyint(1) DEFAULT 1,
+ `chk_suivi` tinyint(1) DEFAULT 0,
+ `date_naissance` date DEFAULT NULL,
+ `dept_naissance` varchar(2) NOT NULL DEFAULT '',
+ `commune_naissance` varchar(60) NOT NULL DEFAULT '',
+ `date_embauche` date DEFAULT NULL,
+ `anciennete` varchar(20) DEFAULT '-',
+ `fk_categorie` int(11) NOT NULL DEFAULT 0,
+ `fk_sous_categorie` int(11) NOT NULL DEFAULT 0,
+ `adresse_1` varchar(50) NOT NULL DEFAULT '',
+ `adresse_2` varchar(50) NOT NULL DEFAULT '',
+ `cp` varchar(5) NOT NULL DEFAULT '',
+ `ville` varchar(60) NOT NULL DEFAULT '',
+ `matricule` varchar(10) NOT NULL DEFAULT '',
+ `fk_grade` int(11) NOT NULL DEFAULT 0,
+ `chk_adherent_ud` tinyint(1) NOT NULL DEFAULT 0,
+ `chk_adherent_ur` tinyint(1) NOT NULL DEFAULT 0,
+ `chk_adherent_fns` tinyint(1) NOT NULL DEFAULT 0,
+ `chk_archive` tinyint(1) NOT NULL DEFAULT 0,
+ `chk_double_affectation` tinyint(1) NOT NULL DEFAULT 0,
+ `fk_user_creat` int(11) DEFAULT NULL,
+ `date_creat` datetime DEFAULT NULL,
+ `fk_user_modif` int(11) DEFAULT NULL,
+ `date_modif` datetime DEFAULT NULL,
+ `active` tinyint(1) DEFAULT 1,
+ PRIMARY KEY (`rowid`),
+ KEY `fk_entite` (`fk_entite`),
+ KEY `libelle` (`libelle`),
+ KEY `username` (`username`)
+) ENGINE=InnoDB AUTO_INCREMENT=10029982 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `users_entites`
+--
+
+DROP TABLE IF EXISTS `users_entites`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `users_entites` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `appname` varchar(20) NOT NULL DEFAULT 'geo_sector',
+ `http_host` varchar(255) NOT NULL DEFAULT 'admin.geosector.fr adm.geosector.fr mobile.goesector.fr mob.geosector.fr',
+ `libelle` varchar(45) DEFAULT '',
+ `adresse1` varchar(45) DEFAULT '',
+ `adresse2` varchar(45) DEFAULT '',
+ `cp` varchar(5) DEFAULT '',
+ `ville` varchar(45) DEFAULT '',
+ `fk_region` int(11) DEFAULT NULL,
+ `fk_type` int(11) DEFAULT 1,
+ `tva_intra` varchar(15) DEFAULT '',
+ `rcs` varchar(45) DEFAULT '',
+ `siret` varchar(17) DEFAULT NULL,
+ `ape` varchar(5) DEFAULT '',
+ `tel1` varchar(20) DEFAULT '',
+ `tel2` varchar(20) DEFAULT '',
+ `couleur` varchar(10) DEFAULT '',
+ `prefecture` varchar(45) DEFAULT '',
+ `fk_titre_gerant` int(11) DEFAULT 1,
+ `gerant_prenom` varchar(45) DEFAULT '',
+ `gerant_nom` varchar(45) DEFAULT '',
+ `email` varchar(45) DEFAULT '',
+ `gps_lat` varchar(10) NOT NULL DEFAULT '',
+ `gps_lng` varchar(10) NOT NULL DEFAULT '',
+ `site_url` varchar(45) DEFAULT '',
+ `gerant_signature` varchar(45) DEFAULT '',
+ `tampon_signature` varchar(45) DEFAULT '',
+ `banque_libelle` varchar(25) DEFAULT '',
+ `banque_adresse` varchar(45) DEFAULT '',
+ `banque_cp` varchar(5) DEFAULT '',
+ `banque_ville` varchar(40) DEFAULT '',
+ `iban` varchar(30) DEFAULT '',
+ `bic` varchar(15) DEFAULT '',
+ `genbase` varchar(45) NOT NULL,
+ `groupebase` varchar(45) NOT NULL,
+ `userbase` varchar(45) NOT NULL,
+ `passbase` varchar(45) NOT NULL,
+ `demo` tinyint(1) DEFAULT 0,
+ `lib_vert` varchar(25) DEFAULT 'Effectué',
+ `lib_verts` varchar(25) DEFAULT 'Effectués',
+ `lib_orange` varchar(25) DEFAULT 'A repasser',
+ `lib_oranges` varchar(25) DEFAULT 'A repasser',
+ `lib_rouge` varchar(25) DEFAULT 'Refusé',
+ `lib_rouges` varchar(25) DEFAULT 'Refusés',
+ `lib_bleu` varchar(25) DEFAULT 'Autre (Don)',
+ `lib_bleus` varchar(25) DEFAULT 'Autres (Dons)',
+ `icon_siege` varchar(15) DEFAULT 'fire',
+ `icon_siege_color` varchar(15) DEFAULT 'red',
+ `btn_width` varchar(3) DEFAULT '100',
+ `chk_mdp_manuel` tinyint(1) NOT NULL DEFAULT 1,
+ `chk_copie_mail_recu` tinyint(1) NOT NULL DEFAULT 0,
+ `chk_accept_sms` tinyint(1) NOT NULL DEFAULT 0,
+ `nbmembres` int(11) DEFAULT 0,
+ `nbconnex` int(11) DEFAULT 0,
+ `date_modif` datetime DEFAULT NULL,
+ `fk_user_modif` int(11) DEFAULT NULL,
+ `active` tinyint(1) DEFAULT 1,
+ PRIMARY KEY (`rowid`)
+) ENGINE=InnoDB AUTO_INCREMENT=1299 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `users_lastpos`
+--
+
+DROP TABLE IF EXISTS `users_lastpos`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `users_lastpos` (
+ `fk_user` int(11) NOT NULL,
+ `fk_operation` int(11) DEFAULT NULL,
+ `gps_lat` varchar(20) DEFAULT NULL,
+ `gps_lng` varchar(20) DEFAULT NULL,
+ `date_pos` datetime DEFAULT NULL
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `x_civilites`
+--
+
+DROP TABLE IF EXISTS `x_civilites`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `x_civilites` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `libelle` varchar(45) DEFAULT NULL,
+ `active` tinyint(1) DEFAULT 1,
+ PRIMARY KEY (`rowid`),
+ UNIQUE KEY `rowid_UNIQUE` (`rowid`)
+) ENGINE=InnoDB AUTO_INCREMENT=7 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `x_departements`
+--
+
+DROP TABLE IF EXISTS `x_departements`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `x_departements` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `code` varchar(3) DEFAULT NULL,
+ `fk_region` int(11) DEFAULT 1,
+ `libelle` varchar(45) DEFAULT NULL,
+ `active` tinyint(1) DEFAULT 1,
+ PRIMARY KEY (`rowid`),
+ UNIQUE KEY `rowid_UNIQUE` (`rowid`)
+) ENGINE=InnoDB AUTO_INCREMENT=105 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `x_devises`
+--
+
+DROP TABLE IF EXISTS `x_devises`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `x_devises` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `code` varchar(3) DEFAULT NULL,
+ `symbole` varchar(6) DEFAULT NULL,
+ `libelle` varchar(45) DEFAULT NULL,
+ `active` tinyint(1) DEFAULT 1,
+ PRIMARY KEY (`rowid`),
+ UNIQUE KEY `rowid_UNIQUE` (`rowid`)
+) ENGINE=InnoDB AUTO_INCREMENT=2 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `x_entites_types`
+--
+
+DROP TABLE IF EXISTS `x_entites_types`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `x_entites_types` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `libelle` varchar(45) DEFAULT NULL,
+ `active` tinyint(1) DEFAULT NULL,
+ PRIMARY KEY (`rowid`),
+ UNIQUE KEY `rowid_UNIQUE` (`rowid`)
+) ENGINE=InnoDB AUTO_INCREMENT=3 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `x_pays`
+--
+
+DROP TABLE IF EXISTS `x_pays`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `x_pays` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `code` varchar(3) DEFAULT NULL,
+ `fk_continent` int(11) DEFAULT NULL,
+ `fk_devise` int(11) DEFAULT 1,
+ `libelle` varchar(45) DEFAULT NULL,
+ `active` tinyint(1) DEFAULT 1,
+ PRIMARY KEY (`rowid`),
+ UNIQUE KEY `rowid_UNIQUE` (`rowid`)
+) ENGINE=InnoDB AUTO_INCREMENT=10 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci COMMENT='Table des pays avec leurs codes' `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `x_regions`
+--
+
+DROP TABLE IF EXISTS `x_regions`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `x_regions` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `fk_pays` int(11) DEFAULT 1,
+ `libelle` varchar(45) DEFAULT NULL,
+ `libelle_long` varchar(45) DEFAULT NULL,
+ `table_osm` varchar(45) DEFAULT NULL,
+ `departements` varchar(45) DEFAULT NULL,
+ `active` tinyint(1) DEFAULT 1,
+ PRIMARY KEY (`rowid`),
+ UNIQUE KEY `rowid_UNIQUE` (`rowid`)
+) ENGINE=InnoDB AUTO_INCREMENT=29 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `x_types_passages`
+--
+
+DROP TABLE IF EXISTS `x_types_passages`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `x_types_passages` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `libelle` varchar(10) DEFAULT NULL,
+ `color_button` varchar(15) DEFAULT NULL,
+ `color_mark` varchar(15) DEFAULT NULL,
+ `color_table` varchar(15) DEFAULT NULL,
+ `chk_active` tinyint(1) NOT NULL DEFAULT 1,
+ PRIMARY KEY (`rowid`)
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_general_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `x_types_reglements`
+--
+
+DROP TABLE IF EXISTS `x_types_reglements`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `x_types_reglements` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `libelle` varchar(45) DEFAULT NULL,
+ `active` tinyint(1) DEFAULT 1,
+ PRIMARY KEY (`rowid`)
+) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `x_users_categories`
+--
+
+DROP TABLE IF EXISTS `x_users_categories`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `x_users_categories` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `libelle` varchar(30) NOT NULL,
+ `active` tinyint(1) NOT NULL DEFAULT 1,
+ PRIMARY KEY (`rowid`),
+ KEY `x_users_categories__libelle` (`libelle`)
+) ENGINE=InnoDB AUTO_INCREMENT=10 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `x_users_grades`
+--
+
+DROP TABLE IF EXISTS `x_users_grades`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `x_users_grades` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `libelle` varchar(90) NOT NULL DEFAULT '',
+ `active` tinyint(1) NOT NULL DEFAULT 1,
+ PRIMARY KEY (`rowid`),
+ KEY `x_users_grades__libelle` (`libelle`)
+) ENGINE=InnoDB AUTO_INCREMENT=84 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `x_users_roles`
+--
+
+DROP TABLE IF EXISTS `x_users_roles`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `x_users_roles` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `libelle` varchar(45) DEFAULT NULL,
+ `active` tinyint(1) DEFAULT 1,
+ PRIMARY KEY (`rowid`),
+ UNIQUE KEY `rowid_UNIQUE` (`rowid`)
+) ENGINE=InnoDB AUTO_INCREMENT=10 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci COMMENT='Les différents rôles des utilisateurs' `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `x_users_sous_categories`
+--
+
+DROP TABLE IF EXISTS `x_users_sous_categories`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `x_users_sous_categories` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `fk_user_categorie` int(11) NOT NULL,
+ `libelle` varchar(40) NOT NULL DEFAULT '',
+ `active` tinyint(1) NOT NULL DEFAULT 1,
+ PRIMARY KEY (`rowid`),
+ KEY `x_users_sous_categories__libelle` (`libelle`),
+ KEY `x_users_sous_categories_fk_user_categorie_index` (`fk_user_categorie`)
+) ENGINE=InnoDB AUTO_INCREMENT=20 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `x_villes`
+--
+
+DROP TABLE IF EXISTS `x_villes`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `x_villes` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `fk_departement` int(11) DEFAULT 1,
+ `libelle` varchar(65) DEFAULT NULL,
+ `cp` varchar(5) DEFAULT NULL,
+ `code_insee` varchar(5) DEFAULT NULL,
+ `departement` varchar(65) DEFAULT NULL,
+ `active` tinyint(1) DEFAULT 1,
+ PRIMARY KEY (`rowid`),
+ UNIQUE KEY `rowid_UNIQUE` (`rowid`)
+) ENGINE=InnoDB AUTO_INCREMENT=38950 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `y_conf`
+--
+
+DROP TABLE IF EXISTS `y_conf`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `y_conf` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `admin` tinyint(1) NOT NULL DEFAULT 0,
+ `appenv` varchar(5) NOT NULL DEFAULT 'dev',
+ `apptitle` varchar(75) NOT NULL DEFAULT '',
+ `appversion` varchar(20) NOT NULL DEFAULT '0.5 du 15/01/2016',
+ `appscript` varchar(25) NOT NULL DEFAULT 'login',
+ `appicon` varchar(25) NOT NULL DEFAULT 'favicon.png',
+ `pathimg` varchar(45) NOT NULL DEFAULT '/files/img',
+ `pathupload` varchar(45) NOT NULL DEFAULT '/files/upload',
+ `brandgroupe` varchar(45) NOT NULL DEFAULT '',
+ `brandmulti` tinyint(1) DEFAULT 0,
+ `date_maintenance` datetime DEFAULT NULL,
+ `date_renouvellement` date DEFAULT NULL,
+ `piwikid` varchar(45) DEFAULT '0',
+ `googlid` varchar(45) DEFAULT '0',
+ `active` tinyint(1) DEFAULT 1,
+ PRIMARY KEY (`rowid`),
+ UNIQUE KEY `rowid_UNIQUE` (`rowid`)
+) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `y_menus`
+--
+
+DROP TABLE IF EXISTS `y_menus`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `y_menus` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `type_menu` varchar(3) NOT NULL DEFAULT 'mnu',
+ `admin` tinyint(4) DEFAULT 0,
+ `only_type_entite` varchar(45) DEFAULT '',
+ `only_fk_entite` varchar(45) DEFAULT '',
+ `only_fk_role` varchar(45) DEFAULT '',
+ `divider_before` tinyint(1) DEFAULT 0,
+ `ordre` tinyint(4) DEFAULT 0,
+ `fk_parent` int(11) DEFAULT 0,
+ `libelle` varchar(45) DEFAULT '',
+ `icone` varchar(45) DEFAULT '',
+ `color` varchar(35) DEFAULT '',
+ `back-color` varchar(35) DEFAULT '',
+ `title` varchar(75) DEFAULT '',
+ `script` varchar(45) NOT NULL,
+ `script_command` varchar(45) DEFAULT '',
+ `active` tinyint(1) DEFAULT 1,
+ PRIMARY KEY (`rowid`),
+ UNIQUE KEY `rowid_UNIQUE` (`rowid`),
+ KEY `type_menu` (`type_menu`),
+ KEY `script` (`script`),
+ KEY `ordre` (`ordre`),
+ KEY `fk_parent` (`fk_parent`)
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `y_modules`
+--
+
+DROP TABLE IF EXISTS `y_modules`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `y_modules` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `fk_parent` int(11) DEFAULT 0,
+ `ordre` int(11) DEFAULT 0,
+ `libelle` varchar(45) NOT NULL,
+ `tip` varchar(150) DEFAULT '',
+ `description` text DEFAULT NULL,
+ `script` varchar(20) DEFAULT '',
+ `couleur` varchar(7) DEFAULT '#bcbcbc',
+ `icone` varchar(30) DEFAULT NULL,
+ `taille_tuile` int(11) DEFAULT 70,
+ `admin` tinyint(1) DEFAULT 1,
+ `active` tinyint(1) DEFAULT 0,
+ PRIMARY KEY (`rowid`),
+ UNIQUE KEY `rowid_UNIQUE` (`rowid`)
+) ENGINE=InnoDB AUTO_INCREMENT=7 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `y_modules_rules`
+--
+
+DROP TABLE IF EXISTS `y_modules_rules`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `y_modules_rules` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `fk_module` int(11) DEFAULT 0,
+ `libelle` varchar(30) NOT NULL DEFAULT '',
+ `tip` varchar(250) NOT NULL DEFAULT '',
+ `val_default` varchar(20) NOT NULL DEFAULT '',
+ `ce_apa` varchar(20) DEFAULT '',
+ `ce_csfouest` varchar(20) DEFAULT '',
+ `ce_demo` varchar(20) DEFAULT '',
+ `ce_natixis` varchar(20) DEFAULT '',
+ `ce_purina` varchar(20) DEFAULT '',
+ `ce_tfn44` varchar(20) DEFAULT '',
+ `ce_webasto` varchar(20) DEFAULT '',
+ `active` tinyint(1) NOT NULL DEFAULT 1,
+ PRIMARY KEY (`rowid`),
+ UNIQUE KEY `y_modules_regles_rowid_uindex` (`rowid`)
+) ENGINE=InnoDB AUTO_INCREMENT=16 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `y_pages`
+--
+
+DROP TABLE IF EXISTS `y_pages`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `y_pages` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `libelle` varchar(45) DEFAULT NULL,
+ `titre` varchar(75) DEFAULT NULL,
+ `tooltip` varchar(45) DEFAULT NULL,
+ `description` varchar(200) DEFAULT NULL,
+ `keywords` varchar(200) DEFAULT NULL,
+ `script` varchar(45) DEFAULT NULL,
+ `enmaintenance` tinyint(1) DEFAULT 0,
+ `admin` tinyint(1) DEFAULT 0,
+ `mail` tinyint(1) DEFAULT 0,
+ `admtools` tinyint(1) DEFAULT 0,
+ `magazine` tinyint(1) DEFAULT 0,
+ `files` tinyint(1) DEFAULT 1,
+ `maps` tinyint(1) DEFAULT 0,
+ `editor` tinyint(1) DEFAULT 0,
+ `jqui` tinyint(1) DEFAULT 0,
+ `form` tinyint(1) DEFAULT 0,
+ `sidebar` tinyint(1) DEFAULT 0,
+ `chart` tinyint(1) DEFAULT 0,
+ `agenda` tinyint(1) DEFAULT 0,
+ `scheduler` tinyint(1) DEFAULT 0,
+ `osm` tinyint(1) DEFAULT 0,
+ `zz` tinyint(1) DEFAULT 0,
+ `maintenance` tinyint(1) DEFAULT 0,
+ `layout` varchar(45) DEFAULT 'default.php',
+ `active` tinyint(1) DEFAULT 1,
+ PRIMARY KEY (`rowid`),
+ UNIQUE KEY `rowid_UNIQUE` (`rowid`),
+ KEY `script` (`script`),
+ KEY `admin` (`admin`)
+) ENGINE=InnoDB AUTO_INCREMENT=40 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `z_logs`
+--
+
+DROP TABLE IF EXISTS `z_logs`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `z_logs` (
+ `date` datetime NOT NULL,
+ `ip` varchar(15) NOT NULL,
+ `host` varchar(50) NOT NULL,
+ `adrhost` varchar(50) NOT NULL,
+ `infos` varchar(200) DEFAULT '',
+ `fk_user` int(11) DEFAULT 0,
+ `page` varchar(200) NOT NULL,
+ `commentaire` text NOT NULL
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci COMMENT='Table des logs' `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `z_sessions`
+--
+
+DROP TABLE IF EXISTS `z_sessions`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `z_sessions` (
+ `sid` text NOT NULL,
+ `fk_user` int(11) NOT NULL,
+ `role` varchar(10) DEFAULT NULL,
+ `date_modified` timestamp NOT NULL DEFAULT current_timestamp() ON UPDATE current_timestamp(),
+ `ip` varchar(50) NOT NULL,
+ `browser` varchar(150) NOT NULL,
+ `data` mediumtext DEFAULT NULL
+) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+
+--
+-- Table structure for table `z_stats`
+--
+
+DROP TABLE IF EXISTS `z_stats`;
+/*!40101 SET @saved_cs_client = @@character_set_client */;
+/*!40101 SET character_set_client = utf8 */;
+CREATE TABLE `z_stats` (
+ `rowid` int(11) NOT NULL AUTO_INCREMENT,
+ `libelle` varchar(75) DEFAULT NULL,
+ `fk_user` int(11) DEFAULT NULL,
+ `date` datetime DEFAULT NULL,
+ `ip` varchar(15) DEFAULT NULL,
+ `browser` varchar(75) DEFAULT NULL,
+ `origine` varchar(45) DEFAULT NULL,
+ `status` varchar(10) DEFAULT NULL,
+ `active` tinyint(1) DEFAULT NULL,
+ PRIMARY KEY (`rowid`)
+) ENGINE=InnoDB AUTO_INCREMENT=215481 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci `PAGE_COMPRESSED`='ON';
+/*!40101 SET character_set_client = @saved_cs_client */;
+/*!40103 SET TIME_ZONE=@OLD_TIME_ZONE */;
+
+/*!40101 SET SQL_MODE=@OLD_SQL_MODE */;
+/*!40014 SET FOREIGN_KEY_CHECKS=@OLD_FOREIGN_KEY_CHECKS */;
+/*!40014 SET UNIQUE_CHECKS=@OLD_UNIQUE_CHECKS */;
+/*!40101 SET CHARACTER_SET_CLIENT=@OLD_CHARACTER_SET_CLIENT */;
+/*!40101 SET CHARACTER_SET_RESULTS=@OLD_CHARACTER_SET_RESULTS */;
+/*!40101 SET COLLATION_CONNECTION=@OLD_COLLATION_CONNECTION */;
+/*!40111 SET SQL_NOTES=@OLD_SQL_NOTES */;
+
+-- Dump completed on 2025-10-02 17:31:13
diff --git a/api/scripts/migration2/logs/.gitignore b/api/scripts/migration2/logs/.gitignore
new file mode 100644
index 00000000..397b4a76
--- /dev/null
+++ b/api/scripts/migration2/logs/.gitignore
@@ -0,0 +1 @@
+*.log
diff --git a/api/scripts/migration2/migrate_batch.sh b/api/scripts/migration2/migrate_batch.sh
new file mode 100755
index 00000000..64947fba
--- /dev/null
+++ b/api/scripts/migration2/migrate_batch.sh
@@ -0,0 +1,467 @@
+#!/bin/bash
+
+###############################################################################
+# Script de migration en batch des entités depuis geosector_20251008
+#
+# Usage: ./migrate_batch.sh [options]
+#
+# Options:
+# --start N Commencer à partir de l'entité N (défaut: 1)
+# --limit N Migrer seulement N entités (défaut: toutes)
+# --dry-run Simuler sans exécuter
+# --continue Continuer après une erreur (défaut: s'arrêter)
+# --interactive Mode interactif (défaut si aucune option)
+#
+# Exemple:
+# ./migrate_batch.sh --start 10 --limit 5
+# ./migrate_batch.sh --continue
+# ./migrate_batch.sh --interactive
+###############################################################################
+
+# Configuration
+SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
+JSON_FILE="${SCRIPT_DIR}/migrations_entites.json"
+LOG_DIR="/var/www/geosector/api/logs/migrations"
+MIGRATION_SCRIPT="${SCRIPT_DIR}/php/migrate_from_backup.php"
+SOURCE_DB="geosector_20251013_13"
+TARGET_DB="pra_geo"
+
+# Paramètres par défaut
+START_INDEX=1
+LIMIT=0
+DRY_RUN=0
+CONTINUE_ON_ERROR=0
+INTERACTIVE_MODE=0
+SPECIFIC_ENTITY_ID=""
+SPECIFIC_CP=""
+
+# Couleurs
+RED='\033[0;31m'
+GREEN='\033[0;32m'
+YELLOW='\033[1;33m'
+BLUE='\033[0;34m'
+CYAN='\033[0;36m'
+NC='\033[0m' # No Color
+
+# Sauvegarder le nombre d'arguments avant le parsing
+INITIAL_ARGS=$#
+
+# Parse des arguments
+while [[ $# -gt 0 ]]; do
+ case $1 in
+ --start)
+ START_INDEX="$2"
+ shift 2
+ ;;
+ --limit)
+ LIMIT="$2"
+ shift 2
+ ;;
+ --dry-run)
+ DRY_RUN=1
+ shift
+ ;;
+ --continue)
+ CONTINUE_ON_ERROR=1
+ shift
+ ;;
+ --interactive|-i)
+ INTERACTIVE_MODE=1
+ shift
+ ;;
+ --help)
+ grep "^#" "$0" | grep -v "^#!/bin/bash" | sed 's/^# //'
+ exit 0
+ ;;
+ *)
+ echo "Option inconnue: $1"
+ echo "Utilisez --help pour l'aide"
+ exit 1
+ ;;
+ esac
+done
+
+# Activer le mode interactif si aucun argument n'a été fourni
+if [ $INITIAL_ARGS -eq 0 ]; then
+ INTERACTIVE_MODE=1
+fi
+
+# Vérifications préalables
+if [ ! -f "$JSON_FILE" ]; then
+ echo -e "${RED}❌ Fichier JSON introuvable: $JSON_FILE${NC}"
+ exit 1
+fi
+
+if [ ! -f "$MIGRATION_SCRIPT" ]; then
+ echo -e "${RED}❌ Script de migration introuvable: $MIGRATION_SCRIPT${NC}"
+ exit 1
+fi
+
+# Créer le répertoire de logs
+mkdir -p "$LOG_DIR"
+
+# Fichiers de log
+BATCH_LOG="${LOG_DIR}/batch_$(date +%Y%m%d_%H%M%S).log"
+SUCCESS_LOG="${LOG_DIR}/success.log"
+ERROR_LOG="${LOG_DIR}/errors.log"
+
+# MODE INTERACTIF
+if [ $INTERACTIVE_MODE -eq 1 ]; then
+ echo ""
+ echo -e "${CYAN}═══════════════════════════════════════════════════════════${NC}"
+ echo -e "${CYAN} 🔧 Mode interactif - Migration d'entités GeoSector${NC}"
+ echo -e "${CYAN}═══════════════════════════════════════════════════════════${NC}"
+ echo ""
+
+ # Question 1: Migration globale ou ciblée ?
+ echo -e "${YELLOW}1️⃣ Type de migration :${NC}"
+ echo -e " ${CYAN}a)${NC} Migration globale (toutes les entités)"
+ echo -e " ${CYAN}b)${NC} Migration par lot (plage d'entités)"
+ echo -e " ${CYAN}c)${NC} Migration par code postal"
+ echo -e " ${CYAN}d)${NC} Migration d'une entité spécifique (ID)"
+ echo ""
+ echo -ne "${YELLOW}Votre choix (a/b/c/d) : ${NC}"
+ read -r MIGRATION_TYPE
+ echo ""
+
+ case $MIGRATION_TYPE in
+ a|A)
+ # Migration globale - garder les valeurs par défaut
+ START_INDEX=1
+ LIMIT=0
+ echo -e "${GREEN}✓${NC} Migration globale sélectionnée"
+ ;;
+ b|B)
+ # Migration par lot
+ echo -e "${YELLOW}2️⃣ Configuration du lot :${NC}"
+ echo -ne " Première entité (index, défaut=1) : "
+ read -r USER_START
+ if [ -n "$USER_START" ]; then
+ START_INDEX=$USER_START
+ fi
+
+ echo -ne " Limite (nombre d'entités, défaut=toutes) : "
+ read -r USER_LIMIT
+ if [ -n "$USER_LIMIT" ]; then
+ LIMIT=$USER_LIMIT
+ fi
+ echo ""
+ echo -e "${GREEN}✓${NC} Migration par lot : de l'index $START_INDEX, limite de $LIMIT entités"
+ ;;
+ c|C)
+ # Migration par code postal
+ echo -ne "${YELLOW}2️⃣ Code postal à migrer : ${NC}"
+ read -r SPECIFIC_CP
+ echo ""
+ if [ -z "$SPECIFIC_CP" ]; then
+ echo -e "${RED}❌ Code postal requis${NC}"
+ exit 1
+ fi
+ echo -e "${GREEN}✓${NC} Migration pour le code postal : $SPECIFIC_CP"
+ ;;
+ d|D)
+ # Migration d'une entité spécifique - bypass complet du JSON
+ echo -ne "${YELLOW}2️⃣ ID de l'entité à migrer : ${NC}"
+ read -r SPECIFIC_ENTITY_ID
+ echo ""
+ if [ -z "$SPECIFIC_ENTITY_ID" ]; then
+ echo -e "${RED}❌ ID d'entité requis${NC}"
+ exit 1
+ fi
+ echo -e "${GREEN}✓${NC} Migration de l'entité ID : $SPECIFIC_ENTITY_ID"
+ echo ""
+
+ # Demander si suppression des données de l'entité avant migration
+ echo -ne "${YELLOW}3️⃣ Supprimer les données existantes de cette entité dans la TARGET avant migration ? (y/N): ${NC}"
+ read -r DELETE_BEFORE
+ DELETE_FLAG=""
+ if [[ $DELETE_BEFORE =~ ^[Yy]$ ]]; then
+ echo -e "${GREEN}✓${NC} Les données seront supprimées avant migration"
+ DELETE_FLAG="--delete-before"
+ else
+ echo -e "${BLUE}ℹ${NC} Les données seront conservées (ON DUPLICATE KEY UPDATE)"
+ fi
+ echo ""
+
+ # Confirmer la migration
+ echo -ne "${YELLOW}⚠️ Confirmer la migration de l'entité #${SPECIFIC_ENTITY_ID} ? (y/N): ${NC}"
+ read -r CONFIRM
+ if [[ ! $CONFIRM =~ ^[Yy]$ ]]; then
+ echo -e "${RED}❌ Migration annulée${NC}"
+ exit 0
+ fi
+
+ # Exécuter directement la migration sans passer par le JSON
+ ENTITY_LOG="${LOG_DIR}/entity_${SPECIFIC_ENTITY_ID}_$(date +%Y%m%d_%H%M%S).log"
+
+ echo ""
+ echo -e "${BLUE}⏳ Migration de l'entité #${SPECIFIC_ENTITY_ID} en cours...${NC}"
+
+ php "$MIGRATION_SCRIPT" \
+ --source-db="$SOURCE_DB" \
+ --target-db="$TARGET_DB" \
+ --mode=entity \
+ --entity-id="$SPECIFIC_ENTITY_ID" \
+ --log="$ENTITY_LOG" \
+ $DELETE_FLAG
+
+ EXIT_CODE=$?
+
+ if [ $EXIT_CODE -eq 0 ]; then
+ echo -e "${GREEN}✅ Entité #${SPECIFIC_ENTITY_ID} migrée avec succès${NC}"
+ echo -e "${BLUE}📋 Log détaillé : $ENTITY_LOG${NC}"
+ else
+ echo -e "${RED}❌ Erreur lors de la migration de l'entité #${SPECIFIC_ENTITY_ID}${NC}"
+ echo -e "${RED}📋 Voir le log : $ENTITY_LOG${NC}"
+ exit 1
+ fi
+
+ exit 0
+ ;;
+ *)
+ echo -e "${RED}❌ Choix invalide${NC}"
+ exit 1
+ ;;
+ esac
+
+ echo ""
+fi
+
+# Fonctions utilitaires
+log() {
+ echo -e "$1" | tee -a "$BATCH_LOG"
+}
+
+log_success() {
+ echo "$1" >> "$SUCCESS_LOG"
+ log "${GREEN}✓${NC} $1"
+}
+
+log_error() {
+ echo "$1" >> "$ERROR_LOG"
+ log "${RED}✗${NC} $1"
+}
+
+# Extraire les entity_id du JSON (compatible sans jq)
+get_entity_ids() {
+ if [ -n "$SPECIFIC_ENTITY_ID" ]; then
+ # Entité spécifique par ID - chercher exactement "entity_id" : ID,
+ grep "\"entity_id\" : ${SPECIFIC_ENTITY_ID}," "$JSON_FILE" | sed 's/.*: \([0-9]*\).*/\1/'
+ elif [ -n "$SPECIFIC_CP" ]; then
+ # Entités par code postal
+ grep -B 2 "\"code_postal\" : \"$SPECIFIC_CP\"" "$JSON_FILE" | grep '"entity_id"' | sed 's/.*: \([0-9]*\).*/\1/'
+ else
+ # Toutes les entités
+ grep '"entity_id"' "$JSON_FILE" | sed 's/.*: \([0-9]*\).*/\1/'
+ fi
+}
+
+# Compter le nombre total d'entités
+TOTAL_ENTITIES=$(get_entity_ids | wc -l)
+
+# Vérifier si des entités ont été trouvées
+if [ $TOTAL_ENTITIES -eq 0 ]; then
+ if [ -n "$SPECIFIC_ENTITY_ID" ]; then
+ echo -e "${RED}❌ Entité #${SPECIFIC_ENTITY_ID} introuvable dans le fichier JSON${NC}"
+ elif [ -n "$SPECIFIC_CP" ]; then
+ echo -e "${RED}❌ Aucune entité trouvée pour le code postal ${SPECIFIC_CP}${NC}"
+ else
+ echo -e "${RED}❌ Aucune entité trouvée${NC}"
+ fi
+ exit 1
+fi
+
+# Calculer le nombre d'entités à migrer
+if [ $LIMIT -gt 0 ]; then
+ END_INDEX=$((START_INDEX + LIMIT - 1))
+ if [ $END_INDEX -gt $TOTAL_ENTITIES ]; then
+ END_INDEX=$TOTAL_ENTITIES
+ fi
+else
+ END_INDEX=$TOTAL_ENTITIES
+fi
+
+# Bannière de démarrage
+echo ""
+log "${BLUE}═══════════════════════════════════════════════════════════${NC}"
+log "${BLUE} Migration en batch des entités GeoSector${NC}"
+log "${BLUE}═══════════════════════════════════════════════════════════${NC}"
+log "📅 Date: $(date '+%Y-%m-%d %H:%M:%S')"
+log "📁 Source: $SOURCE_DB"
+log "📁 Cible: $TARGET_DB"
+
+# Afficher les informations selon le mode
+if [ -n "$SPECIFIC_ENTITY_ID" ]; then
+ log "🎯 Mode: Migration d'une entité spécifique"
+ log "📊 Entité ID: $SPECIFIC_ENTITY_ID"
+elif [ -n "$SPECIFIC_CP" ]; then
+ log "🎯 Mode: Migration par code postal"
+ log "📮 Code postal: $SPECIFIC_CP"
+ log "📊 Entités trouvées: $TOTAL_ENTITIES"
+else
+ TOTAL_AVAILABLE=$(grep '"entity_id"' "$JSON_FILE" | wc -l)
+ log "📊 Total entités disponibles: $TOTAL_AVAILABLE"
+ log "🎯 Entités à migrer: $START_INDEX à $END_INDEX"
+fi
+
+if [ $DRY_RUN -eq 1 ]; then
+ log "${YELLOW}🔍 Mode DRY-RUN (simulation)${NC}"
+fi
+log "${BLUE}═══════════════════════════════════════════════════════════${NC}"
+echo ""
+
+# Confirmation utilisateur
+if [ $DRY_RUN -eq 0 ]; then
+ echo -ne "${YELLOW}⚠️ Confirmer la migration ? (y/N): ${NC}"
+ read -r CONFIRM
+ if [[ ! $CONFIRM =~ ^[Yy]$ ]]; then
+ log "❌ Migration annulée par l'utilisateur"
+ exit 0
+ fi
+ echo ""
+fi
+
+# Compteurs
+SUCCESS_COUNT=0
+ERROR_COUNT=0
+SKIPPED_COUNT=0
+CURRENT_INDEX=0
+
+# Début de la migration
+START_TIME=$(date +%s)
+
+# Lire les entity_id et migrer
+get_entity_ids | while read -r ENTITY_ID; do
+ CURRENT_INDEX=$((CURRENT_INDEX + 1))
+
+ # Filtrer par index
+ if [ $CURRENT_INDEX -lt $START_INDEX ]; then
+ continue
+ fi
+
+ if [ $CURRENT_INDEX -gt $END_INDEX ]; then
+ break
+ fi
+
+ # Récupérer les détails de l'entité depuis le JSON (match exact avec la virgule)
+ ENTITY_INFO=$(grep -A 8 "\"entity_id\" : ${ENTITY_ID}," "$JSON_FILE")
+ ENTITY_NAME=$(echo "$ENTITY_INFO" | grep '"nom"' | sed 's/.*: "\(.*\)".*/\1/')
+ ENTITY_CP=$(echo "$ENTITY_INFO" | grep '"code_postal"' | sed 's/.*: "\(.*\)".*/\1/')
+ NB_USERS=$(echo "$ENTITY_INFO" | grep '"nb_users"' | sed 's/.*: \([0-9]*\).*/\1/')
+ NB_PASSAGES=$(echo "$ENTITY_INFO" | grep '"nb_passages"' | sed 's/.*: \([0-9]*\).*/\1/')
+
+ # Afficher la progression
+ PROGRESS=$((CURRENT_INDEX - START_INDEX + 1))
+ TOTAL=$((END_INDEX - START_INDEX + 1))
+ PERCENT=$((PROGRESS * 100 / TOTAL))
+
+ log ""
+ log "${BLUE}━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━${NC}"
+ log "${BLUE}[$PROGRESS/$TOTAL - ${PERCENT}%]${NC} Entité #${ENTITY_ID}: ${ENTITY_NAME} (${ENTITY_CP})"
+ log " 👥 Users: ${NB_USERS} | 📍 Passages: ${NB_PASSAGES}"
+
+ # Mode dry-run
+ if [ $DRY_RUN -eq 1 ]; then
+ log "${YELLOW} 🔍 [DRY-RUN] Simulation de la migration${NC}"
+ SKIPPED_COUNT=$((SKIPPED_COUNT + 1))
+ continue
+ fi
+
+ # Exécuter la migration
+ ENTITY_LOG="${LOG_DIR}/entity_${ENTITY_ID}_$(date +%Y%m%d_%H%M%S).log"
+
+ log " ⏳ Migration en cours..."
+ php "$MIGRATION_SCRIPT" \
+ --source-db="$SOURCE_DB" \
+ --target-db="$TARGET_DB" \
+ --mode=entity \
+ --entity-id="$ENTITY_ID" \
+ --log="$ENTITY_LOG" > /tmp/migration_output_$$.txt 2>&1
+
+ EXIT_CODE=$?
+
+ if [ $EXIT_CODE -eq 0 ]; then
+ # Succès
+ SUCCESS_COUNT=$((SUCCESS_COUNT + 1))
+ log_success "Entité #${ENTITY_ID} (${ENTITY_NAME}) migrée avec succès"
+
+ # Afficher un résumé du log avec détails
+ if [ -f "$ENTITY_LOG" ]; then
+ # Chercher la ligne avec les marqueurs #STATS#
+ STATS_LINE=$(grep "#STATS#" "$ENTITY_LOG" 2>/dev/null)
+
+ if [ -n "$STATS_LINE" ]; then
+ # Extraire chaque compteur
+ OPE=$(echo "$STATS_LINE" | grep -oE 'OPE:[0-9]+' | cut -d: -f2)
+ USERS=$(echo "$STATS_LINE" | grep -oE 'USER:[0-9]+' | cut -d: -f2)
+ SECTORS=$(echo "$STATS_LINE" | grep -oE 'SECTOR:[0-9]+' | cut -d: -f2)
+ PASSAGES=$(echo "$STATS_LINE" | grep -oE 'PASS:[0-9]+' | cut -d: -f2)
+
+ # Valeurs par défaut si extraction échoue
+ OPE=${OPE:-0}
+ USERS=${USERS:-0}
+ SECTORS=${SECTORS:-0}
+ PASSAGES=${PASSAGES:-0}
+
+ log " 📊 ope: ${OPE} | users: ${USERS} | sectors: ${SECTORS} | passages: ${PASSAGES}"
+ else
+ log " 📊 Statistiques non disponibles"
+ fi
+ fi
+ else
+ # Erreur
+ ERROR_COUNT=$((ERROR_COUNT + 1))
+ log_error "Entité #${ENTITY_ID} (${ENTITY_NAME}) - Erreur code $EXIT_CODE"
+
+ # Afficher les dernières lignes du log d'erreur
+ if [ -f "/tmp/migration_output_$$.txt" ]; then
+ log "${RED} 📋 Dernières erreurs:${NC}"
+ tail -5 "/tmp/migration_output_$$.txt" | sed 's/^/ /' | tee -a "$BATCH_LOG"
+ fi
+
+ # Arrêter ou continuer ?
+ if [ $CONTINUE_ON_ERROR -eq 0 ]; then
+ log ""
+ log "${RED}❌ Migration interrompue suite à une erreur${NC}"
+ log " Utilisez --continue pour continuer malgré les erreurs"
+ exit 1
+ fi
+ fi
+
+ # Nettoyage
+ rm -f "/tmp/migration_output_$$.txt"
+
+ # Pause entre les migrations (pour éviter de surcharger)
+ sleep 1
+done
+
+# Fin de la migration
+END_TIME=$(date +%s)
+DURATION=$((END_TIME - START_TIME))
+HOURS=$((DURATION / 3600))
+MINUTES=$(((DURATION % 3600) / 60))
+SECONDS=$((DURATION % 60))
+
+# Résumé final
+log ""
+log "${BLUE}═══════════════════════════════════════════════════════════${NC}"
+log "${BLUE} Résumé de la migration${NC}"
+log "${BLUE}═══════════════════════════════════════════════════════════${NC}"
+log "✅ Succès: ${GREEN}${SUCCESS_COUNT}${NC}"
+log "❌ Erreurs: ${RED}${ERROR_COUNT}${NC}"
+log "⏭️ Ignorées: ${YELLOW}${SKIPPED_COUNT}${NC}"
+log "⏱️ Durée: ${HOURS}h ${MINUTES}m ${SECONDS}s"
+log ""
+log "📋 Logs détaillés:"
+log " - Batch: $BATCH_LOG"
+log " - Succès: $SUCCESS_LOG"
+log " - Erreurs: $ERROR_LOG"
+log " - Individuels: $LOG_DIR/entity_*.log"
+log "${BLUE}═══════════════════════════════════════════════════════════${NC}"
+
+# Code de sortie
+if [ $ERROR_COUNT -gt 0 ]; then
+ exit 1
+else
+ exit 0
+fi
diff --git a/api/scripts/migration2/php/lib/DataMigrator.php b/api/scripts/migration2/php/lib/DataMigrator.php
new file mode 100644
index 00000000..1195f0e8
--- /dev/null
+++ b/api/scripts/migration2/php/lib/DataMigrator.php
@@ -0,0 +1,176 @@
+connection = $connection;
+ $this->logger = $logger;
+ $this->sourceDb = $connection->getSourceDb();
+ $this->targetDb = $connection->getTargetDb();
+ }
+
+ /**
+ * Méthode principale de migration (à implémenter dans chaque migrator)
+ *
+ * @param int|null $entityId ID de l'entité à migrer (null = toutes)
+ * @param bool $deleteBefore Supprimer les données existantes avant migration
+ * @return array ['success' => int, 'errors' => int]
+ */
+ abstract public function migrate(?int $entityId = null, bool $deleteBefore = false): array;
+
+ /**
+ * Retourne le nom de la table gérée par ce migrator
+ */
+ abstract public function getTableName(): string;
+
+ /**
+ * Supprime les données d'une entité dans la cible
+ * À surcharger si la logique de suppression est spécifique
+ *
+ * @param int $entityId ID de l'entité
+ * @return int Nombre de lignes supprimées
+ */
+ protected function deleteEntityData(int $entityId): int
+ {
+ $table = $this->getTableName();
+
+ try {
+ // Par défaut: suppression simple avec fk_entite
+ $stmt = $this->targetDb->prepare("DELETE FROM $table WHERE fk_entite = ?");
+ $stmt->execute([$entityId]);
+ $deleted = $stmt->rowCount();
+
+ if ($deleted > 0) {
+ $this->logger->debug(" Supprimé $deleted ligne(s) de $table pour entité #$entityId");
+ }
+
+ return $deleted;
+
+ } catch (PDOException $e) {
+ $this->logger->warning(" Erreur suppression $table: " . $e->getMessage());
+ return 0;
+ }
+ }
+
+ /**
+ * Compte les lignes dans la source
+ *
+ * @param int|null $entityId ID de l'entité (null = toutes)
+ * @return int Nombre de lignes
+ */
+ protected function countSourceRows(?int $entityId = null): int
+ {
+ return $this->connection->countSourceRows($this->getTableName(), $entityId);
+ }
+
+ /**
+ * Compte les lignes dans la cible
+ *
+ * @param int|null $entityId ID de l'entité (null = toutes)
+ * @return int Nombre de lignes
+ */
+ protected function countTargetRows(?int $entityId = null): int
+ {
+ return $this->connection->countTargetRows($this->getTableName(), $entityId);
+ }
+
+ /**
+ * Log le début de la migration d'une table
+ */
+ protected function logStart(?int $entityId = null): void
+ {
+ $table = $this->getTableName();
+ $entityStr = $entityId ? " pour entité #$entityId" : " (toutes les entités)";
+ $this->logger->info("🔄 Migration de $table{$entityStr}...");
+ }
+
+ /**
+ * Log la fin de la migration avec statistiques
+ *
+ * @param int $success Nombre de succès
+ * @param int $errors Nombre d'erreurs
+ * @param int|null $entityId ID de l'entité
+ */
+ protected function logEnd(int $success, int $errors, ?int $entityId = null): void
+ {
+ $table = $this->getTableName();
+ $sourceCount = $this->countSourceRows($entityId);
+ $targetCount = $this->countTargetRows($entityId);
+ $diff = $targetCount - $sourceCount;
+ $diffStr = $diff >= 0 ? "+$diff" : "$diff";
+
+ if ($errors > 0) {
+ $this->logger->warning(" ⚠️ $table: $success succès, $errors erreurs");
+ } else {
+ $this->logger->success(" ✓ $table: $success enregistrement(s) migré(s)");
+ }
+
+ $this->logger->info(" 📊 SOURCE: $sourceCount → CIBLE: $targetCount (différence: $diffStr)");
+ }
+
+ /**
+ * Exécute une requête INSERT avec ON DUPLICATE KEY UPDATE
+ *
+ * @param string $insertSql SQL d'insertion
+ * @param array $data Données à insérer
+ * @return bool True si succès
+ */
+ protected function insertOrUpdate(string $insertSql, array $data): bool
+ {
+ try {
+ $stmt = $this->targetDb->prepare($insertSql);
+ $stmt->execute($data);
+ return true;
+ } catch (PDOException $e) {
+ $this->logger->debug(" Erreur INSERT: " . $e->getMessage());
+ return false;
+ }
+ }
+
+ /**
+ * Démarre une transaction sur la cible
+ */
+ protected function beginTransaction(): void
+ {
+ if (!$this->targetDb->inTransaction()) {
+ $this->targetDb->beginTransaction();
+ }
+ }
+
+ /**
+ * Commit la transaction
+ */
+ protected function commit(): void
+ {
+ if ($this->targetDb->inTransaction()) {
+ $this->targetDb->commit();
+ }
+ }
+
+ /**
+ * Rollback la transaction
+ */
+ protected function rollback(): void
+ {
+ if ($this->targetDb->inTransaction()) {
+ $this->targetDb->rollBack();
+ }
+ }
+}
diff --git a/api/scripts/migration2/php/lib/DatabaseConfig.php b/api/scripts/migration2/php/lib/DatabaseConfig.php
new file mode 100644
index 00000000..d01f0447
--- /dev/null
+++ b/api/scripts/migration2/php/lib/DatabaseConfig.php
@@ -0,0 +1,192 @@
+ [
+ 'name' => 'DÉVELOPPEMENT',
+ 'hostname' => 'dapp.geosector.fr',
+ 'source_db' => 'geosector',
+ 'target_db' => 'dva_geo'
+ ],
+ 'rca' => [
+ 'name' => 'RECETTE',
+ 'hostname' => 'rapp.geosector.fr',
+ 'source_db' => 'geosector',
+ 'target_db' => 'rca_geo'
+ ],
+ 'pra' => [
+ 'name' => 'PRODUCTION',
+ 'hostname' => 'app3.geosector.fr',
+ 'source_db' => 'geosector',
+ 'target_db' => 'pra_geo'
+ ]
+ ];
+
+ private $env;
+ private $config;
+ private $appConfig;
+
+ /**
+ * Constructeur
+ *
+ * @param string $env Environnement: 'dva', 'rca' ou 'pra'
+ * @throws Exception Si l'environnement est invalide
+ */
+ public function __construct(string $env)
+ {
+ if (!isset(self::ENV_MAPPING[$env])) {
+ throw new Exception("Invalid environment: $env. Use 'dva', 'rca' or 'pra'");
+ }
+
+ $this->env = $env;
+
+ // Charger AppConfig (remonter de 4 niveaux: lib -> php -> migration2 -> scripts -> api)
+ $appConfigPath = dirname(__DIR__, 4) . '/src/Config/AppConfig.php';
+ if (!file_exists($appConfigPath)) {
+ throw new Exception("AppConfig not found at: $appConfigPath");
+ }
+ require_once $appConfigPath;
+
+ // Simuler le host pour AppConfig en CLI
+ $hostname = self::ENV_MAPPING[$env]['hostname'];
+ $_SERVER['SERVER_NAME'] = $hostname;
+ $_SERVER['HTTP_HOST'] = $hostname;
+
+ $this->appConfig = AppConfig::getInstance();
+
+ // Récupérer la config DB depuis AppConfig
+ $dbConfig = $this->appConfig->getDatabaseConfig();
+
+ if (!$dbConfig || !isset($dbConfig['host'])) {
+ throw new Exception("Database configuration not found for hostname: $hostname");
+ }
+
+ // Construire la config pour la migration
+ $this->config = [
+ 'name' => self::ENV_MAPPING[$env]['name'],
+ 'host' => $dbConfig['host'],
+ 'port' => $dbConfig['port'] ?? 3306,
+ 'user' => $dbConfig['username'],
+ 'pass' => $dbConfig['password'],
+ 'source_db' => self::ENV_MAPPING[$env]['source_db'],
+ 'target_db' => self::ENV_MAPPING[$env]['target_db']
+ ];
+ }
+
+ /**
+ * Retourne l'environnement actuel
+ */
+ public function getEnv(): string
+ {
+ return $this->env;
+ }
+
+ /**
+ * Retourne le nom complet de l'environnement
+ */
+ public function getEnvName(): string
+ {
+ return $this->config['name'];
+ }
+
+ /**
+ * Retourne l'hôte de la base de données
+ */
+ public function getHost(): string
+ {
+ return $this->config['host'];
+ }
+
+ /**
+ * Retourne le port de la base de données
+ */
+ public function getPort(): int
+ {
+ return $this->config['port'];
+ }
+
+ /**
+ * Retourne l'utilisateur de la base de données
+ */
+ public function getUser(): string
+ {
+ return $this->config['user'];
+ }
+
+ /**
+ * Retourne le mot de passe de la base de données
+ */
+ public function getPassword(): string
+ {
+ return $this->config['pass'];
+ }
+
+ /**
+ * Retourne le nom de la base source
+ */
+ public function getSourceDb(): string
+ {
+ return $this->config['source_db'];
+ }
+
+ /**
+ * Retourne le nom de la base cible
+ */
+ public function getTargetDb(): string
+ {
+ return $this->config['target_db'];
+ }
+
+ /**
+ * Retourne toute la configuration
+ */
+ public function getConfig(): array
+ {
+ return $this->config;
+ }
+
+ /**
+ * Détecte automatiquement l'environnement depuis le hostname
+ *
+ * @return string 'dva', 'rca' ou 'pra' (défaut: 'dva')
+ */
+ public static function autoDetect(): string
+ {
+ $hostname = gethostname();
+
+ switch ($hostname) {
+ case 'dva-geo':
+ return 'dva';
+ case 'rca-geo':
+ return 'rca';
+ case 'pra-geo':
+ return 'pra';
+ default:
+ return 'dva'; // Défaut
+ }
+ }
+
+ /**
+ * Vérifie si un environnement existe
+ */
+ public static function exists(string $env): bool
+ {
+ return isset(self::ENV_MAPPING[$env]);
+ }
+
+ /**
+ * Retourne la liste des environnements disponibles
+ */
+ public static function getAvailableEnvironments(): array
+ {
+ return array_keys(self::ENV_MAPPING);
+ }
+}
diff --git a/api/scripts/migration2/php/lib/DatabaseConnection.php b/api/scripts/migration2/php/lib/DatabaseConnection.php
new file mode 100644
index 00000000..19e9a868
--- /dev/null
+++ b/api/scripts/migration2/php/lib/DatabaseConnection.php
@@ -0,0 +1,201 @@
+config = $config;
+ $this->logger = $logger;
+ }
+
+ /**
+ * Établit les connexions aux bases source et cible
+ *
+ * @return bool True si succès
+ */
+ public function connect(): bool
+ {
+ try {
+ // Connexion à la base source
+ $this->connectSource();
+
+ // Connexion à la base cible
+ $this->connectTarget();
+
+ // Vérifier les versions MariaDB
+ $this->checkVersions();
+
+ return true;
+
+ } catch (PDOException $e) {
+ $this->logger->error("Erreur de connexion: " . $e->getMessage());
+ return false;
+ }
+ }
+
+ /**
+ * Connexion à la base source
+ */
+ private function connectSource(): void
+ {
+ $dsn = sprintf(
+ 'mysql:host=%s;port=%d;dbname=%s;charset=utf8mb4',
+ $this->config->getHost(),
+ $this->config->getPort(),
+ $this->config->getSourceDb()
+ );
+
+ $this->sourceDb = new PDO($dsn, $this->config->getUser(), $this->config->getPassword(), [
+ PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION,
+ PDO::ATTR_DEFAULT_FETCH_MODE => PDO::FETCH_ASSOC,
+ PDO::ATTR_TIMEOUT => 600,
+ PDO::MYSQL_ATTR_INIT_COMMAND => "SET NAMES utf8mb4"
+ ]);
+
+ $this->logger->success("✓ Connexion SOURCE: {$this->config->getSourceDb()} sur {$this->config->getHost()}");
+ }
+
+ /**
+ * Connexion à la base cible
+ */
+ private function connectTarget(): void
+ {
+ $dsn = sprintf(
+ 'mysql:host=%s;port=%d;dbname=%s;charset=utf8mb4',
+ $this->config->getHost(),
+ $this->config->getPort(),
+ $this->config->getTargetDb()
+ );
+
+ $this->targetDb = new PDO($dsn, $this->config->getUser(), $this->config->getPassword(), [
+ PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION,
+ PDO::ATTR_DEFAULT_FETCH_MODE => PDO::FETCH_ASSOC,
+ PDO::ATTR_TIMEOUT => 600,
+ PDO::MYSQL_ATTR_INIT_COMMAND => "SET NAMES utf8mb4"
+ ]);
+
+ $this->logger->success("✓ Connexion CIBLE: {$this->config->getTargetDb()} sur {$this->config->getHost()}");
+ }
+
+ /**
+ * Vérifie et affiche les versions MariaDB
+ */
+ private function checkVersions(): void
+ {
+ $sourceVersion = $this->sourceDb->query("SELECT VERSION()")->fetchColumn();
+ $targetVersion = $this->targetDb->query("SELECT VERSION()")->fetchColumn();
+
+ $this->logger->info(" Version SOURCE: $sourceVersion");
+ $this->logger->info(" Version CIBLE: $targetVersion");
+ }
+
+ /**
+ * Retourne la connexion à la base source
+ */
+ public function getSourceDb(): PDO
+ {
+ if (!$this->sourceDb) {
+ throw new Exception("Source database not connected. Call connect() first.");
+ }
+ return $this->sourceDb;
+ }
+
+ /**
+ * Retourne la connexion à la base cible
+ */
+ public function getTargetDb(): PDO
+ {
+ if (!$this->targetDb) {
+ throw new Exception("Target database not connected. Call connect() first.");
+ }
+ return $this->targetDb;
+ }
+
+ /**
+ * Compte le nombre de lignes dans une table de la source
+ *
+ * @param string $table Nom de la table
+ * @param int|null $entityId Filtrer par fk_entite (optionnel)
+ * @return int Nombre de lignes
+ */
+ public function countSourceRows(string $table, ?int $entityId = null): int
+ {
+ $sql = "SELECT COUNT(*) FROM $table";
+
+ if ($entityId !== null) {
+ // Tables avec fk_entite direct
+ if (in_array($table, ['users', 'operations', 'entites'])) {
+ $sql .= " WHERE fk_entite = :entity_id";
+ }
+ // Tables liées via operations
+ elseif (in_array($table, ['ope_sectors', 'ope_users', 'ope_pass'])) {
+ $sql .= " WHERE fk_operation IN (SELECT id FROM operations WHERE fk_entite = :entity_id)";
+ }
+ }
+
+ $stmt = $this->sourceDb->prepare($sql);
+ if ($entityId !== null) {
+ $stmt->execute(['entity_id' => $entityId]);
+ } else {
+ $stmt->execute();
+ }
+
+ return (int) $stmt->fetchColumn();
+ }
+
+ /**
+ * Compte le nombre de lignes dans une table de la cible
+ *
+ * @param string $table Nom de la table
+ * @param int|null $entityId Filtrer par fk_entite (optionnel)
+ * @return int Nombre de lignes
+ */
+ public function countTargetRows(string $table, ?int $entityId = null): int
+ {
+ $sql = "SELECT COUNT(*) FROM $table";
+
+ if ($entityId !== null) {
+ if (in_array($table, ['users', 'operations', 'entites'])) {
+ $sql .= " WHERE fk_entite = :entity_id";
+ }
+ elseif (in_array($table, ['ope_sectors', 'ope_users', 'ope_pass'])) {
+ $sql .= " WHERE fk_operation IN (SELECT id FROM operations WHERE fk_entite = :entity_id)";
+ }
+ }
+
+ $stmt = $this->targetDb->prepare($sql);
+ if ($entityId !== null) {
+ $stmt->execute(['entity_id' => $entityId]);
+ } else {
+ $stmt->execute();
+ }
+
+ return (int) $stmt->fetchColumn();
+ }
+
+ /**
+ * Ferme les connexions
+ */
+ public function close(): void
+ {
+ $this->sourceDb = null;
+ $this->targetDb = null;
+ $this->logger->info("Connexions fermées");
+ }
+}
diff --git a/api/scripts/migration2/php/lib/MigrationLogger.php b/api/scripts/migration2/php/lib/MigrationLogger.php
new file mode 100644
index 00000000..1d0d6ec5
--- /dev/null
+++ b/api/scripts/migration2/php/lib/MigrationLogger.php
@@ -0,0 +1,219 @@
+logFile = $logFile ?? $defaultLogDir . '/migration_' . date('Ymd_His') . '.log';
+ $this->verbose = $verbose;
+
+ // Créer le dossier parent si nécessaire
+ $dir = dirname($this->logFile);
+ if (!is_dir($dir)) {
+ mkdir($dir, 0755, true);
+ }
+
+ // Vérifier que le fichier est accessible en écriture
+ if (!is_writable(dirname($this->logFile))) {
+ throw new Exception("Log directory is not writable: " . dirname($this->logFile));
+ }
+ }
+
+ /**
+ * Log un message avec niveau INFO
+ */
+ public function info(string $message): void
+ {
+ $this->log($message, 'INFO');
+ }
+
+ /**
+ * Log un message avec niveau SUCCESS
+ */
+ public function success(string $message): void
+ {
+ $this->log($message, 'SUCCESS');
+ }
+
+ /**
+ * Log un message avec niveau WARNING
+ */
+ public function warning(string $message): void
+ {
+ $this->log($message, 'WARNING');
+ }
+
+ /**
+ * Log un message avec niveau ERROR
+ */
+ public function error(string $message): void
+ {
+ $this->log($message, 'ERROR');
+ }
+
+ /**
+ * Log un message avec niveau DEBUG
+ */
+ public function debug(string $message): void
+ {
+ $this->log($message, 'DEBUG');
+ }
+
+ /**
+ * Log une ligne de séparation
+ */
+ public function separator(): void
+ {
+ $this->log(str_repeat('=', 80), 'INFO');
+ }
+
+ /**
+ * Log générique
+ *
+ * @param string $message Message à logger
+ * @param string $level Niveau: INFO, SUCCESS, WARNING, ERROR, DEBUG
+ */
+ private function log(string $message, string $level = 'INFO'): void
+ {
+ $timestamp = date('Y-m-d H:i:s');
+ $logLine = "[{$timestamp}] [{$level}] {$message}\n";
+
+ // Écriture dans le fichier
+ file_put_contents($this->logFile, $logLine, FILE_APPEND);
+
+ // Affichage à l'écran si verbose
+ if ($this->verbose) {
+ $this->printColored($message, $level);
+ }
+ }
+
+ /**
+ * Affiche un message coloré selon le niveau
+ */
+ private function printColored(string $message, string $level): void
+ {
+ $colors = [
+ 'INFO' => "\033[0;37m", // Blanc
+ 'SUCCESS' => "\033[0;32m", // Vert
+ 'WARNING' => "\033[0;33m", // Jaune
+ 'ERROR' => "\033[0;31m", // Rouge
+ 'DEBUG' => "\033[0;36m" // Cyan
+ ];
+
+ $reset = "\033[0m";
+ $color = $colors[$level] ?? $colors['INFO'];
+
+ echo $color . $message . $reset . "\n";
+ }
+
+ /**
+ * Retourne le chemin du fichier de log
+ */
+ public function getLogFile(): string
+ {
+ return $this->logFile;
+ }
+
+ /**
+ * Log des statistiques de migration
+ *
+ * @param array $stats Tableau associatif [table => count]
+ */
+ public function logStats(array $stats): void
+ {
+ $this->separator();
+ $this->info("📊 Statistiques de migration:");
+
+ foreach ($stats as $table => $count) {
+ $this->info(" - {$table}: {$count} enregistrement(s)");
+ }
+
+ $this->separator();
+ }
+
+ /**
+ * Log une ligne spéciale pour parsing automatique
+ * Format: #STATS# KEY1:VAL1 KEY2:VAL2 ...
+ */
+ public function logParsableStats(array $stats): void
+ {
+ $pairs = [];
+ foreach ($stats as $key => $value) {
+ $pairs[] = strtoupper($key) . ':' . $value;
+ }
+
+ $line = '#STATS# ' . implode(' ', $pairs);
+ $this->log($line, 'INFO');
+ }
+
+ /**
+ * Affiche et log un récapitulatif complet de migration
+ *
+ * @param array $summary Tableau de statistiques hiérarchique
+ */
+ public function logMigrationSummary(array $summary): void
+ {
+ $this->separator();
+ $this->separator();
+ $this->info("📊 RÉCAPITULATIF DE LA MIGRATION");
+ $this->separator();
+
+ // Entité
+ if (isset($summary['entity'])) {
+ $this->info("Entité: {$summary['entity']['name']} (ID: {$summary['entity']['id']})");
+ }
+ $this->info("Date: " . date('Y-m-d H:i:s'));
+ $this->info("");
+
+ // Nombre total d'opérations
+ $totalOperations = count($summary['operations'] ?? []);
+ $this->success("Opérations migrées: {$totalOperations}");
+ $this->info("");
+
+ // Détail par opération
+ $operationNum = 1;
+ foreach ($summary['operations'] ?? [] as $operation) {
+ $this->info("Opération #{$operationNum}: \"{$operation['name']}\" (ID: {$operation['id']})");
+ $this->info(" ├─ Utilisateurs: {$operation['users']}");
+ $this->info(" ├─ Secteurs: {$operation['sectors']}");
+ $this->info(" ├─ Passages totaux: {$operation['total_passages']}");
+
+ if (!empty($operation['sectors_detail'])) {
+ $this->info(" └─ Détail par secteur:");
+
+ $sectorCount = count($operation['sectors_detail']);
+ $sectorNum = 0;
+ foreach ($operation['sectors_detail'] as $sector) {
+ $sectorNum++;
+ $isLast = ($sectorNum === $sectorCount);
+ $prefix = $isLast ? " └─" : " ├─";
+
+ $this->info("{$prefix} {$sector['name']} (ID: {$sector['id']})");
+ $this->info(" " . ($isLast ? " " : "│") . " ├─ Utilisateurs affectés: {$sector['users']}");
+ $this->info(" " . ($isLast ? " " : "│") . " └─ Passages: {$sector['passages']}");
+ }
+ }
+
+ $this->info("");
+ $operationNum++;
+ }
+
+ $this->separator();
+ }
+}
diff --git a/api/scripts/migration2/php/lib/OperationMigrator.php b/api/scripts/migration2/php/lib/OperationMigrator.php
new file mode 100644
index 00000000..bd7fe195
--- /dev/null
+++ b/api/scripts/migration2/php/lib/OperationMigrator.php
@@ -0,0 +1,312 @@
+sourceDb = $sourceDb;
+ $this->targetDb = $targetDb;
+ $this->logger = $logger;
+ $this->userMigrator = $userMigrator;
+ $this->sectorMigrator = $sectorMigrator;
+ }
+
+ /**
+ * Récupère les opérations à migrer pour une entité
+ * - 1 opération active
+ * - 2 dernières opérations inactives avec au moins 10 passages effectués
+ *
+ * @param int $entityId ID de l'entité
+ * @return array Liste des IDs d'opérations à migrer
+ */
+ public function getOperationsToMigrate(int $entityId): array
+ {
+ $operationIds = [];
+
+ // 1. Récupérer l'opération active (pour vérification)
+ $stmt = $this->sourceDb->prepare("
+ SELECT rowid
+ FROM operations
+ WHERE fk_entite = :entity_id AND active = 1
+ LIMIT 1
+ ");
+ $stmt->execute([':entity_id' => $entityId]);
+ $activeOp = $stmt->fetch(PDO::FETCH_COLUMN);
+
+ // 2. Récupérer les 2 dernières opérations inactives avec >= 10 passages effectués
+ // ORDER BY DESC pour avoir les plus récentes, puis on inverse
+ $stmt = $this->sourceDb->prepare("
+ SELECT o.rowid, COUNT(p.rowid) as nb_passages
+ FROM operations o
+ LEFT JOIN ope_pass p ON p.fk_operation = o.rowid AND p.fk_type = 1
+ WHERE o.fk_entite = :entity_id
+ AND o.active = 0
+ " . ($activeOp ? "AND o.rowid != :active_id" : "") . "
+ GROUP BY o.rowid
+ HAVING nb_passages >= 10
+ ORDER BY o.rowid DESC
+ LIMIT 2
+ ");
+
+ $params = [':entity_id' => $entityId];
+ if ($activeOp) {
+ $params[':active_id'] = $activeOp;
+ }
+
+ $stmt->execute($params);
+ $inactiveOps = $stmt->fetchAll(PDO::FETCH_ASSOC);
+
+ // Inverser pour avoir l'ordre chronologique (plus ancienne → plus récente)
+ $inactiveOps = array_reverse($inactiveOps);
+
+ foreach ($inactiveOps as $op) {
+ $operationIds[] = $op['rowid'];
+ $this->logger->info("✓ Opération inactive trouvée: {$op['rowid']} ({$op['nb_passages']} passages)");
+ }
+
+ // 3. Ajouter l'opération active EN DERNIER
+ if ($activeOp) {
+ $operationIds[] = $activeOp;
+ $this->logger->info("✓ Opération active trouvée: {$activeOp}");
+ }
+
+ $this->logger->info("📊 Total: " . count($operationIds) . " opération(s) à migrer");
+
+ return $operationIds;
+ }
+
+ /**
+ * Migre une opération complète avec tous ses utilisateurs et secteurs
+ *
+ * @param int $oldOperationId ID de l'opération dans l'ancienne base
+ * @return array|null Tableau de statistiques ou null en cas d'erreur
+ */
+ public function migrateOperation(int $oldOperationId): ?array
+ {
+ $this->logger->separator();
+ $this->logger->info("🔄 Migration de l'opération ID: {$oldOperationId}");
+
+ try {
+ // 1. Récupérer l'opération source
+ $stmt = $this->sourceDb->prepare("
+ SELECT * FROM operations
+ WHERE rowid = :id
+ ");
+ $stmt->execute([':id' => $oldOperationId]);
+ $operation = $stmt->fetch(PDO::FETCH_ASSOC);
+
+ if (!$operation) {
+ $this->logger->warning("Opération {$oldOperationId} non trouvée");
+ return null;
+ }
+
+ // 2. Créer l'opération dans la nouvelle base
+ $newOperationId = $this->createOperation($operation);
+
+ if (!$newOperationId) {
+ return null;
+ }
+
+ $this->logger->success("✓ Opération créée avec ID: {$newOperationId}");
+
+ // 3. Migrer les utilisateurs de l'opération
+ // Pour opération active : tous les users actifs de l'entité
+ // Pour opération inactive : uniquement ceux dans ope_users_sectors
+ $entityId = (int)$operation['fk_entite'];
+ $isActiveOperation = (int)$operation['active'] === 1;
+
+ $userResult = $this->userMigrator->migrateOperationUsers(
+ $oldOperationId,
+ $newOperationId,
+ $entityId,
+ $isActiveOperation
+ );
+ $userMapping = $userResult['mapping'];
+ $usersCount = $userResult['count'];
+
+ if (empty($userMapping)) {
+ $this->logger->warning("Aucun utilisateur migré, abandon de l'opération {$oldOperationId}");
+ return null;
+ }
+
+ // 4. Récupérer les secteurs DISTINCTS de l'opération
+ $stmt = $this->sourceDb->prepare("
+ SELECT DISTINCT fk_sector
+ FROM ope_users_sectors
+ WHERE fk_operation = :operation_id AND active = 1
+ ");
+ $stmt->execute([':operation_id' => $oldOperationId]);
+ $sectors = $stmt->fetchAll(PDO::FETCH_COLUMN);
+
+ $this->logger->info("📍 " . count($sectors) . " secteur(s) distinct(s) à migrer");
+
+ // 5. Migrer chaque secteur et collecter les stats
+ $sectorsDetail = [];
+ $totalPassages = 0;
+
+ foreach ($sectors as $oldSectorId) {
+ $sectorStats = $this->sectorMigrator->migrateSector(
+ $oldOperationId,
+ $newOperationId,
+ $oldSectorId,
+ $userMapping
+ );
+
+ if ($sectorStats) {
+ $sectorsDetail[] = $sectorStats;
+ $totalPassages += $sectorStats['passages'];
+ }
+ }
+
+ // 6. Migrer les médias de l'opération (support='operations')
+ $this->migrateOperationMedias($oldOperationId, $newOperationId);
+
+ $this->logger->success("✅ Migration de l'opération {$oldOperationId} terminée");
+
+ // 7. Retourner les statistiques
+ return [
+ 'id' => $newOperationId,
+ 'name' => $operation['libelle'],
+ 'users' => $usersCount,
+ 'sectors' => count($sectorsDetail),
+ 'total_passages' => $totalPassages,
+ 'sectors_detail' => $sectorsDetail
+ ];
+
+ } catch (Exception $e) {
+ $this->logger->error("❌ Erreur migration opération {$oldOperationId}: " . $e->getMessage());
+ return null;
+ }
+ }
+
+ /**
+ * Crée une opération dans la nouvelle base
+ *
+ * @param array $operation Données de l'opération
+ * @return int|null ID de la nouvelle opération ou null en cas d'erreur
+ */
+ private function createOperation(array $operation): ?int
+ {
+ try {
+ $stmt = $this->targetDb->prepare("
+ INSERT INTO operations (
+ fk_entite, libelle, date_deb, date_fin,
+ chk_distinct_sectors,
+ created_at, fk_user_creat, updated_at, fk_user_modif, chk_active
+ ) VALUES (
+ :fk_entite, :libelle, :date_deb, :date_fin,
+ :chk_distinct_sectors,
+ :created_at, :fk_user_creat, :updated_at, :fk_user_modif, :chk_active
+ )
+ ");
+
+ $stmt->execute([
+ ':fk_entite' => $operation['fk_entite'],
+ ':libelle' => $operation['libelle'],
+ ':date_deb' => $operation['date_deb'],
+ ':date_fin' => $operation['date_fin'],
+ ':chk_distinct_sectors' => $operation['chk_distinct_sectors'],
+ ':created_at' => $operation['date_creat'],
+ ':fk_user_creat' => $operation['fk_user_creat'],
+ ':updated_at' => $operation['date_modif'],
+ ':fk_user_modif' => $operation['fk_user_modif'] ?? 0,
+ ':chk_active' => $operation['active']
+ ]);
+
+ return (int)$this->targetDb->lastInsertId();
+
+ } catch (Exception $e) {
+ $this->logger->error("❌ Erreur création opération: " . $e->getMessage());
+ return null;
+ }
+ }
+
+ /**
+ * Migre les médias d'une opération
+ *
+ * @param int $oldOperationId ID ancienne opération
+ * @param int $newOperationId ID nouvelle opération
+ * @return int Nombre de médias migrés
+ */
+ private function migrateOperationMedias(int $oldOperationId, int $newOperationId): int
+ {
+ $stmt = $this->sourceDb->prepare("
+ SELECT * FROM medias
+ WHERE support = 'operations' AND support_rowid = :operation_id
+ ");
+ $stmt->execute([':operation_id' => $oldOperationId]);
+ $medias = $stmt->fetchAll(PDO::FETCH_ASSOC);
+
+ if (empty($medias)) {
+ return 0;
+ }
+
+ $count = 0;
+ foreach ($medias as $media) {
+ $stmt = $this->targetDb->prepare("
+ INSERT INTO medias (
+ dir0, dir1, dir2, support, support_rowid,
+ fichier, type_fichier, description, position,
+ hauteur, largeur, niveaugris,
+ created_at, fk_user_creat, updated_at, fk_user_modif
+ ) VALUES (
+ :dir0, :dir1, :dir2, :support, :support_rowid,
+ :fichier, :type_fichier, :description, :position,
+ :hauteur, :largeur, :niveaugris,
+ :created_at, :fk_user_creat, :updated_at, :fk_user_modif
+ )
+ ");
+
+ $stmt->execute([
+ ':dir0' => $media['dir0'],
+ ':dir1' => $media['dir1'],
+ ':dir2' => $media['dir2'],
+ ':support' => $media['support'],
+ ':support_rowid' => $newOperationId,
+ ':fichier' => $media['fichier'],
+ ':type_fichier' => $media['type_fichier'],
+ ':description' => $media['description'],
+ ':position' => $media['position'],
+ ':hauteur' => $media['hauteur'],
+ ':largeur' => $media['largeur'],
+ ':niveaugris' => $media['niveaugris'],
+ ':created_at' => $media['date_creat'],
+ ':fk_user_creat' => $media['fk_user_creat'],
+ ':updated_at' => $media['date_modif'],
+ ':fk_user_modif' => $media['fk_user_modif']
+ ]);
+
+ $count++;
+ }
+
+ $this->logger->success("✓ {$count} média(s) migré(s)");
+
+ return $count;
+ }
+}
diff --git a/api/scripts/migration2/php/lib/PassageMigrator.php b/api/scripts/migration2/php/lib/PassageMigrator.php
new file mode 100644
index 00000000..ac3c0db7
--- /dev/null
+++ b/api/scripts/migration2/php/lib/PassageMigrator.php
@@ -0,0 +1,256 @@
+sourceDb = $sourceDb;
+ $this->targetDb = $targetDb;
+ $this->logger = $logger;
+ }
+
+ /**
+ * Migre les passages d'un secteur dans une opération
+ *
+ * @param int $oldOperationId ID ancienne opération
+ * @param int $newOperationId ID nouvelle opération
+ * @param int $oldSectorId ID ancien secteur
+ * @param int $newOpeSectorId ID nouveau ope_sectors
+ * @param array $userMapping Mapping oldUserId => newOpeUserId
+ * @return int Nombre de passages migrés
+ */
+ public function migratePassages(
+ int $oldOperationId,
+ int $newOperationId,
+ int $oldSectorId,
+ int $newOpeSectorId,
+ array $userMapping
+ ): int {
+ $stmt = $this->sourceDb->prepare("
+ SELECT * FROM ope_pass
+ WHERE fk_operation = :operation_id
+ AND fk_sector = :sector_id
+ ");
+ $stmt->execute([
+ ':operation_id' => $oldOperationId,
+ ':sector_id' => $oldSectorId
+ ]);
+ $passages = $stmt->fetchAll(PDO::FETCH_ASSOC);
+
+ if (empty($passages)) {
+ return 0;
+ }
+
+ $count = 0;
+ foreach ($passages as $passage) {
+ // Vérifier que l'utilisateur a été migré
+ if (!isset($userMapping[$passage['fk_user']])) {
+ $this->logger->warning(" ⚠ Passage {$passage['rowid']}: User {$passage['fk_user']} non trouvé dans mapping");
+ continue;
+ }
+
+ // Récupérer l'ID de ope_users depuis le mapping
+ $newOpeUserId = $userMapping[$passage['fk_user']];
+
+ // Vérifier que le trio (operation, user, sector) existe dans ope_users_sectors
+ if (!$this->verifyUserSectorAssociation($newOperationId, $newOpeUserId, $newOpeSectorId)) {
+ $this->logger->warning(" ⚠ Passage {$passage['rowid']}: Trio (op={$newOperationId}, user={$newOpeUserId}, sector={$newOpeSectorId}) inexistant");
+ continue;
+ }
+
+ // Insérer le passage avec l'ID de ope_users
+ $newPassId = $this->insertPassage($passage, $newOperationId, $newOpeSectorId, $newOpeUserId);
+
+ if ($newPassId) {
+ // Migrer l'historique du passage
+ $this->migratePassageHisto($passage['rowid'], $newPassId, $userMapping);
+ $count++;
+ }
+ }
+
+ if ($count > 0) {
+ $this->logger->success(" ✓ {$count} passage(s) migré(s)");
+ }
+
+ return $count;
+ }
+
+ /**
+ * Vérifie qu'une association user-sector existe dans ope_users_sectors
+ *
+ * @param int $operationId ID opération
+ * @param int $userId ID ope_users (mapping)
+ * @param int $sectorId ID ope_sectors
+ * @return bool True si l'association existe
+ */
+ private function verifyUserSectorAssociation(int $operationId, int $userId, int $sectorId): bool
+ {
+ $stmt = $this->targetDb->prepare("
+ SELECT COUNT(*) FROM ope_users_sectors
+ WHERE fk_operation = :operation_id
+ AND fk_user = :user_id
+ AND fk_sector = :sector_id
+ ");
+ $stmt->execute([
+ ':operation_id' => $operationId,
+ ':user_id' => $userId,
+ ':sector_id' => $sectorId
+ ]);
+
+ return $stmt->fetchColumn() > 0;
+ }
+
+ /**
+ * Insère un passage dans la nouvelle base
+ *
+ * @param array $passage Données du passage
+ * @param int $newOperationId ID nouvelle opération
+ * @param int $newOpeSectorId ID nouveau secteur
+ * @param int $userId ID de ope_users (mapping)
+ * @return int|null ID du nouveau passage ou null en cas d'erreur
+ */
+ private function insertPassage(
+ array $passage,
+ int $newOperationId,
+ int $newOpeSectorId,
+ int $userId
+ ): ?int {
+ try {
+ $stmt = $this->targetDb->prepare("
+ INSERT INTO ope_pass (
+ fk_operation, fk_sector, fk_user, fk_adresse,
+ passed_at, fk_type, numero, rue, rue_bis, ville,
+ fk_habitat, appt, niveau, residence,
+ gps_lat, gps_lng, encrypted_name, montant, fk_type_reglement,
+ remarque, nom_recu, encrypted_email, email_erreur, chk_email_sent,
+ encrypted_phone, docremis, date_repasser, nb_passages,
+ chk_gps_maj, chk_map_create, chk_mobile, chk_synchro,
+ chk_api_adresse, chk_maj_adresse, anomalie,
+ created_at, fk_user_creat, updated_at, fk_user_modif, chk_active
+ ) VALUES (
+ :fk_operation, :fk_sector, :fk_user, :fk_adresse,
+ :passed_at, :fk_type, :numero, :rue, :rue_bis, :ville,
+ :fk_habitat, :appt, :niveau, :residence,
+ :gps_lat, :gps_lng, :encrypted_name, :montant, :fk_type_reglement,
+ :remarque, :nom_recu, :encrypted_email, :email_erreur, :chk_email_sent,
+ :encrypted_phone, :docremis, :date_repasser, :nb_passages,
+ :chk_gps_maj, :chk_map_create, :chk_mobile, :chk_synchro,
+ :chk_api_adresse, :chk_maj_adresse, :anomalie,
+ :created_at, :fk_user_creat, :updated_at, :fk_user_modif, :chk_active
+ )
+ ");
+
+ // Chiffrer les données sensibles
+ require_once dirname(__DIR__, 4) . '/src/Services/ApiService.php';
+
+ $stmt->execute([
+ ':fk_operation' => $newOperationId,
+ ':fk_sector' => $newOpeSectorId,
+ ':fk_user' => $userId, // ID de ope_users (mapping)
+ ':fk_adresse' => $passage['fk_adresse'],
+ ':passed_at' => $passage['date_eve'],
+ ':fk_type' => $passage['fk_type'],
+ ':numero' => $passage['numero'],
+ ':rue' => $passage['rue'],
+ ':rue_bis' => $passage['rue_bis'],
+ ':ville' => $passage['ville'],
+ ':fk_habitat' => $passage['fk_habitat'],
+ ':appt' => $passage['appt'],
+ ':niveau' => $passage['niveau'],
+ ':residence' => $passage['lieudit'] ?? null,
+ ':gps_lat' => $passage['gps_lat'],
+ ':gps_lng' => $passage['gps_lng'],
+ ':encrypted_name' => $passage['libelle'] ? ApiService::encryptData($passage['libelle']) : '', // Chiffrer avec IV aléatoire
+ ':montant' => $passage['montant'],
+ ':fk_type_reglement' => (!empty($passage['fk_type_reglement']) && $passage['fk_type_reglement'] > 0) ? $passage['fk_type_reglement'] : 4,
+ ':remarque' => $passage['remarque'],
+ ':nom_recu' => $passage['recu'] ?? null,
+ ':encrypted_email' => $passage['email'] ? ApiService::encryptSearchableData($passage['email']) : null,
+ ':email_erreur' => $passage['email_erreur'],
+ ':chk_email_sent' => $passage['chk_email_sent'],
+ ':encrypted_phone' => $passage['phone'] ? ApiService::encryptData($passage['phone']) : '',
+ ':docremis' => $passage['docremis'],
+ ':date_repasser' => $passage['date_repasser'],
+ ':nb_passages' => ($passage['fk_type'] == 2) ? 0 : $passage['nb_passages'],
+ ':chk_gps_maj' => $passage['chk_gps_maj'],
+ ':chk_map_create' => $passage['chk_map_create'],
+ ':chk_mobile' => $passage['chk_mobile'],
+ ':chk_synchro' => $passage['chk_synchro'],
+ ':chk_api_adresse' => $passage['chk_api_adresse'],
+ ':chk_maj_adresse' => $passage['chk_maj_adresse'],
+ ':anomalie' => $passage['anomalie'],
+ ':created_at' => $passage['date_creat'],
+ ':fk_user_creat' => $passage['fk_user_creat'] ?? 0,
+ ':updated_at' => $passage['date_modif'],
+ ':fk_user_modif' => $passage['fk_user_modif'] ?? 0,
+ ':chk_active' => $passage['active']
+ ]);
+
+ return (int)$this->targetDb->lastInsertId();
+
+ } catch (Exception $e) {
+ $this->logger->error(" ❌ Erreur insertion passage {$passage['rowid']}: " . $e->getMessage());
+ return null;
+ }
+ }
+
+ /**
+ * Migre l'historique d'un passage
+ *
+ * @param int $oldPassId ID ancien passage
+ * @param int $newPassId ID nouveau passage
+ * @param array $userMapping Non utilisé (conservé pour compatibilité)
+ * @return int Nombre d'entrées d'historique migrées
+ */
+ public function migratePassageHisto(int $oldPassId, int $newPassId, array $userMapping): int
+ {
+ $stmt = $this->sourceDb->prepare("
+ SELECT * FROM ope_pass_histo WHERE fk_pass = :pass_id
+ ");
+ $stmt->execute([':pass_id' => $oldPassId]);
+ $histos = $stmt->fetchAll(PDO::FETCH_ASSOC);
+
+ if (empty($histos)) {
+ return 0;
+ }
+
+ $count = 0;
+ foreach ($histos as $histo) {
+ $stmt = $this->targetDb->prepare("
+ INSERT INTO ope_pass_histo (
+ fk_pass, date_histo, sujet, remarque
+ ) VALUES (
+ :fk_pass, :date_histo, :sujet, :remarque
+ )
+ ");
+
+ $stmt->execute([
+ ':fk_pass' => $newPassId,
+ ':date_histo' => $histo['date_histo'],
+ ':sujet' => $histo['sujet'],
+ ':remarque' => $histo['remarque']
+ ]);
+
+ $count++;
+ }
+
+ return $count;
+ }
+}
diff --git a/api/scripts/migration2/php/lib/SectorMigrator.php b/api/scripts/migration2/php/lib/SectorMigrator.php
new file mode 100644
index 00000000..044550e4
--- /dev/null
+++ b/api/scripts/migration2/php/lib/SectorMigrator.php
@@ -0,0 +1,289 @@
+sourceDb = $sourceDb;
+ $this->targetDb = $targetDb;
+ $this->logger = $logger;
+ $this->passageMigrator = $passageMigrator;
+ }
+
+ /**
+ * Migre un secteur dans le contexte d'une opération
+ *
+ * @param int $oldOperationId ID ancienne opération
+ * @param int $newOperationId ID nouvelle opération
+ * @param int $oldSectorId ID ancien secteur
+ * @param array $userMapping Mapping oldUserId => newOpeUserId
+ * @return array|null ['id' => int, 'name' => string, 'users' => int, 'passages' => int] ou null en cas d'erreur
+ */
+ public function migrateSector(
+ int $oldOperationId,
+ int $newOperationId,
+ int $oldSectorId,
+ array $userMapping
+ ): ?array {
+ $this->logger->info(" 📍 Migration secteur ID: {$oldSectorId}");
+
+ try {
+ // 1. Récupérer le secteur source
+ $stmt = $this->sourceDb->prepare("
+ SELECT * FROM sectors WHERE rowid = :id
+ ");
+ $stmt->execute([':id' => $oldSectorId]);
+ $sector = $stmt->fetch(PDO::FETCH_ASSOC);
+
+ if (!$sector) {
+ $this->logger->warning(" Secteur {$oldSectorId} non trouvé");
+ return null;
+ }
+
+ // 2. Créer dans ope_sectors
+ $newOpeSectorId = $this->createOpeSector($sector, $newOperationId);
+
+ if (!$newOpeSectorId) {
+ return null;
+ }
+
+ // 3. Mapper "operationId_sectorId" → newOpeSectorId
+ $mappingKey = "{$oldOperationId}_{$oldSectorId}";
+ $this->sectorMapping[$mappingKey] = $newOpeSectorId;
+
+ $this->logger->success(" ✓ Secteur créé avec ID: {$newOpeSectorId}");
+
+ // 4. Migrer sectors_adresses
+ $this->migrateSectorAddresses($oldSectorId, $newOpeSectorId);
+
+ // 5. Migrer ope_users_sectors
+ $usersCount = $this->migrateUsersSectors($oldOperationId, $newOperationId, $oldSectorId, $newOpeSectorId, $userMapping);
+
+ // 6. Migrer ope_pass
+ $passagesCount = $this->passageMigrator->migratePassages(
+ $oldOperationId,
+ $newOperationId,
+ $oldSectorId,
+ $newOpeSectorId,
+ $userMapping
+ );
+
+ return [
+ 'id' => $newOpeSectorId,
+ 'name' => $sector['libelle'],
+ 'users' => $usersCount,
+ 'passages' => $passagesCount
+ ];
+
+ } catch (Exception $e) {
+ $this->logger->error(" ❌ Erreur migration secteur {$oldSectorId}: " . $e->getMessage());
+ return null;
+ }
+ }
+
+ /**
+ * Crée un secteur dans ope_sectors
+ *
+ * @param array $sector Données du secteur
+ * @param int $newOperationId ID nouvelle opération
+ * @return int|null ID du nouveau secteur ou null en cas d'erreur
+ */
+ private function createOpeSector(array $sector, int $newOperationId): ?int
+ {
+ try {
+ $stmt = $this->targetDb->prepare("
+ INSERT INTO ope_sectors (
+ fk_operation, libelle, sector, color,
+ created_at, fk_user_creat, updated_at, fk_user_modif, chk_active
+ ) VALUES (
+ :fk_operation, :libelle, :sector, :color,
+ :created_at, :fk_user_creat, :updated_at, :fk_user_modif, :chk_active
+ )
+ ");
+
+ $stmt->execute([
+ ':fk_operation' => $newOperationId,
+ ':libelle' => $sector['libelle'],
+ ':sector' => $sector['sector'],
+ ':color' => $sector['color'],
+ ':created_at' => $sector['date_creat'],
+ ':fk_user_creat' => $sector['fk_user_creat'] ?? 0,
+ ':updated_at' => $sector['date_modif'],
+ ':fk_user_modif' => $sector['fk_user_modif'] ?? 0,
+ ':chk_active' => $sector['active']
+ ]);
+
+ return (int)$this->targetDb->lastInsertId();
+
+ } catch (Exception $e) {
+ $this->logger->error(" ❌ Erreur création secteur: " . $e->getMessage());
+ return null;
+ }
+ }
+
+ /**
+ * Migre les adresses d'un secteur
+ *
+ * @param int $oldSectorId ID ancien secteur
+ * @param int $newOpeSectorId ID nouveau ope_sectors
+ * @return int Nombre d'adresses migrées
+ */
+ private function migrateSectorAddresses(int $oldSectorId, int $newOpeSectorId): int
+ {
+ $stmt = $this->sourceDb->prepare("
+ SELECT * FROM sectors_adresses WHERE fk_sector = :sector_id
+ ");
+ $stmt->execute([':sector_id' => $oldSectorId]);
+ $addresses = $stmt->fetchAll(PDO::FETCH_ASSOC);
+
+ if (empty($addresses)) {
+ return 0;
+ }
+
+ $count = 0;
+ foreach ($addresses as $address) {
+ $stmt = $this->targetDb->prepare("
+ INSERT INTO sectors_adresses (
+ fk_adresse, fk_sector, numero, rue_bis, rue, cp, ville,
+ gps_lat, gps_lng
+ ) VALUES (
+ :fk_adresse, :fk_sector, :numero, :rue_bis, :rue, :cp, :ville,
+ :gps_lat, :gps_lng
+ )
+ ");
+
+ $stmt->execute([
+ ':fk_adresse' => $address['fk_adresse'], // Garde la valeur telle quelle
+ ':fk_sector' => $newOpeSectorId,
+ ':numero' => $address['numero'],
+ ':rue_bis' => $address['rue_bis'],
+ ':rue' => $address['rue'],
+ ':cp' => $address['cp'],
+ ':ville' => $address['ville'],
+ ':gps_lat' => $address['gps_lat'],
+ ':gps_lng' => $address['gps_lng']
+ ]);
+
+ $count++;
+ }
+
+ $this->logger->success(" ✓ {$count} adresse(s) migrée(s)");
+
+ return $count;
+ }
+
+ /**
+ * Migre les associations utilisateurs-secteurs
+ *
+ * @param int $oldOperationId ID ancienne opération
+ * @param int $newOperationId ID nouvelle opération
+ * @param int $oldSectorId ID ancien secteur
+ * @param int $newOpeSectorId ID nouveau ope_sectors
+ * @param array $userMapping Mapping oldUserId => newOpeUserId
+ * @return int Nombre d'associations migrées
+ */
+ private function migrateUsersSectors(
+ int $oldOperationId,
+ int $newOperationId,
+ int $oldSectorId,
+ int $newOpeSectorId,
+ array $userMapping
+ ): int {
+ $stmt = $this->sourceDb->prepare("
+ SELECT * FROM ope_users_sectors
+ WHERE fk_operation = :operation_id
+ AND fk_sector = :sector_id
+ AND active = 1
+ ");
+ $stmt->execute([
+ ':operation_id' => $oldOperationId,
+ ':sector_id' => $oldSectorId
+ ]);
+ $usersSectors = $stmt->fetchAll(PDO::FETCH_ASSOC);
+
+ if (empty($usersSectors)) {
+ return 0;
+ }
+
+ $count = 0;
+ foreach ($usersSectors as $us) {
+ // Vérifier que l'utilisateur existe dans le mapping
+ // (le mapping sert juste à vérifier que l'user a été migré)
+ if (!isset($userMapping[$us['fk_user']])) {
+ $this->logger->warning(" ⚠ User {$us['fk_user']} non trouvé dans mapping");
+ continue;
+ }
+
+ $stmt = $this->targetDb->prepare("
+ INSERT INTO ope_users_sectors (
+ fk_operation, fk_user, fk_sector,
+ created_at, fk_user_creat, updated_at, fk_user_modif, chk_active
+ ) VALUES (
+ :fk_operation, :fk_user, :fk_sector,
+ :created_at, :fk_user_creat, :updated_at, :fk_user_modif, :chk_active
+ )
+ ");
+
+ $stmt->execute([
+ ':fk_operation' => $newOperationId,
+ ':fk_user' => $userMapping[$us['fk_user']], // ID de ope_users (mapping)
+ ':fk_sector' => $newOpeSectorId,
+ ':created_at' => date('Y-m-d H:i:s'),
+ ':fk_user_creat' => 0,
+ ':updated_at' => null,
+ ':fk_user_modif' => null,
+ ':chk_active' => $us['active']
+ ]);
+
+ $count++;
+ }
+
+ $this->logger->success(" ✓ {$count} association(s) user-secteur migrée(s)");
+
+ return $count;
+ }
+
+ /**
+ * Retourne le mapping des secteurs
+ *
+ * @return array "operationId_sectorId" => newOpeSectorId
+ */
+ public function getSectorMapping(): array
+ {
+ return $this->sectorMapping;
+ }
+
+ /**
+ * Définit le mapping des secteurs (utile pour réutilisation)
+ *
+ * @param array $mapping "operationId_sectorId" => newOpeSectorId
+ */
+ public function setSectorMapping(array $mapping): void
+ {
+ $this->sectorMapping = $mapping;
+ }
+}
diff --git a/api/scripts/migration2/php/lib/UserMigrator.php b/api/scripts/migration2/php/lib/UserMigrator.php
new file mode 100644
index 00000000..dfab0225
--- /dev/null
+++ b/api/scripts/migration2/php/lib/UserMigrator.php
@@ -0,0 +1,163 @@
+sourceDb = $sourceDb;
+ $this->targetDb = $targetDb;
+ $this->logger = $logger;
+ }
+
+ /**
+ * Migre les utilisateurs d'une opération
+ * - Si opération active : TOUS les users actifs de l'entité
+ * - Si opération inactive : Uniquement ceux dans ope_users_sectors
+ *
+ * @param int $oldOperationId ID ancienne opération
+ * @param int $newOperationId ID nouvelle opération
+ * @param int $entityId ID de l'entité
+ * @param bool $isActiveOperation True si opération active
+ * @return array ['mapping' => array, 'count' => int]
+ */
+ public function migrateOperationUsers(
+ int $oldOperationId,
+ int $newOperationId,
+ int $entityId,
+ bool $isActiveOperation
+ ): array {
+ $this->logger->info("👥 Migration des utilisateurs de l'opération...");
+
+ // Réinitialiser le mapping pour cette opération
+ $this->userMapping = [];
+
+ // Récupérer les utilisateurs selon le type d'opération
+ if ($isActiveOperation) {
+ // Pour l'opération active : TOUS les users actifs de l'entité
+ $this->logger->info(" ℹ Opération ACTIVE : migration de tous les users actifs de l'entité");
+ $stmt = $this->sourceDb->prepare("
+ SELECT rowid
+ FROM users
+ WHERE fk_entite = :entity_id AND active = 1
+ ");
+ $stmt->execute([':entity_id' => $entityId]);
+ $userIds = $stmt->fetchAll(PDO::FETCH_COLUMN);
+ } else {
+ // Pour les opérations inactives : uniquement ceux dans ope_users_sectors
+ $this->logger->info(" ℹ Opération INACTIVE : migration des users affectés aux secteurs");
+ $stmt = $this->sourceDb->prepare("
+ SELECT DISTINCT fk_user
+ FROM ope_users_sectors
+ WHERE fk_operation = :operation_id AND active = 1
+ ");
+ $stmt->execute([':operation_id' => $oldOperationId]);
+ $userIds = $stmt->fetchAll(PDO::FETCH_COLUMN);
+ }
+
+ if (empty($userIds)) {
+ $this->logger->warning("Aucun utilisateur trouvé pour l'opération {$oldOperationId}");
+ return ['mapping' => [], 'count' => 0];
+ }
+
+ $count = 0;
+ foreach ($userIds as $oldUserId) {
+ // Récupérer les infos utilisateur depuis la table users
+ $stmt = $this->sourceDb->prepare("
+ SELECT * FROM users WHERE rowid = :id AND active = 1
+ ");
+ $stmt->execute([':id' => $oldUserId]);
+ $user = $stmt->fetch(PDO::FETCH_ASSOC);
+
+ if (!$user) {
+ $this->logger->warning(" ⚠ Utilisateur {$oldUserId} non trouvé ou inactif");
+ continue;
+ }
+
+ // Créer dans ope_users de la nouvelle base
+ $stmt = $this->targetDb->prepare("
+ INSERT INTO ope_users (
+ fk_operation, fk_user, fk_role,
+ first_name, encrypted_name, sect_name,
+ created_at, fk_user_creat, updated_at, fk_user_modif, chk_active
+ ) VALUES (
+ :fk_operation, :fk_user, :fk_role,
+ :first_name, :encrypted_name, :sect_name,
+ :created_at, :fk_user_creat, :updated_at, :fk_user_modif, :chk_active
+ )
+ ");
+
+ $stmt->execute([
+ ':fk_operation' => $newOperationId,
+ ':fk_user' => $oldUserId, // Référence vers users.id
+ ':fk_role' => $user['fk_role'],
+ ':first_name' => $user['prenom'],
+ ':encrypted_name' => ApiService::encryptData($user['libelle']), // Chiffrer le nom avec IV aléatoire
+ ':sect_name' => $user['nom_tournee'],
+ ':created_at' => $user['date_creat'],
+ ':fk_user_creat' => $user['fk_user_creat'],
+ ':updated_at' => $user['date_modif'],
+ ':fk_user_modif' => $user['fk_user_modif'],
+ ':chk_active' => $user['active']
+ ]);
+
+ $newOpeUserId = (int)$this->targetDb->lastInsertId();
+
+ // Mapper oldUserId → newOpeUserId
+ $this->userMapping[$oldUserId] = $newOpeUserId;
+ $count++;
+ }
+
+ $this->logger->success(" ✓ {$count} utilisateur(s) migré(s)");
+
+ return ['mapping' => $this->userMapping, 'count' => $count];
+ }
+
+ /**
+ * Retourne le mapping des utilisateurs
+ *
+ * @return array oldUserId => newOpeUserId
+ */
+ public function getUserMapping(): array
+ {
+ return $this->userMapping;
+ }
+
+ /**
+ * Définit le mapping des utilisateurs (utile pour réutilisation)
+ *
+ * @param array $mapping oldUserId => newOpeUserId
+ */
+ public function setUserMapping(array $mapping): void
+ {
+ $this->userMapping = $mapping;
+ }
+
+ /**
+ * Récupère le nouvel ID ope_users depuis le mapping
+ *
+ * @param int $oldUserId ID ancien utilisateur
+ * @return int|null Nouvel ID ope_users ou null si non trouvé
+ */
+ public function getMappedUserId(int $oldUserId): ?int
+ {
+ return $this->userMapping[$oldUserId] ?? null;
+ }
+}
diff --git a/api/scripts/migration2/php/migrate_from_backup.php b/api/scripts/migration2/php/migrate_from_backup.php
new file mode 100755
index 00000000..294c4d29
--- /dev/null
+++ b/api/scripts/migration2/php/migrate_from_backup.php
@@ -0,0 +1,471 @@
+#!/usr/bin/env php
+config = new DatabaseConfig($env);
+ $this->mode = $mode;
+ $this->entityId = $entityId;
+ $this->deleteBefore = $deleteBefore;
+
+ // Générer le nom du fichier log selon le mode si non spécifié
+ if (!$logFile) {
+ $logDir = dirname(__DIR__, 2) . '/logs';
+ $timestamp = date('Ymd_His');
+
+ if ($mode === 'entity' && $entityId) {
+ $logFile = "{$logDir}/migration_entite_{$entityId}_{$timestamp}.log";
+ } else {
+ $logFile = "{$logDir}/migration_global_{$timestamp}.log";
+ }
+ }
+
+ $this->logger = new MigrationLogger($logFile);
+
+ // Log header
+ $this->logHeader();
+
+ // Connexions
+ $dbConnection = new DatabaseConnection($this->config, $this->logger);
+ $dbConnection->connect();
+ $this->sourceDb = $dbConnection->getSourceDb();
+ $this->targetDb = $dbConnection->getTargetDb();
+
+ // Initialiser les migrators
+ $this->initializeMigrators();
+ }
+
+ private function initializeMigrators(): void
+ {
+ // Créer les migrators dans l'ordre de dépendance
+ $passageMigrator = new PassageMigrator($this->sourceDb, $this->targetDb, $this->logger);
+ $sectorMigrator = new SectorMigrator($this->sourceDb, $this->targetDb, $this->logger, $passageMigrator);
+ $userMigrator = new UserMigrator($this->sourceDb, $this->targetDb, $this->logger);
+
+ $this->operationMigrator = new OperationMigrator(
+ $this->sourceDb,
+ $this->targetDb,
+ $this->logger,
+ $userMigrator,
+ $sectorMigrator
+ );
+ }
+
+ public function run(): void
+ {
+ if ($this->mode === 'entity') {
+ if (!$this->entityId) {
+ throw new Exception("entity-id requis en mode entity");
+ }
+ $this->migrateEntity($this->entityId);
+ } else {
+ $this->migrateAllEntities();
+ }
+
+ // Afficher le récapitulatif
+ if (!empty($this->migrationStats)) {
+ $this->logger->logMigrationSummary($this->migrationStats);
+ }
+
+ $this->logger->separator();
+ $this->logger->success("🎉 Migration terminée !");
+ $this->logger->info("📄 Log: " . $this->logger->getLogFile());
+ }
+
+ private function migrateEntity(int $entityId): void
+ {
+ $this->logger->separator();
+ $this->logger->info("🏢 Migration de l'entité ID: {$entityId}");
+
+ // Supprimer les données existantes si demandé
+ if ($this->deleteBefore) {
+ $this->deleteEntityData($entityId);
+ }
+
+ // Migrer l'entité elle-même
+ $this->migrateEntityRecord($entityId);
+
+ // Migrer les users de l'entité (table centrale users)
+ $this->migrateEntityUsers($entityId);
+
+ // Récupérer le nom de l'entité pour les stats
+ $stmt = $this->sourceDb->prepare("SELECT libelle FROM users_entites WHERE rowid = :id");
+ $stmt->execute([':id' => $entityId]);
+ $entityName = $stmt->fetchColumn();
+
+ // Récupérer et migrer les opérations
+ $operationIds = $this->operationMigrator->getOperationsToMigrate($entityId);
+
+ $operations = [];
+ foreach ($operationIds as $oldOperationId) {
+ $operationStats = $this->operationMigrator->migrateOperation($oldOperationId);
+ if ($operationStats) {
+ $operations[] = $operationStats;
+ }
+ }
+
+ // Stocker les stats pour cette entité
+ $this->migrationStats = [
+ 'entity' => [
+ 'id' => $entityId,
+ 'name' => $entityName ?: "Entité #{$entityId}"
+ ],
+ 'operations' => $operations
+ ];
+ }
+
+ private function migrateAllEntities(): void
+ {
+ // Récupérer toutes les entités actives
+ $stmt = $this->sourceDb->query("SELECT rowid FROM users_entites WHERE active = 1 ORDER BY rowid");
+ $entities = $stmt->fetchAll(PDO::FETCH_COLUMN);
+
+ $this->logger->info("📊 " . count($entities) . " entité(s) à migrer");
+
+ $allOperations = [];
+ foreach ($entities as $entityId) {
+ // Sauvegarder les stats actuelles avant de migrer
+ $previousStats = $this->migrationStats;
+
+ $this->migrateEntity($entityId);
+
+ // Agréger les opérations de toutes les entités
+ if (!empty($this->migrationStats['operations'])) {
+ $allOperations = array_merge($allOperations, $this->migrationStats['operations']);
+ }
+ }
+
+ // Stocker les stats globales
+ $this->migrationStats = [
+ 'operations' => $allOperations
+ ];
+ }
+
+ private function deleteEntityData(int $entityId): void
+ {
+ $this->logger->separator();
+ $this->logger->warning("🗑️ Suppression des données de l'entité {$entityId}...");
+
+ // Ordre inverse des contraintes FK
+ $tables = [
+ 'medias' => "fk_entite = {$entityId} OR fk_operation IN (SELECT id FROM operations WHERE fk_entite = {$entityId})",
+ 'ope_pass_histo' => "fk_pass IN (SELECT id FROM ope_pass WHERE fk_operation IN (SELECT id FROM operations WHERE fk_entite = {$entityId}))",
+ 'ope_pass' => "fk_operation IN (SELECT id FROM operations WHERE fk_entite = {$entityId})",
+ 'ope_users_sectors' => "fk_operation IN (SELECT id FROM operations WHERE fk_entite = {$entityId})",
+ 'ope_users' => "fk_operation IN (SELECT id FROM operations WHERE fk_entite = {$entityId})",
+ 'sectors_adresses' => "fk_sector IN (SELECT id FROM ope_sectors WHERE fk_operation IN (SELECT id FROM operations WHERE fk_entite = {$entityId}))",
+ 'ope_sectors' => "fk_operation IN (SELECT id FROM operations WHERE fk_entite = {$entityId})",
+ 'operations' => "fk_entite = {$entityId}",
+ 'users' => "fk_entite = {$entityId}"
+ ];
+
+ foreach ($tables as $table => $condition) {
+ $stmt = $this->targetDb->query("DELETE FROM {$table} WHERE {$condition}");
+ $count = $stmt->rowCount();
+ if ($count > 0) {
+ $this->logger->info(" ✓ {$table}: {$count} ligne(s) supprimée(s)");
+ }
+ }
+
+ $this->logger->success("✓ Suppression terminée");
+ }
+
+ private function migrateEntityRecord(int $entityId): void
+ {
+ // Vérifier si existe déjà
+ $stmt = $this->targetDb->prepare("SELECT COUNT(*) FROM entites WHERE id = :id");
+ $stmt->execute([':id' => $entityId]);
+
+ if ($stmt->fetchColumn() > 0) {
+ $this->logger->info("Entité {$entityId} existe déjà, skip");
+ return;
+ }
+
+ // Récupérer depuis source
+ $stmt = $this->sourceDb->prepare("SELECT * FROM users_entites WHERE rowid = :id");
+ $stmt->execute([':id' => $entityId]);
+ $entity = $stmt->fetch(PDO::FETCH_ASSOC);
+
+ if (!$entity) {
+ throw new Exception("Entité {$entityId} non trouvée");
+ }
+
+ // Insérer dans cible (schéma geo_app)
+ $stmt = $this->targetDb->prepare("
+ INSERT INTO entites (
+ id, encrypted_name, adresse1, adresse2, code_postal, ville,
+ fk_region, fk_type, encrypted_phone, encrypted_mobile, encrypted_email,
+ gps_lat, gps_lng, chk_stripe, encrypted_stripe_id, encrypted_iban, encrypted_bic,
+ chk_demo, chk_mdp_manuel, chk_username_manuel, chk_user_delete_pass,
+ chk_copie_mail_recu, chk_accept_sms, chk_lot_actif,
+ created_at, fk_user_creat, updated_at, fk_user_modif, chk_active
+ ) VALUES (
+ :id, :encrypted_name, :adresse1, :adresse2, :code_postal, :ville,
+ :fk_region, :fk_type, :encrypted_phone, :encrypted_mobile, :encrypted_email,
+ :gps_lat, :gps_lng, :chk_stripe, :encrypted_stripe_id, :encrypted_iban, :encrypted_bic,
+ :chk_demo, :chk_mdp_manuel, :chk_username_manuel, :chk_user_delete_pass,
+ :chk_copie_mail_recu, :chk_accept_sms, :chk_lot_actif,
+ :created_at, :fk_user_creat, :updated_at, :fk_user_modif, :chk_active
+ )
+ ");
+
+ $stmt->execute([
+ ':id' => $entityId,
+ ':encrypted_name' => $entity['libelle'] ? ApiService::encryptData($entity['libelle']) : '',
+ ':adresse1' => $entity['adresse1'] ?? '',
+ ':adresse2' => $entity['adresse2'] ?? '',
+ ':code_postal' => $entity['cp'] ?? '',
+ ':ville' => $entity['ville'] ?? '',
+ ':fk_region' => $entity['fk_region'],
+ ':fk_type' => $entity['fk_type'] ?? 1,
+ ':encrypted_phone' => $entity['tel1'] ? ApiService::encryptData($entity['tel1']) : '',
+ ':encrypted_mobile' => $entity['tel2'] ? ApiService::encryptData($entity['tel2']) : '',
+ ':encrypted_email' => $entity['email'] ? ApiService::encryptSearchableData($entity['email']) : '',
+ ':gps_lat' => $entity['gps_lat'] ?? '',
+ ':gps_lng' => $entity['gps_lng'] ?? '',
+ ':chk_stripe' => 0,
+ ':encrypted_stripe_id' => '',
+ ':encrypted_iban' => $entity['iban'] ? ApiService::encryptData($entity['iban']) : '',
+ ':encrypted_bic' => $entity['bic'] ? ApiService::encryptData($entity['bic']) : '',
+ ':chk_demo' => $entity['demo'] ?? 1,
+ ':chk_mdp_manuel' => $entity['chk_mdp_manuel'] ?? 0,
+ ':chk_username_manuel' => 0,
+ ':chk_user_delete_pass' => 0,
+ ':chk_copie_mail_recu' => $entity['chk_copie_mail_recu'] ?? 0,
+ ':chk_accept_sms' => $entity['chk_accept_sms'] ?? 0,
+ ':chk_lot_actif' => 0,
+ ':created_at' => date('Y-m-d H:i:s'),
+ ':fk_user_creat' => 0,
+ ':updated_at' => $entity['date_modif'],
+ ':fk_user_modif' => $entity['fk_user_modif'] ?? 0,
+ ':chk_active' => $entity['active'] ?? 1
+ ]);
+
+ $this->logger->success("✓ Entité {$entityId} migrée");
+ }
+
+ private function migrateEntityUsers(int $entityId): void
+ {
+ $stmt = $this->sourceDb->prepare("SELECT * FROM users WHERE fk_entite = :entity_id AND active = 1");
+ $stmt->execute([':entity_id' => $entityId]);
+ $users = $stmt->fetchAll(PDO::FETCH_ASSOC);
+
+ $count = 0;
+ foreach ($users as $user) {
+ // Vérifier si existe déjà
+ $stmt = $this->targetDb->prepare("SELECT COUNT(*) FROM users WHERE id = :id");
+ $stmt->execute([':id' => $user['rowid']]);
+
+ if ($stmt->fetchColumn() > 0) {
+ continue; // Skip si existe
+ }
+
+ // Insérer l'utilisateur
+ $stmt = $this->targetDb->prepare("
+ INSERT INTO users (
+ id, fk_entite, fk_role, first_name, encrypted_name,
+ encrypted_user_name, user_pass_hash, encrypted_email, encrypted_phone, encrypted_mobile,
+ created_at, fk_user_creat, updated_at, fk_user_modif, chk_active
+ ) VALUES (
+ :id, :fk_entite, :fk_role, :first_name, :encrypted_name,
+ :encrypted_user_name, :user_pass_hash, :encrypted_email, :encrypted_phone, :encrypted_mobile,
+ :created_at, :fk_user_creat, :updated_at, :fk_user_modif, :chk_active
+ )
+ ");
+
+ $stmt->execute([
+ ':id' => $user['rowid'],
+ ':fk_entite' => $user['fk_entite'],
+ ':fk_role' => $user['fk_role'],
+ ':first_name' => $user['prenom'],
+ ':encrypted_name' => ApiService::encryptData($user['libelle']), // Chiffrer avec IV aléatoire
+ ':encrypted_user_name' => ApiService::encryptSearchableData($user['username']),
+ ':user_pass_hash' => $user['userpswd'], // Hash bcrypt du mot de passe
+ ':encrypted_email' => $user['email'] ? ApiService::encryptSearchableData($user['email']) : null,
+ ':encrypted_phone' => $user['telephone'] ? ApiService::encryptData($user['telephone']) : null,
+ ':encrypted_mobile' => $user['mobile'] ? ApiService::encryptData($user['mobile']) : null,
+ ':created_at' => $user['date_creat'],
+ ':fk_user_creat' => $user['fk_user_creat'],
+ ':updated_at' => $user['date_modif'],
+ ':fk_user_modif' => $user['fk_user_modif'],
+ ':chk_active' => $user['active']
+ ]);
+
+ $count++;
+ }
+
+ $this->logger->success("✓ {$count} utilisateur(s) de l'entité migré(s)");
+ }
+
+ private function logHeader(): void
+ {
+ $this->logger->separator();
+ $this->logger->info("🚀 Migration v2 - Architecture modulaire");
+ $this->logger->info("📅 Date: " . date('Y-m-d H:i:s'));
+ $this->logger->info("🌍 Environnement: " . $this->config->getEnvName());
+ $this->logger->info("🔧 Mode: " . $this->mode);
+ if ($this->entityId) {
+ $this->logger->info("🏢 Entité: " . $this->entityId);
+ }
+ $this->logger->info("🗑️ Suppression avant: " . ($this->deleteBefore ? 'OUI' : 'NON'));
+ $this->logger->separator();
+ }
+}
+
+// === GESTION DES ARGUMENTS CLI ===
+
+function parseArguments(array $argv): array
+{
+ $options = [
+ 'env' => DatabaseConfig::autoDetect(),
+ 'mode' => 'global',
+ 'entity-id' => null,
+ 'log' => null,
+ 'delete-before' => true,
+ 'help' => false
+ ];
+
+ foreach ($argv as $arg) {
+ if ($arg === '--help') {
+ $options['help'] = true;
+ } elseif (preg_match('/^--env=(.+)$/', $arg, $matches)) {
+ $options['env'] = $matches[1];
+ } elseif (preg_match('/^--mode=(.+)$/', $arg, $matches)) {
+ $options['mode'] = $matches[1];
+ } elseif (preg_match('/^--entity-id=(\d+)$/', $arg, $matches)) {
+ $options['entity-id'] = (int)$matches[1];
+ } elseif (preg_match('/^--log=(.+)$/', $arg, $matches)) {
+ $options['log'] = $matches[1];
+ } elseif ($arg === '--delete-before=false') {
+ $options['delete-before'] = false;
+ }
+ }
+
+ return $options;
+}
+
+function showHelp(): void
+{
+ echo <<run();
+
+} catch (Exception $e) {
+ echo "❌ ERREUR: " . $e->getMessage() . "\n";
+ exit(1);
+}
diff --git a/api/scripts/migration2/php/migrate_from_backup.php.backup b/api/scripts/migration2/php/migrate_from_backup.php.backup
new file mode 100755
index 00000000..c4c91477
--- /dev/null
+++ b/api/scripts/migration2/php/migrate_from_backup.php.backup
@@ -0,0 +1,2047 @@
+#!/usr/bin/env php
+sourceDbName = $sourceDbName;
+ $this->targetDbName = $targetDbName;
+ $this->mode = $mode;
+ $this->entityId = $entityId;
+ $this->logFile = $logFile ?? '/var/back/migration_' . date('Ymd_His') . '.log';
+ $this->deleteBefore = $deleteBefore;
+
+ $this->log("=== Migration depuis backup PM7 ===");
+ $this->log("Source: {$sourceDbName}");
+ $this->log("Cible: {$targetDbName}");
+ $this->log("Mode: {$mode}");
+ if ($entityId) {
+ $this->log("Entité ID: {$entityId}");
+ }
+ if ($deleteBefore) {
+ $this->log("⚠️ Suppression des données existantes activée");
+ }
+ }
+
+ /**
+ * Connexion aux bases de données
+ */
+ public function connect() {
+ try {
+ // Connexion à la base source (backup restauré) - avec user ROOT pour accès multi-bases
+ $dsn = sprintf('mysql:host=%s;port=%d;dbname=%s;charset=utf8mb4',
+ self::DB_HOST, self::DB_PORT, $this->sourceDbName);
+ $this->sourceDb = new PDO($dsn, self::DB_USER_ROOT, self::DB_PASS_ROOT, [
+ PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION,
+ PDO::ATTR_DEFAULT_FETCH_MODE => PDO::FETCH_ASSOC,
+ PDO::ATTR_TIMEOUT => 600,
+ PDO::MYSQL_ATTR_INIT_COMMAND => "SET NAMES utf8mb4"
+ ]);
+ $this->log("✓ Connexion à la base source: {$this->sourceDbName} sur " . self::DB_HOST);
+
+ // Connexion à la base cible (pra_geo) - avec user dédié
+ $dsn = sprintf('mysql:host=%s;port=%d;dbname=%s;charset=utf8mb4',
+ self::DB_HOST, self::DB_PORT, $this->targetDbName);
+ $this->targetDb = new PDO($dsn, self::DB_USER, self::DB_PASS, [
+ PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION,
+ PDO::ATTR_DEFAULT_FETCH_MODE => PDO::FETCH_ASSOC,
+ PDO::ATTR_TIMEOUT => 600,
+ PDO::MYSQL_ATTR_INIT_COMMAND => "SET NAMES utf8mb4"
+ ]);
+ $this->log("✓ Connexion à la base cible: {$this->targetDbName} sur " . self::DB_HOST);
+
+ // Vérifier les versions MariaDB
+ $sourceVersion = $this->sourceDb->query("SELECT VERSION()")->fetchColumn();
+ $targetVersion = $this->targetDb->query("SELECT VERSION()")->fetchColumn();
+ $this->log(" Source version: $sourceVersion");
+ $this->log(" Cible version: $targetVersion");
+
+ return true;
+ } catch (PDOException $e) {
+ $this->log("✗ Erreur de connexion: " . $e->getMessage(), 'ERROR');
+ return false;
+ }
+ }
+
+ /**
+ * Suppression des données d'une entité dans la TARGET
+ * L'ordre de suppression respecte les contraintes FK (inverse de la migration)
+ */
+ private function deleteEntityData($entityId) {
+ try {
+ $this->targetDb->beginTransaction();
+
+ // Ordre de suppression inverse pour respecter les FK
+ $deletionOrder = [
+ 'medias',
+ 'ope_pass_histo',
+ 'ope_pass',
+ 'ope_users_sectors',
+ 'ope_users',
+ 'sectors_adresses',
+ 'ope_sectors',
+ 'operations',
+ 'users',
+ // NE PAS supprimer entites car peut avoir d'autres données liées
+ ];
+
+ foreach ($deletionOrder as $table) {
+ $deleted = 0;
+
+ switch ($table) {
+ case 'users':
+ $stmt = $this->targetDb->prepare("DELETE FROM $table WHERE fk_entite = ?");
+ $stmt->execute([$entityId]);
+ $deleted = $stmt->rowCount();
+ break;
+
+ case 'operations':
+ $stmt = $this->targetDb->prepare("DELETE FROM $table WHERE fk_entite = ?");
+ $stmt->execute([$entityId]);
+ $deleted = $stmt->rowCount();
+ break;
+
+ case 'sectors_adresses':
+ // Via ope_sectors -> operations
+ $stmt = $this->targetDb->prepare("
+ DELETE sa FROM $table sa
+ INNER JOIN ope_sectors s ON sa.fk_sector = s.id
+ INNER JOIN operations o ON s.fk_operation = o.id
+ WHERE o.fk_entite = ?
+ ");
+ $stmt->execute([$entityId]);
+ $deleted = $stmt->rowCount();
+ break;
+
+ case 'medias':
+ // Medias peut avoir fk_entite OU fk_operation
+ $stmt = $this->targetDb->prepare("
+ DELETE FROM $table
+ WHERE fk_entite = ?
+ OR fk_operation IN (SELECT id FROM operations WHERE fk_entite = ?)
+ ");
+ $stmt->execute([$entityId, $entityId]);
+ $deleted = $stmt->rowCount();
+ break;
+
+ case 'ope_pass_histo':
+ // Via ope_pass -> operations
+ $stmt = $this->targetDb->prepare("
+ DELETE h FROM $table h
+ INNER JOIN ope_pass p ON h.fk_pass = p.id
+ INNER JOIN operations o ON p.fk_operation = o.id
+ WHERE o.fk_entite = ?
+ ");
+ $stmt->execute([$entityId]);
+ $deleted = $stmt->rowCount();
+ break;
+
+ default:
+ // Tables avec fk_operation directe (ope_pass, ope_users, ope_users_sectors, ope_sectors)
+ $stmt = $this->targetDb->prepare("
+ DELETE t FROM $table t
+ INNER JOIN operations o ON t.fk_operation = o.id
+ WHERE o.fk_entite = ?
+ ");
+ $stmt->execute([$entityId]);
+ $deleted = $stmt->rowCount();
+ break;
+ }
+
+ if ($deleted > 0) {
+ $this->log(" ✓ $table: $deleted ligne(s) supprimée(s)");
+ }
+ }
+
+ $this->targetDb->commit();
+ $this->log("✓ Suppression terminée\n");
+
+ } catch (PDOException $e) {
+ if ($this->targetDb->inTransaction()) {
+ $this->targetDb->rollBack();
+ }
+ $this->log("✗ Erreur lors de la suppression: " . $e->getMessage(), 'ERROR');
+ throw $e;
+ }
+ }
+
+ /**
+ * Exécution de la migration
+ */
+ public function migrate() {
+ $startTime = microtime(true);
+ $totalRecords = 0;
+ $totalErrors = 0;
+
+ try {
+ if ($this->mode === 'global') {
+ // Migration globale de toutes les amicales
+ $this->log("\n=== Migration GLOBALE de toutes les amicales ===\n");
+
+ foreach ($this->migrationSteps as $step) {
+ $this->log("--- Migration: $step ---");
+ $result = $this->migrateTable($step);
+ $totalRecords += $result['success'];
+ $totalErrors += $result['errors'];
+ $this->log(" ✓ Réussis: {$result['success']}, Erreurs: {$result['errors']}");
+ }
+
+ } elseif ($this->mode === 'entity' && $this->entityId) {
+ // Migration d'une amicale spécifique
+ $this->log("\n=== Migration de l'amicale ID: {$this->entityId} ===\n");
+
+ // Vérifier que l'entité existe dans la source
+ $stmt = $this->sourceDb->prepare("SELECT rowid, libelle FROM users_entites WHERE rowid = ?");
+ $stmt->execute([$this->entityId]);
+ $entity = $stmt->fetch();
+
+ if (!$entity) {
+ $this->log("✗ Entité ID {$this->entityId} introuvable dans la source", 'ERROR');
+ return false;
+ }
+
+ $this->log("Entité trouvée: {$entity['libelle']}");
+
+ // Supprimer les données existantes si demandé
+ if ($this->deleteBefore) {
+ $this->log("\n⚠️ Suppression des données existantes de l'entité...");
+ $this->deleteEntityData($this->entityId);
+ }
+
+ // Migrer les données de l'entité
+ $entityTables = ['entites', 'users', 'operations', 'ope_sectors',
+ 'sectors_adresses', 'ope_users', 'ope_users_sectors',
+ 'ope_pass', 'ope_pass_histo', 'medias'];
+
+ // Compteurs pour le résumé
+ $stats = ['entites' => 0, 'users' => 0, 'operations' => 0, 'ope_sectors' => 0,
+ 'sectors_adresses' => 0, 'ope_users' => 0, 'ope_users_sectors' => 0,
+ 'ope_pass' => 0, 'ope_pass_histo' => 0, 'medias' => 0];
+
+ foreach ($entityTables as $table) {
+ $this->log("--- Migration: $table (entité {$this->entityId}) ---");
+ $result = $this->migrateTable($table, $this->entityId);
+ $totalRecords += $result['success'];
+ $totalErrors += $result['errors'];
+ $stats[$table] = $result['success'];
+ $this->log(" ✓ Réussis: {$result['success']}, Erreurs: {$result['errors']}");
+ }
+
+ // Ligne spéciale pour parsing par le script bash
+ // Note: sectorFromGPS sera défini dans la fonction migrateOpePass, donc on ne l'a pas ici
+ $this->log("#STATS# OPE:{$stats['operations']} USER:{$stats['users']} SECTOR:{$stats['ope_sectors']} PASS:{$stats['ope_pass']} #END#");
+ } else {
+ $this->log("✗ Mode invalide ou entity_id manquant", 'ERROR');
+ return false;
+ }
+
+ $duration = round(microtime(true) - $startTime, 2);
+
+ $this->log("\n=== Migration terminée ===");
+ $this->log("Durée totale: {$duration}s");
+ $this->log("Total enregistrements migrés: $totalRecords");
+ $this->log("Total erreurs: $totalErrors");
+
+ return true;
+
+ } catch (Exception $e) {
+ $this->log("✗ Erreur critique: " . $e->getMessage(), 'ERROR');
+ return false;
+ }
+ }
+
+ /**
+ * Migration d'une table spécifique
+ */
+ private function migrateTable($tableName, $entityId = null) {
+ try {
+ switch ($tableName) {
+ // Tables de référence (x_*)
+ case 'x_devises':
+ case 'x_entites_types':
+ case 'x_types_passages':
+ case 'x_types_reglements':
+ case 'x_users_roles':
+ case 'x_users_titres':
+ return $this->migrateReferenceTable($tableName, $entityId);
+
+ // Tables géographiques (avec FK vers x_devises, x_regions, x_pays)
+ case 'x_pays':
+ case 'x_regions':
+ case 'x_departements':
+ case 'x_villes':
+ return $this->migrateGeoTable($tableName, $entityId);
+
+ // Tables principales
+ case 'entites':
+ return $this->migrateEntites($entityId);
+
+ case 'users':
+ return $this->migrateUsers($entityId);
+
+ case 'operations':
+ return $this->migrateOperations($entityId);
+
+ case 'ope_sectors':
+ return $this->migrateOpeSectors($entityId);
+
+ case 'sectors_adresses':
+ return $this->migrateSectorsAdresses($entityId);
+
+ case 'ope_users':
+ return $this->migrateOpeUsers($entityId);
+
+ case 'ope_users_sectors':
+ return $this->migrateOpeUsersSectors($entityId);
+
+ case 'ope_pass':
+ return $this->migrateOpePass($entityId);
+
+ case 'ope_pass_histo':
+ return $this->migrateOpePassHisto($entityId);
+
+ case 'medias':
+ return $this->migrateMedias($entityId);
+
+ default:
+ $this->log(" ⚠ Table non gérée: $tableName", 'WARNING');
+ return ['success' => 0, 'errors' => 0];
+ }
+ } catch (Exception $e) {
+ $this->log(" ✗ Erreur lors de la migration de $tableName: " . $e->getMessage(), 'ERROR');
+ return ['success' => 0, 'errors' => 1];
+ }
+ }
+
+ /**
+ * Migration des tables de référence simples (sans FK)
+ */
+ private function migrateReferenceTable($tableName, $entityId = null) {
+ $success = 0;
+ $errors = 0;
+
+ try {
+ // Récupérer toutes les données de la source
+ $stmt = $this->sourceDb->query("SELECT * FROM $tableName");
+ $rows = $stmt->fetchAll();
+
+ if (empty($rows)) {
+ $this->log(" → Aucune donnée à migrer pour $tableName");
+ return ['success' => 0, 'errors' => 0];
+ }
+
+ // Préparer la requête d'insertion
+ $columns = array_keys($rows[0]);
+ $placeholders = array_map(fn($col) => ":$col", $columns);
+
+ $sql = sprintf(
+ "INSERT INTO %s (%s) VALUES (%s) ON DUPLICATE KEY UPDATE %s",
+ $tableName,
+ implode(', ', $columns),
+ implode(', ', $placeholders),
+ implode(', ', array_map(fn($col) => "$col = VALUES($col)", $columns))
+ );
+
+ $insertStmt = $this->targetDb->prepare($sql);
+
+ // Insérer chaque ligne
+ foreach ($rows as $row) {
+ try {
+ $insertStmt->execute($row);
+ $success++;
+ } catch (PDOException $e) {
+ $this->log(" ✗ Erreur insertion $tableName: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+ }
+
+ } catch (Exception $e) {
+ $this->log(" ✗ Erreur migration $tableName: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+
+ return ['success' => $success, 'errors' => $errors];
+ }
+
+ /**
+ * Compte le nombre de lignes dans une table TARGET pour une entité donnée
+ */
+ private function countTargetRows(string $tableName, $entityId = null): int {
+ try {
+ if ($entityId) {
+ // Pour entites : juste l'ID
+ if ($tableName === 'entites') {
+ $sql = "SELECT COUNT(*) as count FROM $tableName WHERE id = :entity_id";
+ $stmt = $this->targetDb->prepare($sql);
+ $stmt->execute(['entity_id' => $entityId]);
+ }
+ // Pour users : fk_entite
+ else if ($tableName === 'users') {
+ $sql = "SELECT COUNT(*) as count FROM $tableName WHERE fk_entite = :entity_id";
+ $stmt = $this->targetDb->prepare($sql);
+ $stmt->execute(['entity_id' => $entityId]);
+ }
+ // Pour operations : fk_entite directe
+ else if ($tableName === 'operations') {
+ $sql = "SELECT COUNT(*) as count FROM $tableName WHERE fk_entite = :entity_id";
+ $stmt = $this->targetDb->prepare($sql);
+ $stmt->execute(['entity_id' => $entityId]);
+ }
+ // Pour sectors_adresses : via ope_sectors
+ else if ($tableName === 'sectors_adresses') {
+ $sql = "SELECT COUNT(*) as count FROM $tableName sa
+ INNER JOIN ope_sectors s ON sa.fk_sector = s.id
+ INNER JOIN operations o ON s.fk_operation = o.id
+ WHERE o.fk_entite = :entity_id";
+ $stmt = $this->targetDb->prepare($sql);
+ $stmt->execute(['entity_id' => $entityId]);
+ }
+ // Pour ope_pass_histo : via ope_pass
+ else if ($tableName === 'ope_pass_histo') {
+ $sql = "SELECT COUNT(*) as count FROM $tableName h
+ INNER JOIN ope_pass p ON h.fk_pass = p.id
+ INNER JOIN operations o ON p.fk_operation = o.id
+ WHERE o.fk_entite = :entity_id";
+ $stmt = $this->targetDb->prepare($sql);
+ $stmt->execute(['entity_id' => $entityId]);
+ }
+ // Pour medias : peut avoir fk_entite OU fk_operation
+ else if ($tableName === 'medias') {
+ $sql = "SELECT COUNT(*) as count FROM $tableName
+ WHERE fk_entite = :entity_id
+ OR fk_operation IN (SELECT id FROM operations WHERE fk_entite = :entity_id2)";
+ $stmt = $this->targetDb->prepare($sql);
+ $stmt->execute(['entity_id' => $entityId, 'entity_id2' => $entityId]);
+ }
+ // Pour les tables avec fk_operation directe
+ else if (in_array($tableName, ['ope_sectors', 'ope_users', 'ope_users_sectors', 'ope_pass'])) {
+ $sql = "SELECT COUNT(*) as count FROM $tableName t
+ INNER JOIN operations o ON t.fk_operation = o.id
+ WHERE o.fk_entite = :entity_id";
+ $stmt = $this->targetDb->prepare($sql);
+ $stmt->execute(['entity_id' => $entityId]);
+ } else {
+ return 0;
+ }
+ } else {
+ $sql = "SELECT COUNT(*) as count FROM $tableName";
+ $stmt = $this->targetDb->prepare($sql);
+ $stmt->execute();
+ }
+
+ $result = $stmt->fetch(PDO::FETCH_ASSOC);
+ return (int)$result['count'];
+ } catch (PDOException $e) {
+ $this->log(" ⚠ Erreur comptage TARGET $tableName: " . $e->getMessage(), 'WARNING');
+ return 0;
+ }
+ }
+
+ /**
+ * Migration des tables géographiques (avec FK)
+ */
+ private function migrateGeoTable($tableName, $entityId = null) {
+ // Utiliser la même logique que les tables de référence
+ // car elles respectent déjà l'ordre des FK
+ return $this->migrateReferenceTable($tableName, $entityId);
+ }
+
+ /**
+ * Migration de la table entites (avec chiffrement)
+ */
+ private function migrateEntites($entityId = null) {
+ $success = 0;
+ $errors = 0;
+
+ try {
+ // Construire la requête de sélection
+ // Mapping: users_entites (source) → entites (cible)
+ $sql = "SELECT rowid, libelle, adresse1, adresse2, cp, ville,
+ fk_region, fk_type, tel1, tel2, email,
+ gps_lat, gps_lng, iban, bic,
+ demo, chk_mdp_manuel, chk_copie_mail_recu, chk_accept_sms,
+ date_modif, fk_user_modif, active
+ FROM users_entites";
+
+ if ($entityId) {
+ $sql .= " WHERE rowid = :entity_id";
+ }
+
+ $stmt = $this->sourceDb->prepare($sql);
+ if ($entityId) {
+ $stmt->execute(['entity_id' => $entityId]);
+ } else {
+ $stmt->execute();
+ }
+
+ $rows = $stmt->fetchAll();
+ $this->log(" → " . count($rows) . " entité(s) trouvée(s) dans SOURCE");
+
+ if (empty($rows)) {
+ $this->log(" → Aucune entité à migrer");
+ return ['success' => 0, 'errors' => 0];
+ }
+
+ // Préparer l'insertion
+ $insertSql = "INSERT INTO entites (
+ id, encrypted_name, adresse1, adresse2, code_postal, ville,
+ fk_region, fk_type, encrypted_phone, encrypted_mobile, encrypted_email,
+ gps_lat, gps_lng, chk_stripe, encrypted_stripe_id, encrypted_iban, encrypted_bic,
+ chk_demo, chk_mdp_manuel, chk_username_manuel, chk_user_delete_pass,
+ chk_copie_mail_recu, chk_accept_sms, chk_lot_actif,
+ created_at, fk_user_creat, updated_at, fk_user_modif, chk_active
+ ) VALUES (
+ :id, :encrypted_name, :adresse1, :adresse2, :code_postal, :ville,
+ :fk_region, :fk_type, :encrypted_phone, :encrypted_mobile, :encrypted_email,
+ :gps_lat, :gps_lng, :chk_stripe, :encrypted_stripe_id, :encrypted_iban, :encrypted_bic,
+ :chk_demo, :chk_mdp_manuel, :chk_username_manuel, :chk_user_delete_pass,
+ :chk_copie_mail_recu, :chk_accept_sms, :chk_lot_actif,
+ :created_at, :fk_user_creat, :updated_at, :fk_user_modif, :chk_active
+ ) ON DUPLICATE KEY UPDATE
+ encrypted_name = VALUES(encrypted_name),
+ adresse1 = VALUES(adresse1),
+ adresse2 = VALUES(adresse2),
+ code_postal = VALUES(code_postal),
+ ville = VALUES(ville),
+ fk_region = VALUES(fk_region),
+ fk_type = VALUES(fk_type),
+ encrypted_phone = VALUES(encrypted_phone),
+ encrypted_mobile = VALUES(encrypted_mobile),
+ encrypted_email = VALUES(encrypted_email),
+ updated_at = VALUES(updated_at),
+ fk_user_modif = VALUES(fk_user_modif)";
+
+ $insertStmt = $this->targetDb->prepare($insertSql);
+
+ // Insérer chaque entité
+ foreach ($rows as $row) {
+ try {
+ // Chiffrer les données sensibles
+ $encryptedName = ApiService::encryptSearchableData($row['libelle']);
+ $encryptedPhone = !empty($row['tel1']) ? ApiService::encryptData($row['tel1']) : '';
+ $encryptedMobile = !empty($row['tel2']) ? ApiService::encryptData($row['tel2']) : '';
+ $encryptedEmail = !empty($row['email']) ? ApiService::encryptSearchableData($row['email']) : '';
+ $encryptedIban = !empty($row['iban']) ? ApiService::encryptData($row['iban']) : '';
+ $encryptedBic = !empty($row['bic']) ? ApiService::encryptData($row['bic']) : '';
+
+ $insertStmt->execute([
+ 'id' => $row['rowid'],
+ 'encrypted_name' => $encryptedName,
+ 'adresse1' => $row['adresse1'] ?? '',
+ 'adresse2' => $row['adresse2'] ?? '',
+ 'code_postal' => $row['cp'] ?? '', // cp → code_postal
+ 'ville' => $row['ville'] ?? '',
+ 'fk_region' => $row['fk_region'],
+ 'fk_type' => $row['fk_type'] ?? 1,
+ 'encrypted_phone' => $encryptedPhone, // tel1
+ 'encrypted_mobile' => $encryptedMobile, // tel2
+ 'encrypted_email' => $encryptedEmail,
+ 'gps_lat' => $row['gps_lat'] ?? '',
+ 'gps_lng' => $row['gps_lng'] ?? '',
+ 'chk_stripe' => 0, // Pas dans source
+ 'encrypted_stripe_id' => '', // Pas dans source
+ 'encrypted_iban' => $encryptedIban,
+ 'encrypted_bic' => $encryptedBic,
+ 'chk_demo' => $row['demo'] ?? 0, // demo → chk_demo
+ 'chk_mdp_manuel' => $row['chk_mdp_manuel'] ?? 0,
+ 'chk_username_manuel' => 0, // Pas dans source
+ 'chk_user_delete_pass' => 0, // Pas dans source
+ 'chk_copie_mail_recu' => $row['chk_copie_mail_recu'] ?? 0,
+ 'chk_accept_sms' => $row['chk_accept_sms'] ?? 0,
+ 'chk_lot_actif' => 0, // Pas dans source
+ 'created_at' => null, // Pas date_creat dans source
+ 'fk_user_creat' => null, // Pas dans source
+ 'updated_at' => $row['date_modif'],
+ 'fk_user_modif' => $row['fk_user_modif'] ?? 0, // Peut être NULL
+ 'chk_active' => $row['active'] ?? 1
+ ]);
+
+ $success++;
+ } catch (PDOException $e) {
+ $this->log(" ✗ Erreur insertion entité {$row['rowid']}: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+ }
+
+ } catch (Exception $e) {
+ $this->log(" ✗ Erreur migration entites: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+
+ // Compter les lignes en TARGET après migration
+ $targetCount = $this->countTargetRows('entites', $entityId);
+ $sourceCount = count($rows ?? []);
+ $diff = $targetCount - $sourceCount;
+ $diffStr = $diff >= 0 ? "+$diff" : "$diff";
+
+ $this->log(" ✓ Entités migrées avec succès: $success, Erreurs: $errors");
+ $this->log(" 📊 SOURCE: $sourceCount → TARGET: $targetCount (différence: $diffStr)");
+ return ['success' => $success, 'errors' => $errors];
+ }
+
+ /**
+ * Migration de la table users (avec chiffrement et détection mobile)
+ */
+ private function migrateUsers($entityId = null) {
+ $success = 0;
+ $errors = 0;
+
+ try {
+ // Construire la requête de sélection
+ $sql = "SELECT u.rowid, u.fk_entite, u.fk_role, u.fk_titre,
+ u.libelle, u.prenom, u.nom_tournee, u.username, u.userpass,
+ u.telephone, u.mobile, u.email, u.alert_email, u.chk_suivi,
+ u.date_naissance, u.date_embauche,
+ u.date_creat, u.fk_user_creat, u.date_modif, u.fk_user_modif, u.active
+ FROM users u";
+
+ if ($entityId) {
+ $sql .= " WHERE u.fk_entite = :entity_id";
+ }
+
+ $stmt = $this->sourceDb->prepare($sql);
+ if ($entityId) {
+ $stmt->execute(['entity_id' => $entityId]);
+ } else {
+ $stmt->execute();
+ }
+
+ $rows = $stmt->fetchAll();
+ $this->log(" → " . count($rows) . " user(s) trouvé(s) dans SOURCE");
+
+ if (empty($rows)) {
+ $this->log(" → Aucun user à migrer");
+ return ['success' => 0, 'errors' => 0];
+ }
+
+ // Préparer l'insertion
+ $insertSql = "INSERT INTO users (
+ id, fk_entite, fk_role, fk_titre,
+ encrypted_name, first_name, sect_name, encrypted_user_name, user_pass_hash,
+ encrypted_phone, encrypted_mobile, encrypted_email, chk_alert_email, chk_suivi,
+ date_naissance, date_embauche,
+ created_at, fk_user_creat, updated_at, fk_user_modif, chk_active
+ ) VALUES (
+ :id, :fk_entite, :fk_role, :fk_titre,
+ :encrypted_name, :first_name, :sect_name, :encrypted_user_name, :user_pass_hash,
+ :encrypted_phone, :encrypted_mobile, :encrypted_email, :chk_alert_email, :chk_suivi,
+ :date_naissance, :date_embauche,
+ :created_at, :fk_user_creat, :updated_at, :fk_user_modif, :chk_active
+ ) ON DUPLICATE KEY UPDATE
+ fk_role = VALUES(fk_role),
+ encrypted_name = VALUES(encrypted_name),
+ first_name = VALUES(first_name),
+ encrypted_phone = VALUES(encrypted_phone),
+ encrypted_mobile = VALUES(encrypted_mobile),
+ encrypted_email = VALUES(encrypted_email),
+ updated_at = VALUES(updated_at),
+ fk_user_modif = VALUES(fk_user_modif)";
+
+ $insertStmt = $this->targetDb->prepare($insertSql);
+
+ // Insérer chaque user
+ foreach ($rows as $row) {
+ try {
+ // Chiffrer les données sensibles
+ $encryptedName = ApiService::encryptSearchableData($row['libelle']);
+ $encryptedUsername = !empty($row['username']) ? ApiService::encryptSearchableData($row['username']) : '';
+ $encryptedEmail = !empty($row['email']) ? ApiService::encryptSearchableData($row['email']) : '';
+
+ // Détection mobile vs fixe basée sur 06/07
+ $phone = $row['telephone'] ?? '';
+ $mobile = $row['mobile'] ?? '';
+
+ // Si phone commence par 06 ou 07, c'est un mobile
+ if (preg_match('/^0[67]/', $phone)) {
+ if (empty($mobile)) {
+ $mobile = $phone;
+ $phone = '';
+ }
+ }
+
+ $encryptedPhone = !empty($phone) ? ApiService::encryptData($phone) : '';
+ $encryptedMobile = !empty($mobile) ? ApiService::encryptData($mobile) : '';
+
+ $insertStmt->execute([
+ 'id' => $row['rowid'],
+ 'fk_entite' => $row['fk_entite'],
+ 'fk_role' => $row['fk_role'] ?? 1,
+ 'fk_titre' => $row['fk_titre'] ?? 1,
+ 'encrypted_name' => $encryptedName,
+ 'first_name' => $row['prenom'] ?? '',
+ 'sect_name' => $row['nom_tournee'] ?? '',
+ 'encrypted_user_name' => $encryptedUsername,
+ 'user_pass_hash' => $row['userpass'] ?? '',
+ 'encrypted_phone' => $encryptedPhone,
+ 'encrypted_mobile' => $encryptedMobile,
+ 'encrypted_email' => $encryptedEmail,
+ 'chk_alert_email' => $row['alert_email'] ?? 1,
+ 'chk_suivi' => $row['chk_suivi'] ?? 0,
+ 'date_naissance' => $row['date_naissance'],
+ 'date_embauche' => $row['date_embauche'],
+ 'created_at' => $row['date_creat'],
+ 'fk_user_creat' => $row['fk_user_creat'],
+ 'updated_at' => $row['date_modif'],
+ 'fk_user_modif' => $row['fk_user_modif'],
+ 'chk_active' => $row['active'] ?? 1
+ ]);
+
+ $success++;
+ } catch (PDOException $e) {
+ $this->log(" ✗ Erreur insertion user {$row['rowid']}: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+ }
+
+ } catch (Exception $e) {
+ $this->log(" ✗ Erreur migration users: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+
+ // Compter les lignes en TARGET après migration
+ $targetCount = $this->countTargetRows('users', $entityId);
+ $sourceCount = count($rows ?? []);
+ $diff = $targetCount - $sourceCount;
+ $diffStr = $diff >= 0 ? "+$diff" : "$diff";
+
+ $this->log(" ✓ Users migrés avec succès: $success, Erreurs: $errors");
+ $this->log(" 📊 SOURCE: $sourceCount → TARGET: $targetCount (différence: $diffStr)");
+ return ['success' => $success, 'errors' => $errors];
+ }
+
+ /**
+ * Migration de la table operations
+ */
+ private function migrateOperations($entityId = null) {
+ $success = 0;
+ $errors = 0;
+
+ try {
+ $sql = "SELECT rowid, fk_entite, libelle, date_deb, date_fin,
+ chk_distinct_sectors,
+ date_creat, fk_user_creat, date_modif, fk_user_modif, active
+ FROM operations";
+
+ if ($entityId) {
+ $sql .= " WHERE fk_entite = :entity_id";
+ }
+
+ // Ne migrer que les 3 dernières opérations (plus récentes)
+ $sql .= " ORDER BY rowid DESC LIMIT 3";
+
+ $stmt = $this->sourceDb->prepare($sql);
+ if ($entityId) {
+ $stmt->execute(['entity_id' => $entityId]);
+ } else {
+ $stmt->execute();
+ }
+
+ $rows = $stmt->fetchAll();
+ $this->log(" → " . count($rows) . " opération(s) trouvée(s) dans SOURCE");
+
+ if (empty($rows)) {
+ $this->log(" → Aucune opération à migrer");
+ return ['success' => 0, 'errors' => 0];
+ }
+
+ $insertSql = "INSERT INTO operations (
+ id, fk_entite, libelle, date_deb, date_fin, chk_distinct_sectors,
+ created_at, fk_user_creat, updated_at, fk_user_modif, chk_active
+ ) VALUES (
+ :id, :fk_entite, :libelle, :date_deb, :date_fin, :chk_distinct_sectors,
+ :created_at, :fk_user_creat, :updated_at, :fk_user_modif, :chk_active
+ ) ON DUPLICATE KEY UPDATE
+ libelle = VALUES(libelle),
+ date_deb = VALUES(date_deb),
+ date_fin = VALUES(date_fin),
+ updated_at = VALUES(updated_at)";
+
+ $insertStmt = $this->targetDb->prepare($insertSql);
+
+ foreach ($rows as $row) {
+ try {
+ $insertStmt->execute([
+ 'id' => $row['rowid'],
+ 'fk_entite' => $row['fk_entite'],
+ 'libelle' => $row['libelle'],
+ 'date_deb' => $row['date_deb'],
+ 'date_fin' => $row['date_fin'],
+ 'chk_distinct_sectors' => $row['chk_distinct_sectors'] ?? 0,
+ 'created_at' => $row['date_creat'],
+ 'fk_user_creat' => $row['fk_user_creat'] ?? 0, // 0 au lieu de NULL si vide
+ 'updated_at' => $row['date_modif'],
+ 'fk_user_modif' => $row['fk_user_modif'] ?? 0, // 0 au lieu de NULL si vide
+ 'chk_active' => $row['active'] ?? 1
+ ]);
+ $success++;
+ } catch (PDOException $e) {
+ $this->log(" ✗ Erreur insertion operation {$row['rowid']}: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+ }
+
+ } catch (Exception $e) {
+ $this->log(" ✗ Erreur migration operations: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+
+ // Compter les lignes en TARGET après migration
+ $targetCount = $this->countTargetRows('operations', $entityId);
+ $sourceCount = count($rows ?? []);
+ $diff = $targetCount - $sourceCount;
+ $diffStr = $diff >= 0 ? "+$diff" : "$diff";
+
+ $this->log(" ✓ Opérations migrées avec succès: $success, Erreurs: $errors");
+ $this->log(" 📊 SOURCE: $sourceCount → TARGET: $targetCount (différence: $diffStr)");
+ return ['success' => $success, 'errors' => $errors];
+ }
+
+ // Mapping des anciens secteurs vers les nouveaux
+ private $sectorMapping = [];
+
+ /**
+ * Récupère les IDs des 3 dernières opérations d'une entité
+ */
+ private function getLastOperationIds($entityId) {
+ $sql = "SELECT rowid FROM operations
+ WHERE fk_entite = ?
+ ORDER BY rowid DESC LIMIT 3";
+ $stmt = $this->sourceDb->prepare($sql);
+ $stmt->execute([$entityId]);
+ return array_column($stmt->fetchAll(), 'rowid');
+ }
+
+ /**
+ * Migration de la table ope_sectors
+ * IMPORTANT: La table source est "sectors" (ancien), la cible est "ope_sectors" (nouveau)
+ */
+ private function migrateOpeSectors($entityId = null) {
+ $success = 0;
+ $errors = 0;
+
+ try {
+ // Récupérer les secteurs utilisés depuis ope_users_sectors
+ // Car sectors seul ne contient pas le lien avec les opérations
+ // IMPORTANT: DISTINCT sur (fk_operation, s.rowid) car un secteur peut être réutilisé dans plusieurs opérations
+ $sql = "SELECT DISTINCT
+ ous.fk_operation,
+ s.rowid as old_sector_id,
+ s.libelle,
+ s.sector,
+ s.color,
+ s.date_creat,
+ s.fk_user_creat,
+ s.date_modif,
+ s.fk_user_modif,
+ s.active
+ FROM sectors s
+ INNER JOIN ope_users_sectors ous ON ous.fk_sector = s.rowid";
+
+ if ($entityId) {
+ // Récupérer les IDs des 3 dernières opérations
+ $operationIds = $this->getLastOperationIds($entityId);
+ if (empty($operationIds)) {
+ $this->log(" → Aucune opération trouvée pour l'entité");
+ return ['success' => 0, 'errors' => 0];
+ }
+ $operationIdsStr = implode(',', $operationIds);
+ $sql .= " INNER JOIN operations o ON ous.fk_operation = o.rowid
+ WHERE o.fk_entite = ? AND o.rowid IN ($operationIdsStr)";
+ }
+
+ $sql .= " ORDER BY ous.fk_operation, s.rowid";
+
+ $stmt = $this->sourceDb->prepare($sql);
+ if ($entityId) {
+ $stmt->execute([$entityId]);
+ } else {
+ $stmt->execute();
+ }
+
+ $rows = $stmt->fetchAll();
+ $this->log(" → " . count($rows) . " secteur(s) trouvé(s) dans SOURCE");
+
+ if (empty($rows)) {
+ $this->log(" → Aucun secteur à migrer");
+ return ['success' => 0, 'errors' => 0];
+ }
+
+ // Préparer l'insertion dans ope_sectors (NOUVEAU ID auto-increment)
+ $insertSql = "INSERT INTO ope_sectors (
+ fk_operation, fk_old_sector, libelle, sector, color,
+ created_at, fk_user_creat, updated_at, fk_user_modif, chk_active
+ ) VALUES (
+ :fk_operation, :fk_old_sector, :libelle, :sector, :color,
+ :created_at, :fk_user_creat, :updated_at, :fk_user_modif, :chk_active
+ ) ON DUPLICATE KEY UPDATE
+ libelle = VALUES(libelle),
+ sector = VALUES(sector),
+ color = VALUES(color),
+ updated_at = VALUES(updated_at)";
+
+ $insertStmt = $this->targetDb->prepare($insertSql);
+
+ foreach ($rows as $row) {
+ try {
+ $oldSectorId = $row['old_sector_id'];
+ $operationId = $row['fk_operation'];
+
+ // Vérifier si ce secteur existe déjà pour cette opération
+ $checkStmt = $this->targetDb->prepare("
+ SELECT id FROM ope_sectors
+ WHERE fk_operation = ? AND fk_old_sector = ?
+ ");
+ $checkStmt->execute([$operationId, $oldSectorId]);
+ $existing = $checkStmt->fetch();
+
+ if ($existing) {
+ // Déjà migré, utiliser l'ID existant
+ $newSectorId = $existing['id'];
+ $this->log(" → Secteur #{$oldSectorId} déjà migré (nouvel ID: {$newSectorId})");
+ } else {
+ // Insérer le nouveau secteur
+ $insertStmt->execute([
+ 'fk_operation' => $operationId,
+ 'fk_old_sector' => $oldSectorId,
+ 'libelle' => $row['libelle'],
+ 'sector' => $row['sector'],
+ 'color' => $row['color'] ?? '#4B77BE',
+ 'created_at' => $row['date_creat'],
+ 'fk_user_creat' => $row['fk_user_creat'] ?? 0, // 0 au lieu de NULL si vide
+ 'updated_at' => $row['date_modif'],
+ 'fk_user_modif' => $row['fk_user_modif'] ?? 0, // 0 au lieu de NULL si vide
+ 'chk_active' => $row['active'] ?? 1
+ ]);
+
+ // Récupérer le NOUVEAU ID auto-généré
+ $newSectorId = $this->targetDb->lastInsertId();
+ $this->log(" → Secteur #{$oldSectorId} migré (nouvel ID: {$newSectorId})");
+ }
+
+ // Stocker le mapping pour les tables suivantes
+ $mappingKey = $operationId . '_' . $oldSectorId;
+ $this->sectorMapping[$mappingKey] = $newSectorId;
+
+ $success++;
+ } catch (PDOException $e) {
+ $this->log(" ✗ Erreur insertion secteur {$row['old_sector_id']}: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+ }
+
+ $this->log(" → Mapping créé: " . count($this->sectorMapping) . " correspondances");
+
+ } catch (Exception $e) {
+ $this->log(" ✗ Erreur migration ope_sectors: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+
+ // Compter les lignes en TARGET après migration
+ $targetCount = $this->countTargetRows('ope_sectors', $entityId);
+ $sourceCount = count($rows ?? []);
+ $diff = $targetCount - $sourceCount;
+ $diffStr = $diff >= 0 ? "+$diff" : "$diff";
+
+ $this->log(" ✓ Secteurs migrés avec succès: $success, Erreurs: $errors");
+ $this->log(" 📊 SOURCE: $sourceCount → TARGET: $targetCount (différence: $diffStr)");
+ return ['success' => $success, 'errors' => $errors];
+ }
+
+ /**
+ * Migration de la table sectors_adresses
+ * IMPORTANT: Utilise le mapping créé par migrateOpeSectors()
+ */
+ private function migrateSectorsAdresses($entityId = null) {
+ $success = 0;
+ $errors = 0;
+ $skipped = 0;
+
+ try {
+ // Récupérer les adresses avec le lien vers sectors ET operations (via ope_users_sectors)
+ // NOTE: osm_id, osm_name, osm_date_creat n'existent PAS dans SOURCE, valeurs par défaut ajoutées
+ $sql = "SELECT sa.rowid, sa.fk_adresse, sa.fk_sector as old_sector_id,
+ sa.numero, sa.rue_bis, sa.rue, sa.cp, sa.ville,
+ sa.gps_lat, sa.gps_lng,
+ ous.fk_operation
+ FROM sectors_adresses sa
+ INNER JOIN ope_users_sectors ous ON ous.fk_sector = sa.fk_sector";
+
+ if ($entityId) {
+ $sql .= " INNER JOIN operations o ON ous.fk_operation = o.rowid
+ WHERE o.fk_entite = :entity_id";
+ }
+
+ $sql .= " GROUP BY sa.rowid, ous.fk_operation";
+
+ $stmt = $this->sourceDb->prepare($sql);
+ if ($entityId) {
+ $stmt->execute(['entity_id' => $entityId]);
+ } else {
+ $stmt->execute();
+ }
+
+ $rows = $stmt->fetchAll();
+ $this->log(" → " . count($rows) . " adresse(s) de secteur trouvée(s) dans SOURCE");
+
+ if (empty($rows)) {
+ $this->log(" → Aucune adresse de secteur à migrer");
+ return ['success' => 0, 'errors' => 0];
+ }
+
+ $insertSql = "INSERT INTO sectors_adresses (
+ id, fk_adresse, osm_id, fk_sector, osm_name,
+ numero, rue_bis, rue, cp, ville,
+ gps_lat, gps_lng, osm_date_creat,
+ created_at, updated_at
+ ) VALUES (
+ :id, :fk_adresse, :osm_id, :fk_sector, :osm_name,
+ :numero, :rue_bis, :rue, :cp, :ville,
+ :gps_lat, :gps_lng, :osm_date_creat,
+ :created_at, :updated_at
+ ) ON DUPLICATE KEY UPDATE
+ osm_name = VALUES(osm_name),
+ gps_lat = VALUES(gps_lat),
+ gps_lng = VALUES(gps_lng)";
+
+ $insertStmt = $this->targetDb->prepare($insertSql);
+
+ foreach ($rows as $row) {
+ try {
+ $oldSectorId = $row['old_sector_id'];
+ $operationId = $row['fk_operation'];
+
+ // Chercher le nouveau ID du secteur dans le mapping
+ $mappingKey = $operationId . '_' . $oldSectorId;
+
+ if (!isset($this->sectorMapping[$mappingKey])) {
+ $this->log(" ⚠ Adresse {$row['rowid']}: secteur {$oldSectorId} non trouvé dans mapping", 'WARNING');
+ $skipped++;
+ continue;
+ }
+
+ $newSectorId = $this->sectorMapping[$mappingKey];
+
+ $insertStmt->execute([
+ 'id' => $row['rowid'],
+ 'fk_adresse' => $row['fk_adresse'],
+ 'osm_id' => 0, // N'existe pas dans SOURCE
+ 'fk_sector' => $newSectorId, // NOUVEAU ID depuis mapping
+ 'osm_name' => '', // N'existe pas dans SOURCE
+ 'numero' => $row['numero'],
+ 'rue_bis' => $row['rue_bis'],
+ 'rue' => $row['rue'],
+ 'cp' => $row['cp'],
+ 'ville' => $row['ville'],
+ 'gps_lat' => $row['gps_lat'],
+ 'gps_lng' => $row['gps_lng'],
+ 'osm_date_creat' => null, // N'existe pas dans SOURCE
+ 'created_at' => null, // N'existe pas dans SOURCE
+ 'updated_at' => null // N'existe pas dans SOURCE
+ ]);
+ $success++;
+ } catch (PDOException $e) {
+ $this->log(" ✗ Erreur insertion sectors_adresses {$row['rowid']}: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+ }
+
+ if ($skipped > 0) {
+ $this->log(" → {$skipped} adresses ignorées (secteur non migré)");
+ }
+
+ } catch (Exception $e) {
+ $this->log(" ✗ Erreur migration sectors_adresses: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+
+ // Compter les lignes en TARGET après migration
+ $targetCount = $this->countTargetRows('sectors_adresses', $entityId);
+ $sourceCount = count($rows ?? []);
+ $diff = $targetCount - $sourceCount;
+ $diffStr = $diff >= 0 ? "+$diff" : "$diff";
+
+ $this->log(" ✓ Adresses de secteurs migrées avec succès: $success, Ignorées: $skipped, Erreurs: $errors");
+ $this->log(" 📊 SOURCE: $sourceCount → TARGET: $targetCount (différence: $diffStr)");
+ return ['success' => $success, 'errors' => $errors];
+ }
+
+ /**
+ * Migration de la table ope_users
+ */
+ private function migrateOpeUsers($entityId = null) {
+ $success = 0;
+ $errors = 0;
+
+ try {
+ // Joindre avec users pour récupérer fk_role, prenom, libelle, nom_tournee
+ // IMPORTANT: DISTINCT sur (fk_operation, fk_user) car il peut y avoir des doublons dans SOURCE
+ $sql = "SELECT DISTINCT ou.fk_operation, ou.fk_user,
+ u.fk_role, u.prenom, u.libelle, u.nom_tournee,
+ ou.date_creat, ou.fk_user_creat, ou.date_modif, ou.fk_user_modif, ou.active
+ FROM ope_users ou
+ INNER JOIN users u ON ou.fk_user = u.rowid";
+
+ if ($entityId) {
+ $operationIds = $this->getLastOperationIds($entityId);
+ if (empty($operationIds)) {
+ $this->log(" → Aucune opération trouvée pour l'entité");
+ return ['success' => 0, 'errors' => 0];
+ }
+ $operationIdsStr = implode(',', $operationIds);
+ $sql .= " WHERE ou.fk_operation IN ($operationIdsStr)";
+ }
+
+ $stmt = $this->sourceDb->prepare($sql);
+ $stmt->execute();
+
+ $rows = $stmt->fetchAll();
+ $this->log(" → " . count($rows) . " ope_user(s) trouvé(s) dans SOURCE");
+
+ if (empty($rows)) {
+ $this->log(" → Aucun ope_user à migrer");
+ return ['success' => 0, 'errors' => 0];
+ }
+
+ $insertSql = "INSERT INTO ope_users (
+ fk_operation, fk_user, fk_role,
+ first_name, encrypted_name, sect_name,
+ created_at, fk_user_creat, updated_at, fk_user_modif, chk_active
+ ) VALUES (
+ :fk_operation, :fk_user, :fk_role,
+ :first_name, :encrypted_name, :sect_name,
+ :created_at, :fk_user_creat, :updated_at, :fk_user_modif, :chk_active
+ ) ON DUPLICATE KEY UPDATE
+ updated_at = VALUES(updated_at)";
+
+ $insertStmt = $this->targetDb->prepare($insertSql);
+
+ foreach ($rows as $row) {
+ try {
+ // Vérifier que fk_user existe dans users de la TARGET
+ $checkUser = $this->targetDb->prepare("SELECT id FROM users WHERE id = ?");
+ $checkUser->execute([$row['fk_user']]);
+ if (!$checkUser->fetch()) {
+ // User n'existe pas, ignorer cet enregistrement
+ $this->log(" ⚠ OpeUser (op:{$row['fk_operation']}, user:{$row['fk_user']}): user non trouvé, ignoré", 'WARNING');
+ continue;
+ }
+
+ // Chiffrer les données sensibles depuis users
+ $encryptedName = !empty($row['libelle']) ? ApiService::encryptSearchableData($row['libelle']) : '';
+
+ $insertStmt->execute([
+ 'fk_operation' => $row['fk_operation'],
+ 'fk_user' => $row['fk_user'],
+ 'fk_role' => $row['fk_role'] ?? 1,
+ 'first_name' => $row['prenom'] ?? '',
+ 'encrypted_name' => $encryptedName,
+ 'sect_name' => $row['nom_tournee'] ?? '',
+ 'created_at' => $row['date_creat'],
+ 'fk_user_creat' => $row['fk_user_creat'],
+ 'updated_at' => $row['date_modif'],
+ 'fk_user_modif' => $row['fk_user_modif'],
+ 'chk_active' => $row['active'] ?? 1
+ ]);
+ $success++;
+ } catch (PDOException $e) {
+ $this->log(" ✗ Erreur insertion ope_user (op:{$row['fk_operation']}, user:{$row['fk_user']}): " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+ }
+
+ } catch (Exception $e) {
+ $this->log(" ✗ Erreur migration ope_users: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+
+ // Compter les lignes en TARGET après migration
+ $targetCount = $this->countTargetRows('ope_users', $entityId);
+ $sourceCount = count($rows ?? []);
+ $diff = $targetCount - $sourceCount;
+ $diffStr = $diff >= 0 ? "+$diff" : "$diff";
+
+ $this->log(" ✓ Ope_users migrés avec succès: $success, Erreurs: $errors");
+ $this->log(" 📊 SOURCE: $sourceCount → TARGET: $targetCount (différence: $diffStr)");
+ return ['success' => $success, 'errors' => $errors];
+ }
+
+ /**
+ * Migration de la table ope_users_sectors
+ * IMPORTANT: Utilise le mapping créé par migrateOpeSectors()
+ */
+ private function migrateOpeUsersSectors($entityId = null) {
+ $success = 0;
+ $errors = 0;
+ $skipped = 0;
+
+ try {
+ // NOTE: date_creat, fk_user_creat, date_modif, fk_user_modif n'existent PAS dans SOURCE
+ $sql = "SELECT ous.rowid, ous.fk_operation, ous.fk_user, ous.fk_sector as old_sector_id,
+ ous.active
+ FROM ope_users_sectors ous";
+
+ if ($entityId) {
+ $operationIds = $this->getLastOperationIds($entityId);
+ if (empty($operationIds)) {
+ $this->log(" → Aucune opération trouvée pour l'entité");
+ return ['success' => 0, 'errors' => 0];
+ }
+ $operationIdsStr = implode(',', $operationIds);
+ $sql .= " WHERE ous.fk_operation IN ($operationIdsStr)";
+ }
+
+ $stmt = $this->sourceDb->prepare($sql);
+ $stmt->execute();
+
+ $rows = $stmt->fetchAll();
+ $this->log(" → " . count($rows) . " ope_users_sector(s) trouvé(s) dans SOURCE");
+
+ if (empty($rows)) {
+ $this->log(" → Aucun ope_users_sector à migrer");
+ return ['success' => 0, 'errors' => 0];
+ }
+
+ $insertSql = "INSERT INTO ope_users_sectors (
+ id, fk_operation, fk_user, fk_sector,
+ created_at, fk_user_creat, updated_at, fk_user_modif, chk_active
+ ) VALUES (
+ :id, :fk_operation, :fk_user, :fk_sector,
+ :created_at, :fk_user_creat, :updated_at, :fk_user_modif, :chk_active
+ ) ON DUPLICATE KEY UPDATE
+ updated_at = VALUES(updated_at)";
+
+ $insertStmt = $this->targetDb->prepare($insertSql);
+
+ foreach ($rows as $row) {
+ try {
+ $oldSectorId = $row['old_sector_id'];
+ $operationId = $row['fk_operation'];
+
+ // Chercher le nouveau ID du secteur dans le mapping
+ $mappingKey = $operationId . '_' . $oldSectorId;
+
+ if (!isset($this->sectorMapping[$mappingKey])) {
+ $this->log(" ⚠ OpeUsersSector {$row['rowid']}: secteur {$oldSectorId} non trouvé dans mapping", 'WARNING');
+ $skipped++;
+ continue;
+ }
+
+ // Vérifier que fk_user existe dans users de la TARGET
+ $checkUser = $this->targetDb->prepare("SELECT id FROM users WHERE id = ?");
+ $checkUser->execute([$row['fk_user']]);
+ if (!$checkUser->fetch()) {
+ // User n'existe pas, ignorer cet enregistrement
+ $this->log(" ⚠ OpeUsersSector {$row['rowid']}: user {$row['fk_user']} non trouvé, ignoré", 'WARNING');
+ $skipped++;
+ continue;
+ }
+
+ $newSectorId = $this->sectorMapping[$mappingKey];
+
+ $insertStmt->execute([
+ 'id' => $row['rowid'],
+ 'fk_operation' => $operationId,
+ 'fk_user' => $row['fk_user'],
+ 'fk_sector' => $newSectorId, // NOUVEAU ID depuis mapping
+ 'created_at' => null, // N'existe pas dans SOURCE
+ 'fk_user_creat' => 0, // N'existe pas dans SOURCE - NOT NULL
+ 'updated_at' => null, // N'existe pas dans SOURCE
+ 'fk_user_modif' => 0, // N'existe pas dans SOURCE - NOT NULL
+ 'chk_active' => $row['active'] ?? 1
+ ]);
+ $success++;
+ } catch (PDOException $e) {
+ $this->log(" ✗ Erreur insertion ope_users_sector {$row['rowid']}: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+ }
+
+ if ($skipped > 0) {
+ $this->log(" → {$skipped} associations ignorées (secteur non migré)");
+ }
+
+ } catch (Exception $e) {
+ $this->log(" ✗ Erreur migration ope_users_sectors: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+
+ // Compter les lignes en TARGET après migration
+ $targetCount = $this->countTargetRows('ope_users_sectors', $entityId);
+ $sourceCount = count($rows ?? []);
+ $diff = $targetCount - $sourceCount;
+ $diffStr = $diff >= 0 ? "+$diff" : "$diff";
+
+ $this->log(" ✓ Ope_users_sectors migrés avec succès: $success, Ignorés: $skipped, Erreurs: $errors");
+ $this->log(" 📊 SOURCE: $sourceCount → TARGET: $targetCount (différence: $diffStr)");
+ return ['success' => $success, 'errors' => $errors];
+ }
+
+ /**
+ * Recherche un secteur par position GPS
+ * 1. Cherche si le point est contenu dans un secteur (ST_Contains)
+ * 2. Sinon, cherche le secteur le plus proche (ST_Distance)
+ * 3. Sinon, retourne le premier secteur de l'opération
+ */
+ private function findSectorByGPS($operationId, $lat, $lng) {
+ // Si GPS valide, TENTER une recherche géographique (mais ne pas bloquer si ça échoue)
+ if (!empty($lat) && !empty($lng) && $lat != 0 && $lng != 0) {
+ try {
+ // IMPORTANT : Le champ sector est stocké en BLOB, il faut le convertir avec ST_GeomFromWKB
+ // Construire le POINT avec CONCAT pour éviter le binding de paramètres
+
+ // 1. Chercher si le point est DANS un secteur
+ $sql = "SELECT id FROM ope_sectors
+ WHERE fk_operation = ?
+ AND ST_Contains(
+ ST_GeomFromWKB(sector),
+ ST_GeomFromText(CONCAT('POINT(', ?, ' ', ?, ')'), 4326)
+ )
+ LIMIT 1";
+
+ $stmt = $this->targetDb->prepare($sql);
+ $stmt->execute([$operationId, $lng, $lat]);
+ $result = $stmt->fetch();
+
+ if ($result) {
+ return $result['id'];
+ }
+
+ // 2. Sinon, chercher le secteur le PLUS PROCHE
+ $sql = "SELECT id,
+ ST_Distance(
+ ST_GeomFromWKB(sector),
+ ST_GeomFromText(CONCAT('POINT(', ?, ' ', ?, ')'), 4326)
+ ) as distance
+ FROM ope_sectors
+ WHERE fk_operation = ?
+ AND sector IS NOT NULL
+ ORDER BY distance ASC
+ LIMIT 1";
+
+ $stmt = $this->targetDb->prepare($sql);
+ $stmt->execute([$lng, $lat, $operationId]);
+ $result = $stmt->fetch();
+
+ if ($result) {
+ return $result['id'];
+ }
+ } catch (Exception $e) {
+ // La recherche GPS a échoué, on continue vers le fallback
+ // (Pas de log pour éviter de polluer avec des milliers de messages)
+ }
+ }
+
+ // 3. FALLBACK : Premier secteur de l'opération (TOUJOURS exécuté si GPS échoue)
+ try {
+ $sql = "SELECT id FROM ope_sectors
+ WHERE fk_operation = ?
+ LIMIT 1";
+
+ $stmt = $this->targetDb->prepare($sql);
+ $stmt->execute([$operationId]);
+ $result = $stmt->fetch();
+
+ return $result ? $result['id'] : null;
+ } catch (Exception $e) {
+ $this->log(" ⚠️ Erreur critique recherche secteur fallback: " . $e->getMessage(), 'ERROR');
+ return null;
+ }
+ }
+
+ /**
+ * Recherche un utilisateur valide pour un passage
+ * 1. Cherche un user affecté au secteur (via ope_users_sectors)
+ * 2. Sinon, cherche le premier user de l'opération (via ope_users)
+ */
+ private function findValidUser($operationId, $sectorId) {
+ try {
+ // 1. Chercher un user affecté au secteur
+ $sql = "SELECT fk_user FROM ope_users_sectors
+ WHERE fk_operation = :operation_id
+ AND fk_sector = :sector_id
+ LIMIT 1";
+
+ $stmt = $this->targetDb->prepare($sql);
+ $stmt->execute([
+ 'operation_id' => $operationId,
+ 'sector_id' => $sectorId
+ ]);
+ $result = $stmt->fetch();
+
+ if ($result) {
+ return $result['fk_user'];
+ }
+
+ // 2. FALLBACK : Premier user de l'opération
+ $sql = "SELECT fk_user FROM ope_users
+ WHERE fk_operation = :operation_id
+ LIMIT 1";
+
+ $stmt = $this->targetDb->prepare($sql);
+ $stmt->execute(['operation_id' => $operationId]);
+ $result = $stmt->fetch();
+
+ return $result ? $result['fk_user'] : null;
+
+ } catch (Exception $e) {
+ $this->log(" ⚠️ Erreur recherche user valide: " . $e->getMessage(), 'WARNING');
+ return null;
+ }
+ }
+
+ /**
+ * Migration de la table ope_pass (avec chiffrement)
+ */
+ private function migrateOpePass($entityId = null) {
+ $success = 0;
+ $errors = 0;
+
+ try {
+ $sql = "SELECT p.rowid, p.fk_operation, p.fk_sector, p.fk_user, p.fk_adresse,
+ p.date_eve, p.fk_type, p.numero, p.rue, p.rue_bis, p.ville, p.fk_habitat,
+ p.appt, p.niveau, p.gps_lat, p.gps_lng,
+ p.libelle, p.montant, p.fk_type_reglement, p.remarque, p.email,
+ p.recu, p.email_erreur,
+ p.chk_email_sent, p.phone, p.docremis, p.date_repasser,
+ p.nb_passages, p.chk_gps_maj, p.chk_map_create, p.chk_mobile,
+ p.chk_synchro, p.chk_api_adresse, p.chk_maj_adresse, p.anomalie,
+ p.date_creat, p.fk_user_creat, p.date_modif, p.fk_user_modif, p.active
+ FROM ope_pass p";
+
+ if ($entityId) {
+ $operationIds = $this->getLastOperationIds($entityId);
+ if (empty($operationIds)) {
+ $this->log(" → Aucune opération trouvée pour l'entité");
+ return ['success' => 0, 'errors' => 0];
+ }
+ $operationIdsStr = implode(',', $operationIds);
+ $sql .= " WHERE p.fk_operation IN ($operationIdsStr)";
+ }
+
+ $stmt = $this->sourceDb->prepare($sql);
+ $stmt->execute();
+
+ $rows = $stmt->fetchAll();
+ $this->log(" → " . count($rows) . " passage(s) trouvé(s) dans SOURCE");
+
+ if (empty($rows)) {
+ $this->log(" → Aucun passage à migrer");
+ return ['success' => 0, 'errors' => 0];
+ }
+
+ $insertSql = "INSERT INTO ope_pass (
+ id, fk_operation, fk_sector, fk_user, fk_adresse,
+ passed_at, fk_type, numero, rue, rue_bis, ville, fk_habitat,
+ appt, niveau, residence, gps_lat, gps_lng,
+ encrypted_name, montant, fk_type_reglement, remarque, encrypted_email,
+ nom_recu, date_recu, date_creat_recu, date_sent_recu, email_erreur,
+ chk_email_sent, encrypted_phone, docremis, date_repasser,
+ nb_passages, chk_gps_maj, chk_map_create, chk_mobile,
+ chk_synchro, chk_api_adresse, chk_maj_adresse, anomalie,
+ created_at, fk_user_creat, updated_at, fk_user_modif, chk_active
+ ) VALUES (
+ :id, :fk_operation, :fk_sector, :fk_user, :fk_adresse,
+ :passed_at, :fk_type, :numero, :rue, :rue_bis, :ville, :fk_habitat,
+ :appt, :niveau, :residence, :gps_lat, :gps_lng,
+ :encrypted_name, :montant, :fk_type_reglement, :remarque, :encrypted_email,
+ :nom_recu, :date_recu, :date_creat_recu, :date_sent_recu, :email_erreur,
+ :chk_email_sent, :encrypted_phone, :docremis, :date_repasser,
+ :nb_passages, :chk_gps_maj, :chk_map_create, :chk_mobile,
+ :chk_synchro, :chk_api_adresse, :chk_maj_adresse, :anomalie,
+ :created_at, :fk_user_creat, :updated_at, :fk_user_modif, :chk_active
+ ) ON DUPLICATE KEY UPDATE
+ encrypted_name = VALUES(encrypted_name),
+ montant = VALUES(montant),
+ encrypted_email = VALUES(encrypted_email),
+ encrypted_phone = VALUES(encrypted_phone),
+ updated_at = VALUES(updated_at)";
+
+ $insertStmt = $this->targetDb->prepare($insertSql);
+
+ // Traiter par lots de 1000
+ $batchSize = 1000;
+ $total = count($rows);
+ $batches = ceil($total / $batchSize);
+
+ // Compteurs détaillés
+ $sectorFromMapping = 0;
+ $sectorFromGPS = 0;
+ $sectorFallback = 0;
+ $userOriginal = 0;
+ $userReplaced = 0;
+
+ for ($i = 0; $i < $batches; $i++) {
+ $offset = $i * $batchSize;
+ $batch = array_slice($rows, $offset, $batchSize);
+
+ foreach ($batch as $row) {
+ try {
+ // ========== SECTEUR : Garantir un secteur valide ==========
+ $oldSectorId = $row['fk_sector'];
+ $operationId = $row['fk_operation'];
+ $mappingKey = $operationId . '_' . $oldSectorId;
+
+ $newSectorId = null;
+
+ // 1. Essayer avec le mapping
+ if (isset($this->sectorMapping[$mappingKey])) {
+ $newSectorId = $this->sectorMapping[$mappingKey];
+ $sectorFromMapping++;
+ }
+ // 2. Sinon, recherche géographique (avec fallback intégré)
+ else {
+ $newSectorId = $this->findSectorByGPS($operationId, $row['gps_lat'], $row['gps_lng']);
+ if ($newSectorId) {
+ // Si GPS valides, c'est une vraie recherche GPS
+ if (!empty($row['gps_lat']) && !empty($row['gps_lng'])) {
+ $sectorFromGPS++;
+ } else {
+ // Sinon, c'est le fallback (premier secteur)
+ $sectorFallback++;
+ }
+ }
+ }
+
+ // Si VRAIMENT aucun secteur (opération sans secteur ??)
+ if (!$newSectorId) {
+ $this->log(" ⚠️ Passage #{$row['rowid']} : opération {$operationId} sans secteur, IGNORÉ", 'WARNING');
+ $errors++;
+ continue;
+ }
+
+ // ========== UTILISATEUR : Garantir un user valide ==========
+ $fkUser = $row['fk_user'];
+
+ // Vérifier que fk_user existe dans users de la TARGET
+ if ($fkUser > 0) {
+ $checkUser = $this->targetDb->prepare("SELECT id FROM users WHERE id = ?");
+ $checkUser->execute([$fkUser]);
+ if ($checkUser->fetch()) {
+ // User valide
+ $userOriginal++;
+ } else {
+ // User n'existe pas, chercher un remplaçant
+ $replacementUser = $this->findValidUser($operationId, $newSectorId);
+ if ($replacementUser) {
+ $fkUser = $replacementUser;
+ $userReplaced++;
+ } else {
+ $this->log(" ⚠️ Passage #{$row['rowid']} : aucun user trouvé pour opération {$operationId}, IGNORÉ", 'WARNING');
+ $errors++;
+ continue;
+ }
+ }
+ } else {
+ // fk_user = 0 ou NULL, chercher un user valide
+ $replacementUser = $this->findValidUser($operationId, $newSectorId);
+ if ($replacementUser) {
+ $fkUser = $replacementUser;
+ $userReplaced++;
+ } else {
+ $this->log(" ⚠️ Passage #{$row['rowid']} : aucun user trouvé pour opération {$operationId}, IGNORÉ", 'WARNING');
+ $errors++;
+ continue;
+ }
+ }
+
+ // Chiffrer les données sensibles
+ $encryptedName = ApiService::encryptSearchableData($row['libelle']);
+ $encryptedEmail = !empty($row['email']) ? ApiService::encryptSearchableData($row['email']) : '';
+
+ // Détection mobile vs fixe
+ $phone = $row['phone'] ?? '';
+ $mobile = '';
+
+ if (preg_match('/^0[67]/', $phone)) {
+ $mobile = $phone;
+ $phone = '';
+ }
+
+ $encryptedPhone = !empty($mobile) ? ApiService::encryptData($mobile) : '';
+
+ // Vérification et correction du type de règlement (comme migrate_ope_pass.php)
+ $fkTypeReglement = $row['fk_type_reglement'] ?? 1;
+ if (!in_array($fkTypeReglement, [1, 2, 3])) {
+ $fkTypeReglement = 4; // Forcer à 4 ("-") si différent de 1, 2 ou 3
+ }
+
+ $insertStmt->execute([
+ 'id' => $row['rowid'],
+ 'fk_operation' => $operationId,
+ 'fk_sector' => $newSectorId, // NOUVEAU ID depuis mapping ou GPS
+ 'fk_user' => $fkUser, // User validé ou remplacé
+ 'fk_adresse' => $row['fk_adresse'],
+ 'passed_at' => $row['date_eve'],
+ 'fk_type' => $row['fk_type'],
+ 'numero' => $row['numero'],
+ 'rue' => $row['rue'],
+ 'rue_bis' => $row['rue_bis'],
+ 'ville' => $row['ville'],
+ 'fk_habitat' => $row['fk_habitat'] ?? 1,
+ 'appt' => $row['appt'],
+ 'niveau' => $row['niveau'],
+ 'residence' => '',
+ 'gps_lat' => $row['gps_lat'],
+ 'gps_lng' => $row['gps_lng'],
+ 'encrypted_name' => $encryptedName,
+ 'montant' => $row['montant'],
+ 'fk_type_reglement' => $fkTypeReglement,
+ 'remarque' => $row['remarque'],
+ 'encrypted_email' => $encryptedEmail,
+ 'nom_recu' => $row['recu'],
+ 'date_recu' => null,
+ 'date_creat_recu' => null,
+ 'date_sent_recu' => null,
+ 'email_erreur' => $row['email_erreur'],
+ 'chk_email_sent' => $row['chk_email_sent'] ?? 0,
+ 'encrypted_phone' => $encryptedPhone,
+ 'docremis' => $row['docremis'] ?? 0,
+ 'date_repasser' => $row['date_repasser'],
+ 'nb_passages' => $row['nb_passages'] ?? 1,
+ 'chk_gps_maj' => $row['chk_gps_maj'] ?? 0,
+ 'chk_map_create' => $row['chk_map_create'] ?? 0,
+ 'chk_mobile' => $row['chk_mobile'] ?? 0,
+ 'chk_synchro' => $row['chk_synchro'] ?? 1,
+ 'chk_api_adresse' => $row['chk_api_adresse'] ?? 0,
+ 'chk_maj_adresse' => $row['chk_maj_adresse'] ?? 0,
+ 'anomalie' => $row['anomalie'] ?? 0,
+ 'created_at' => $row['date_creat'],
+ 'fk_user_creat' => $row['fk_user_creat'],
+ 'updated_at' => $row['date_modif'],
+ 'fk_user_modif' => $row['fk_user_modif'],
+ 'chk_active' => $row['active'] ?? 1
+ ]);
+ $success++;
+ } catch (PDOException $e) {
+ $this->log(" ✗ Erreur insertion ope_pass {$row['rowid']}: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+ }
+
+ // Log progression
+ $processed = min(($i + 1) * $batchSize, $total);
+ $this->log(" → Progression: $processed / $total passages");
+ }
+
+ } catch (Exception $e) {
+ $this->log(" ✗ Erreur migration ope_pass: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+
+ // Compter les lignes en TARGET après migration
+ $targetCount = $this->countTargetRows('ope_pass', $entityId);
+ $sourceCount = count($rows ?? []);
+ $diff = $targetCount - $sourceCount;
+ $diffStr = $diff >= 0 ? "+$diff" : "$diff";
+
+ // Afficher les détails de récupération
+ if (isset($sectorFromMapping) && isset($sectorFromGPS) && isset($sectorFallback)) {
+ $this->log(" 🎯 Secteurs: mapping=$sectorFromMapping | GPS=$sectorFromGPS | fallback=$sectorFallback");
+ }
+ if (isset($userOriginal) && isset($userReplaced)) {
+ $this->log(" 👤 Users: originaux=$userOriginal | remplacés=$userReplaced");
+ }
+
+ $this->log(" ✓ Passages migrés avec succès: $success, Erreurs: $errors");
+ $this->log(" 📊 SOURCE: $sourceCount → TARGET: $targetCount (différence: $diffStr)");
+ return ['success' => $success, 'errors' => $errors];
+ }
+
+ /**
+ * Migration de la table ope_pass_histo
+ */
+ private function migrateOpePassHisto($entityId = null) {
+ $success = 0;
+ $errors = 0;
+
+ try {
+ $sql = "SELECT h.rowid, h.fk_pass, h.date_histo, h.sujet, h.remarque
+ FROM ope_pass_histo h";
+
+ if ($entityId) {
+ $operationIds = $this->getLastOperationIds($entityId);
+ if (empty($operationIds)) {
+ $this->log(" → Aucune opération trouvée pour l'entité");
+ return ['success' => 0, 'errors' => 0];
+ }
+ $operationIdsStr = implode(',', $operationIds);
+ $sql .= " INNER JOIN ope_pass p ON h.fk_pass = p.rowid
+ WHERE p.fk_operation IN ($operationIdsStr)";
+ }
+
+ $stmt = $this->sourceDb->prepare($sql);
+ $stmt->execute();
+
+ $rows = $stmt->fetchAll();
+ $this->log(" → " . count($rows) . " historique(s) de passage trouvé(s) dans SOURCE");
+
+ if (empty($rows)) {
+ $this->log(" → Aucun historique de passage à migrer");
+ return ['success' => 0, 'errors' => 0];
+ }
+
+ $insertSql = "INSERT INTO ope_pass_histo (
+ id, fk_pass, date_histo, sujet, remarque
+ ) VALUES (
+ :id, :fk_pass, :date_histo, :sujet, :remarque
+ ) ON DUPLICATE KEY UPDATE
+ sujet = VALUES(sujet),
+ remarque = VALUES(remarque)";
+
+ $insertStmt = $this->targetDb->prepare($insertSql);
+
+ foreach ($rows as $row) {
+ try {
+ $insertStmt->execute([
+ 'id' => $row['rowid'],
+ 'fk_pass' => $row['fk_pass'],
+ 'date_histo' => $row['date_histo'],
+ 'sujet' => $row['sujet'],
+ 'remarque' => $row['remarque']
+ ]);
+ $success++;
+ } catch (PDOException $e) {
+ $this->log(" ✗ Erreur insertion ope_pass_histo {$row['rowid']}: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+ }
+
+ } catch (Exception $e) {
+ $this->log(" ✗ Erreur migration ope_pass_histo: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+
+ // Compter les lignes en TARGET après migration
+ $targetCount = $this->countTargetRows('ope_pass_histo', $entityId);
+ $sourceCount = count($rows ?? []);
+ $diff = $targetCount - $sourceCount;
+ $diffStr = $diff >= 0 ? "+$diff" : "$diff";
+
+ $this->log(" ✓ Historiques de passages migrés avec succès: $success, Erreurs: $errors");
+ $this->log(" 📊 SOURCE: $sourceCount → TARGET: $targetCount (différence: $diffStr)");
+ return ['success' => $success, 'errors' => $errors];
+ }
+
+ /**
+ * Migration de la table medias
+ */
+ private function migrateMedias($entityId = null) {
+ $success = 0;
+ $errors = 0;
+
+ try {
+ // NOTE: SOURCE a support_rowid (pas support_id), type_fichier (pas file_type), hauteur/largeur (pas original_width/height)
+ // Beaucoup de champs n'existent pas dans SOURCE (fk_entite, fk_operation, file_category, etc.)
+ $sql = "SELECT m.rowid, m.support, m.support_rowid, m.fichier, m.type_fichier,
+ m.description, m.hauteur as original_height, m.largeur as original_width,
+ m.date_creat, m.fk_user_creat, m.date_modif, m.fk_user_modif
+ FROM medias m";
+
+ if ($entityId) {
+ $operationIds = $this->getLastOperationIds($entityId);
+ if (empty($operationIds)) {
+ $this->log(" → Aucune opération trouvée pour l'entité");
+ return ['success' => 0, 'errors' => 0];
+ }
+ $operationIdsStr = implode(',', $operationIds);
+ // Filtrer les médias liés aux ope_pass des 3 dernières opérations
+ $sql .= " WHERE (m.support = 'ope_pass' AND m.support_rowid IN (
+ SELECT rowid FROM ope_pass WHERE fk_operation IN ($operationIdsStr)
+ ))";
+ }
+
+ $stmt = $this->sourceDb->prepare($sql);
+ $stmt->execute();
+
+ $rows = $stmt->fetchAll();
+ $this->log(" → " . count($rows) . " media(s) trouvé(s) dans SOURCE");
+
+ if (empty($rows)) {
+ $this->log(" → Aucun media à migrer");
+ return ['success' => 0, 'errors' => 0];
+ }
+
+ $insertSql = "INSERT INTO medias (
+ id, support, support_id, fichier, file_type, file_category,
+ file_size, mime_type, original_name, fk_entite, fk_operation,
+ file_path, original_width, original_height, processed_width, processed_height,
+ is_processed, description,
+ created_at, fk_user_creat, updated_at, fk_user_modif
+ ) VALUES (
+ :id, :support, :support_id, :fichier, :file_type, :file_category,
+ :file_size, :mime_type, :original_name, :fk_entite, :fk_operation,
+ :file_path, :original_width, :original_height, :processed_width, :processed_height,
+ :is_processed, :description,
+ :created_at, :fk_user_creat, :updated_at, :fk_user_modif
+ ) ON DUPLICATE KEY UPDATE
+ file_path = VALUES(file_path),
+ updated_at = VALUES(updated_at)";
+
+ $insertStmt = $this->targetDb->prepare($insertSql);
+
+ foreach ($rows as $row) {
+ try {
+ $insertStmt->execute([
+ 'id' => $row['rowid'],
+ 'support' => $row['support'],
+ 'support_id' => $row['support_rowid'], // SOURCE utilise support_rowid
+ 'fichier' => $row['fichier'],
+ 'file_type' => $row['type_fichier'], // SOURCE utilise type_fichier
+ 'file_category' => null, // N'existe pas dans SOURCE
+ 'file_size' => null, // N'existe pas dans SOURCE
+ 'mime_type' => null, // N'existe pas dans SOURCE
+ 'original_name' => null, // N'existe pas dans SOURCE
+ 'fk_entite' => null, // N'existe pas dans SOURCE
+ 'fk_operation' => null, // N'existe pas dans SOURCE
+ 'file_path' => null, // N'existe pas dans SOURCE
+ 'original_width' => $row['original_width'], // Mappé depuis largeur
+ 'original_height' => $row['original_height'], // Mappé depuis hauteur
+ 'processed_width' => null, // N'existe pas dans SOURCE
+ 'processed_height' => null, // N'existe pas dans SOURCE
+ 'is_processed' => 0, // N'existe pas dans SOURCE
+ 'description' => $row['description'],
+ 'created_at' => $row['date_creat'],
+ 'fk_user_creat' => $row['fk_user_creat'],
+ 'updated_at' => $row['date_modif'],
+ 'fk_user_modif' => $row['fk_user_modif']
+ ]);
+ $success++;
+ } catch (PDOException $e) {
+ $this->log(" ✗ Erreur insertion media {$row['rowid']}: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+ }
+
+ } catch (Exception $e) {
+ $this->log(" ✗ Erreur migration medias: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+
+ // Compter les lignes en TARGET après migration
+ $targetCount = $this->countTargetRows('medias', $entityId);
+ $sourceCount = count($rows ?? []);
+ $diff = $targetCount - $sourceCount;
+ $diffStr = $diff >= 0 ? "+$diff" : "$diff";
+
+ $this->log(" ✓ Medias migrés avec succès: $success, Erreurs: $errors");
+ $this->log(" 📊 SOURCE: $sourceCount → TARGET: $targetCount (différence: $diffStr)");
+ return ['success' => $success, 'errors' => $errors];
+ }
+
+ /**
+ * Logging
+ */
+ private function log($message, $level = 'INFO') {
+ $timestamp = date('Y-m-d H:i:s');
+ $logMessage = "[{$timestamp}] [{$level}] {$message}\n";
+
+ // Afficher dans la console
+ echo $logMessage;
+
+ // Écrire dans le fichier de log
+ file_put_contents($this->logFile, $logMessage, FILE_APPEND);
+ }
+}
+
+// === Fonctions globales pour compatibilité avec les scripts existants ===
+
+/**
+ * Retourne la connexion à la base source (remplace celle de config.php)
+ */
+function getSourceConnection() {
+ return $GLOBALS['__MIGRATION_SOURCE_DB'] ?? null;
+}
+
+/**
+ * Retourne la connexion à la base cible (remplace celle de config.php)
+ */
+function getTargetConnection() {
+ return $GLOBALS['__MIGRATION_TARGET_DB'] ?? null;
+}
+
+/**
+ * Pas de tunnel SSH nécessaire (bases locales)
+ */
+function createSshTunnel() {
+ // Rien à faire, les bases sont locales
+ return true;
+}
+
+/**
+ * Pas de tunnel SSH à fermer
+ */
+function closeSshTunnel() {
+ // Rien à faire
+}
+
+/**
+ * Logging simple
+ */
+function logOperation($message, $level = 'INFO') {
+ echo "[{$level}] {$message}\n";
+}
+
+// === Parse des arguments CLI ===
+
+function parseArguments($argv) {
+ $args = [
+ 'source-db' => null,
+ 'target-db' => 'pra_geo',
+ 'mode' => 'global',
+ 'entity-id' => null,
+ 'log' => null,
+ 'delete-before' => true,
+ 'help' => false
+ ];
+
+ foreach ($argv as $arg) {
+ if (strpos($arg, '--') === 0) {
+ $parts = explode('=', substr($arg, 2), 2);
+ $key = $parts[0];
+ $value = $parts[1] ?? true;
+
+ if (array_key_exists($key, $args)) {
+ $args[$key] = $value;
+ }
+ }
+ }
+
+ return $args;
+}
+
+function showHelp() {
+ echo <<connect()) {
+ exit(1);
+}
+
+$success = $migration->migrate();
+
+exit($success ? 0 : 1);
diff --git a/api/scripts/migrations/add_email_queue_fields.sql b/api/scripts/migrations/add_email_queue_fields.sql
deleted file mode 100644
index b4018161..00000000
--- a/api/scripts/migrations/add_email_queue_fields.sql
+++ /dev/null
@@ -1,57 +0,0 @@
--- Migration : Ajout des champs manquants dans email_queue
--- Date : 2025-01-06
--- Description : Ajoute sent_at et error_message pour le bon fonctionnement du CRON
-
-USE geo_app;
-
--- Vérifier si les champs existent déjà avant de les ajouter
-SET @db_name = DATABASE();
-SET @table_name = 'email_queue';
-
--- Ajouter sent_at si n'existe pas
-SET @column_exists = (
- SELECT COUNT(*)
- FROM information_schema.COLUMNS
- WHERE TABLE_SCHEMA = @db_name
- AND TABLE_NAME = @table_name
- AND COLUMN_NAME = 'sent_at'
-);
-
-SET @sql = IF(@column_exists = 0,
- 'ALTER TABLE email_queue ADD COLUMN sent_at TIMESTAMP NULL DEFAULT NULL AFTER status',
- 'SELECT "Column sent_at already exists" AS message'
-);
-
-PREPARE stmt FROM @sql;
-EXECUTE stmt;
-DEALLOCATE PREPARE stmt;
-
--- Ajouter error_message si n'existe pas
-SET @column_exists = (
- SELECT COUNT(*)
- FROM information_schema.COLUMNS
- WHERE TABLE_SCHEMA = @db_name
- AND TABLE_NAME = @table_name
- AND COLUMN_NAME = 'error_message'
-);
-
-SET @sql = IF(@column_exists = 0,
- 'ALTER TABLE email_queue ADD COLUMN error_message TEXT NULL DEFAULT NULL AFTER attempts',
- 'SELECT "Column error_message already exists" AS message'
-);
-
-PREPARE stmt FROM @sql;
-EXECUTE stmt;
-DEALLOCATE PREPARE stmt;
-
--- Vérifier le résultat
-SELECT
- 'Migration terminée' AS status,
- COLUMN_NAME,
- COLUMN_TYPE,
- IS_NULLABLE,
- COLUMN_DEFAULT
-FROM information_schema.COLUMNS
-WHERE TABLE_SCHEMA = @db_name
-AND TABLE_NAME = @table_name
-AND COLUMN_NAME IN ('sent_at', 'error_message');
diff --git a/api/scripts/migrations/migrate_stripe_payment_id.sql b/api/scripts/migrations/migrate_stripe_payment_id.sql
deleted file mode 100644
index d3f6d3f1..00000000
--- a/api/scripts/migrations/migrate_stripe_payment_id.sql
+++ /dev/null
@@ -1,94 +0,0 @@
--- =====================================================
--- Migration Stripe : is_striped → stripe_payment_id
--- Date : Janvier 2025
--- Description : Refactoring pour simplifier la gestion des paiements Stripe
--- =====================================================
-
--- 1. Modifier la table ope_pass
--- ------------------------------
-ALTER TABLE `ope_pass` DROP COLUMN IF EXISTS `chk_striped`;
-ALTER TABLE `ope_pass` ADD COLUMN `stripe_payment_id` VARCHAR(50) DEFAULT NULL COMMENT 'ID du PaymentIntent Stripe (pi_xxx)';
-ALTER TABLE `ope_pass` ADD INDEX `idx_stripe_payment` (`stripe_payment_id`);
-
--- 2. Modifier stripe_payment_history pour la rendre indépendante
--- ----------------------------------------------------------------
--- Supprimer la clé étrangère vers stripe_payment_intents
-ALTER TABLE `stripe_payment_history`
-DROP FOREIGN KEY IF EXISTS `stripe_payment_history_ibfk_1`;
-
--- Modifier la colonne pour stocker directement l'ID Stripe (totalement indépendante)
-ALTER TABLE `stripe_payment_history`
-DROP INDEX IF EXISTS `idx_fk_payment_intent`,
-CHANGE COLUMN `fk_payment_intent` `stripe_payment_intent_id` VARCHAR(255) DEFAULT NULL COMMENT 'ID du PaymentIntent Stripe',
-ADD INDEX `idx_stripe_payment_intent_id` (`stripe_payment_intent_id`);
-
--- 3. Modifier stripe_refunds pour la rendre indépendante
--- --------------------------------------------------------
-ALTER TABLE `stripe_refunds`
-DROP FOREIGN KEY IF EXISTS `stripe_refunds_ibfk_1`;
-
--- Modifier la colonne pour stocker directement l'ID Stripe (totalement indépendante)
-ALTER TABLE `stripe_refunds`
-DROP INDEX IF EXISTS `idx_fk_payment_intent`,
-CHANGE COLUMN `fk_payment_intent` `stripe_payment_intent_id` VARCHAR(255) NOT NULL COMMENT 'ID du PaymentIntent Stripe',
-ADD INDEX `idx_stripe_payment_intent_id` (`stripe_payment_intent_id`);
-
--- 4. Supprimer la vue qui dépend de stripe_payment_intents
--- ----------------------------------------------------------
-DROP VIEW IF EXISTS `v_stripe_payment_stats`;
-
--- 5. Supprimer la table stripe_payment_intents
--- ---------------------------------------------
-DROP TABLE IF EXISTS `stripe_payment_intents`;
-
--- 6. Créer une nouvelle vue basée sur ope_pass
--- ----------------------------------------------
-CREATE OR REPLACE VIEW `v_stripe_payment_stats` AS
-SELECT
- o.fk_entite,
- e.encrypted_name as entite_name,
- p.fk_user,
- CONCAT(u.first_name, ' ', u.sect_name) as user_name,
- COUNT(DISTINCT p.id) as total_ventes,
- COUNT(DISTINCT CASE WHEN p.stripe_payment_id IS NOT NULL THEN p.id END) as ventes_stripe,
- SUM(CASE WHEN p.stripe_payment_id IS NOT NULL THEN p.montant ELSE 0 END) as montant_stripe,
- SUM(CASE WHEN p.stripe_payment_id IS NULL THEN p.montant ELSE 0 END) as montant_autres,
- DATE(p.created_at) as date_vente
-FROM ope_pass p
-LEFT JOIN operations o ON p.fk_operation = o.id
-LEFT JOIN entites e ON o.fk_entite = e.id
-LEFT JOIN users u ON p.fk_user = u.id
-WHERE p.fk_type = 2 -- Type vente calendrier
-GROUP BY o.fk_entite, p.fk_user, DATE(p.created_at);
-
--- 7. Vue pour les statistiques par entité uniquement
--- ----------------------------------------------------
-CREATE OR REPLACE VIEW `v_stripe_entite_stats` AS
-SELECT
- e.id as entite_id,
- e.encrypted_name as entite_name,
- sa.stripe_account_id,
- sa.charges_enabled,
- sa.payouts_enabled,
- COUNT(DISTINCT p.id) as total_passages,
- COUNT(DISTINCT CASE WHEN p.stripe_payment_id IS NOT NULL THEN p.id END) as passages_stripe,
- SUM(CASE WHEN p.stripe_payment_id IS NOT NULL THEN p.montant ELSE 0 END) as revenue_stripe,
- SUM(p.montant) as revenue_total
-FROM entites e
-LEFT JOIN stripe_accounts sa ON e.id = sa.fk_entite
-LEFT JOIN operations o ON e.id = o.fk_entite
-LEFT JOIN ope_pass p ON o.id = p.fk_operation
-GROUP BY e.id, e.encrypted_name, sa.stripe_account_id;
-
--- 8. Fonction helper pour vérifier si un passage a un paiement Stripe
--- ---------------------------------------------------------------------
--- NOTE: Si vous exécutez en copier/coller, cette fonction est optionnelle
--- Vous pouvez l'ignorer ou l'exécuter séparément avec DELIMITER
-
--- =====================================================
--- FIN DE LA MIGRATION
--- =====================================================
--- Tables supprimées : stripe_payment_intents
--- Tables modifiées : ope_pass, stripe_payment_history, stripe_refunds
--- Tables conservées : stripe_accounts, stripe_terminal_readers, etc.
--- =====================================================
\ No newline at end of file
diff --git a/api/scripts/migrations/stripe_tables.sql b/api/scripts/migrations/stripe_tables.sql
deleted file mode 100644
index 3672b829..00000000
--- a/api/scripts/migrations/stripe_tables.sql
+++ /dev/null
@@ -1,197 +0,0 @@
--- =============================================================
--- Tables pour l'intégration Stripe Connect + Terminal
--- Date: 2025-09-01
--- Version: 1.0.0
--- Préfixe: stripe_
--- =============================================================
-
--- Table pour stocker les comptes Stripe Connect des amicales
-CREATE TABLE IF NOT EXISTS stripe_accounts (
- id INT(10) UNSIGNED PRIMARY KEY AUTO_INCREMENT,
- fk_entite INT(10) UNSIGNED NOT NULL,
- stripe_account_id VARCHAR(255) UNIQUE,
- stripe_location_id VARCHAR(255),
- charges_enabled BOOLEAN DEFAULT FALSE,
- payouts_enabled BOOLEAN DEFAULT FALSE,
- onboarding_completed BOOLEAN DEFAULT FALSE,
- created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
- updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
- FOREIGN KEY (fk_entite) REFERENCES entites(id) ON DELETE CASCADE,
- INDEX idx_fk_entite (fk_entite),
- INDEX idx_stripe_account_id (stripe_account_id)
-) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-
--- Table pour stocker les intentions de paiement
-CREATE TABLE IF NOT EXISTS stripe_payment_intents (
- id INT(10) UNSIGNED PRIMARY KEY AUTO_INCREMENT,
- stripe_payment_intent_id VARCHAR(255) UNIQUE,
- fk_entite INT(10) UNSIGNED NOT NULL,
- fk_user INT(10) UNSIGNED NOT NULL,
- amount INT NOT NULL COMMENT 'Montant en centimes',
- currency VARCHAR(3) DEFAULT 'eur',
- status VARCHAR(50),
- application_fee INT COMMENT 'Commission en centimes',
- metadata JSON,
- created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
- updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
- FOREIGN KEY (fk_entite) REFERENCES entites(id) ON DELETE CASCADE,
- FOREIGN KEY (fk_user) REFERENCES users(id) ON DELETE CASCADE,
- INDEX idx_fk_entite (fk_entite),
- INDEX idx_fk_user (fk_user),
- INDEX idx_status (status),
- INDEX idx_created_at (created_at)
-) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-
--- Table pour les readers Terminal (Tap to Pay virtuel)
-CREATE TABLE IF NOT EXISTS stripe_terminal_readers (
- id INT(10) UNSIGNED PRIMARY KEY AUTO_INCREMENT,
- stripe_reader_id VARCHAR(255) UNIQUE,
- fk_entite INT(10) UNSIGNED NOT NULL,
- label VARCHAR(255),
- location VARCHAR(255),
- status VARCHAR(50),
- device_type VARCHAR(50) COMMENT 'ios_tap_to_pay, android_tap_to_pay',
- device_info JSON COMMENT 'Infos sur le device (modèle, OS, etc)',
- last_seen_at TIMESTAMP NULL,
- created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
- updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
- FOREIGN KEY (fk_entite) REFERENCES entites(id) ON DELETE CASCADE,
- INDEX idx_fk_entite (fk_entite),
- INDEX idx_device_type (device_type)
-) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-
--- Table pour les appareils Android certifiés Tap to Pay
-CREATE TABLE IF NOT EXISTS stripe_android_certified_devices (
- id INT(10) UNSIGNED PRIMARY KEY AUTO_INCREMENT,
- manufacturer VARCHAR(100),
- model VARCHAR(200),
- model_identifier VARCHAR(200),
- tap_to_pay_certified BOOLEAN DEFAULT FALSE,
- certification_date DATE,
- min_android_version INT,
- country VARCHAR(2) DEFAULT 'FR',
- notes TEXT,
- last_verified TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
- created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
- updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
- INDEX idx_manufacturer_model (manufacturer, model),
- INDEX idx_certified (tap_to_pay_certified, country),
- UNIQUE KEY unique_device (manufacturer, model, model_identifier)
-) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-
--- Table pour l'historique des paiements (pour audit et réconciliation)
-CREATE TABLE IF NOT EXISTS stripe_payment_history (
- id INT(10) UNSIGNED PRIMARY KEY AUTO_INCREMENT,
- fk_payment_intent INT(10) UNSIGNED,
- event_type VARCHAR(50) COMMENT 'created, processing, succeeded, failed, refunded',
- event_data JSON,
- webhook_id VARCHAR(255),
- created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
- FOREIGN KEY (fk_payment_intent) REFERENCES stripe_payment_intents(id) ON DELETE CASCADE,
- INDEX idx_fk_payment_intent (fk_payment_intent),
- INDEX idx_event_type (event_type),
- INDEX idx_created_at (created_at)
-) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-
--- Table pour les remboursements
-CREATE TABLE IF NOT EXISTS stripe_refunds (
- id INT(10) UNSIGNED PRIMARY KEY AUTO_INCREMENT,
- stripe_refund_id VARCHAR(255) UNIQUE,
- fk_payment_intent INT(10) UNSIGNED NOT NULL,
- amount INT NOT NULL COMMENT 'Montant remboursé en centimes',
- reason VARCHAR(100) COMMENT 'duplicate, fraudulent, requested_by_customer',
- status VARCHAR(50),
- metadata JSON,
- created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
- updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
- FOREIGN KEY (fk_payment_intent) REFERENCES stripe_payment_intents(id) ON DELETE CASCADE,
- INDEX idx_fk_payment_intent (fk_payment_intent),
- INDEX idx_status (status)
-) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-
--- Table pour les webhooks reçus (pour éviter les doublons et debug)
-CREATE TABLE IF NOT EXISTS stripe_webhooks (
- id INT(10) UNSIGNED PRIMARY KEY AUTO_INCREMENT,
- stripe_event_id VARCHAR(255) UNIQUE,
- event_type VARCHAR(100),
- livemode BOOLEAN DEFAULT FALSE,
- payload JSON,
- processed BOOLEAN DEFAULT FALSE,
- error_message TEXT NULL,
- created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
- processed_at TIMESTAMP NULL,
- INDEX idx_event_type (event_type),
- INDEX idx_processed (processed),
- INDEX idx_created_at (created_at)
-) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
-
--- Insertion des appareils Android certifiés pour Tap to Pay en France
-INSERT INTO stripe_android_certified_devices (manufacturer, model, model_identifier, tap_to_pay_certified, min_android_version, certification_date) VALUES
--- Samsung
-('Samsung', 'Galaxy S21', 'SM-G991B', TRUE, 11, '2023-01-01'),
-('Samsung', 'Galaxy S21+', 'SM-G996B', TRUE, 11, '2023-01-01'),
-('Samsung', 'Galaxy S21 Ultra', 'SM-G998B', TRUE, 11, '2023-01-01'),
-('Samsung', 'Galaxy S22', 'SM-S901B', TRUE, 12, '2023-01-01'),
-('Samsung', 'Galaxy S22+', 'SM-S906B', TRUE, 12, '2023-01-01'),
-('Samsung', 'Galaxy S22 Ultra', 'SM-S908B', TRUE, 12, '2023-01-01'),
-('Samsung', 'Galaxy S23', 'SM-S911B', TRUE, 13, '2023-06-01'),
-('Samsung', 'Galaxy S23+', 'SM-S916B', TRUE, 13, '2023-06-01'),
-('Samsung', 'Galaxy S23 Ultra', 'SM-S918B', TRUE, 13, '2023-06-01'),
-('Samsung', 'Galaxy S24', 'SM-S921B', TRUE, 14, '2024-01-01'),
-('Samsung', 'Galaxy S24+', 'SM-S926B', TRUE, 14, '2024-01-01'),
-('Samsung', 'Galaxy S24 Ultra', 'SM-S928B', TRUE, 14, '2024-01-01'),
--- Google Pixel
-('Google', 'Pixel 6', 'oriole', TRUE, 12, '2023-01-01'),
-('Google', 'Pixel 6 Pro', 'raven', TRUE, 12, '2023-01-01'),
-('Google', 'Pixel 6a', 'bluejay', TRUE, 12, '2023-03-01'),
-('Google', 'Pixel 7', 'panther', TRUE, 13, '2023-03-01'),
-('Google', 'Pixel 7 Pro', 'cheetah', TRUE, 13, '2023-03-01'),
-('Google', 'Pixel 7a', 'lynx', TRUE, 13, '2023-06-01'),
-('Google', 'Pixel 8', 'shiba', TRUE, 14, '2023-10-01'),
-('Google', 'Pixel 8 Pro', 'husky', TRUE, 14, '2023-10-01'),
-('Google', 'Pixel Fold', 'felix', TRUE, 13, '2023-07-01'),
--- OnePlus
-('OnePlus', '9', 'LE2113', TRUE, 11, '2023-03-01'),
-('OnePlus', '9 Pro', 'LE2123', TRUE, 11, '2023-03-01'),
-('OnePlus', '10 Pro', 'NE2213', TRUE, 12, '2023-06-01'),
-('OnePlus', '11', 'CPH2449', TRUE, 13, '2023-09-01'),
--- Xiaomi
-('Xiaomi', 'Mi 11', 'M2011K2G', TRUE, 11, '2023-06-01'),
-('Xiaomi', '12', '2201123G', TRUE, 12, '2023-09-01'),
-('Xiaomi', '12 Pro', '2201122G', TRUE, 12, '2023-09-01'),
-('Xiaomi', '13', '2211133G', TRUE, 13, '2024-01-01'),
-('Xiaomi', '13 Pro', '2210132G', TRUE, 13, '2024-01-01');
-
--- Vue pour faciliter les requêtes de statistiques
-CREATE OR REPLACE VIEW v_stripe_payment_stats AS
-SELECT
- spi.fk_entite,
- e.encrypted_name AS entite_name,
- spi.fk_user,
- u.encrypted_name AS user_nom,
- u.first_name AS user_prenom,
- COUNT(CASE WHEN spi.status = 'succeeded' THEN 1 END) as total_ventes,
- SUM(CASE WHEN spi.status = 'succeeded' THEN spi.amount ELSE 0 END) as total_montant,
- SUM(CASE WHEN spi.status = 'succeeded' THEN spi.application_fee ELSE 0 END) as total_commissions,
- DATE(spi.created_at) as date_vente
-FROM stripe_payment_intents spi
-LEFT JOIN entites e ON spi.fk_entite = e.id
-LEFT JOIN users u ON spi.fk_user = u.id
-GROUP BY spi.fk_entite, spi.fk_user, DATE(spi.created_at);
-
--- Vue pour le dashboard des amicales
-CREATE OR REPLACE VIEW v_stripe_amicale_dashboard AS
-SELECT
- sa.fk_entite,
- e.encrypted_name AS entite_name,
- sa.stripe_account_id,
- sa.charges_enabled,
- sa.payouts_enabled,
- COUNT(DISTINCT spi.id) as total_transactions,
- SUM(CASE WHEN spi.status = 'succeeded' THEN spi.amount ELSE 0 END) as total_revenus,
- SUM(CASE WHEN spi.status = 'succeeded' THEN spi.application_fee ELSE 0 END) as total_frais_plateforme,
- MAX(spi.created_at) as derniere_transaction
-FROM stripe_accounts sa
-LEFT JOIN entites e ON sa.fk_entite = e.id
-LEFT JOIN stripe_payment_intents spi ON sa.fk_entite = spi.fk_entite
-GROUP BY sa.fk_entite, sa.stripe_account_id;
\ No newline at end of file
diff --git a/api/scripts/migrations_entites.json b/api/scripts/migrations_entites.json
new file mode 100644
index 00000000..f9077dca
--- /dev/null
+++ b/api/scripts/migrations_entites.json
@@ -0,0 +1,4469 @@
+{
+"migrations": [
+ {
+ "entity_id" : 1178,
+ "code_postal" : "01110",
+ "nom" : "2025 TEST AMICALE 01 PLATEAU D'HAUTEVILLE 011",
+ "ville" : "PLATEAU D'HAUTEVILLE",
+ "nb_users" : 5,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-01-13T17:30:23.000Z",
+ "derniere_operation" : "2025-01-13T17:30:23.000Z",
+ "nb_passages" : 1317
+ },
+ {
+ "entity_id" : 1077,
+ "code_postal" : "01200",
+ "nom" : "2024 TEST AMICALE 01 VALSERINE",
+ "ville" : "VALSERINE",
+ "nb_users" : 74,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-09T19:28:46.000Z",
+ "derniere_operation" : "2024-09-09T19:28:46.000Z",
+ "nb_passages" : 6575
+ },
+ {
+ "entity_id" : 948,
+ "code_postal" : "01210",
+ "nom" : "2023 TEST AMICALE 01 VERSONNEX",
+ "ville" : "VERSONNEX",
+ "nb_users" : 17,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-10-10T07:21:13.000Z",
+ "derniere_operation" : "2023-10-10T07:21:13.000Z",
+ "nb_passages" : 87
+ },
+ {
+ "entity_id" : 1113,
+ "code_postal" : "01480",
+ "nom" : "2024 AMICALE 01 JASSANS ",
+ "ville" : "JASSANS",
+ "nb_users" : 40,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-07-03T09:00:32.000Z",
+ "derniere_operation" : "2025-07-03T09:00:32.000Z",
+ "nb_passages" : 6784
+ },
+ {
+ "entity_id" : 812,
+ "code_postal" : "01500",
+ "nom" : "2024 AMICALE 01 AMBRONAY",
+ "ville" : "AMBRONAY",
+ "nb_users" : 23,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-15T17:50:48.000Z",
+ "derniere_operation" : "2024-11-15T17:50:48.000Z",
+ "nb_passages" : 568
+ },
+ {
+ "entity_id" : 1144,
+ "code_postal" : "02310",
+ "nom" : "2024 TEST AMICALE 02 CHARLY SUR MARNE",
+ "ville" : "CHARLY SUR MARNE",
+ "nb_users" : 15,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-22T11:43:32.000Z",
+ "derniere_operation" : "2024-10-22T11:43:32.000Z",
+ "nb_passages" : 5164
+ },
+ {
+ "entity_id" : 1134,
+ "code_postal" : "02850",
+ "nom" : "2024 AMICALE 02 TRELOU-SUR-MARNE",
+ "ville" : "TRELOU-SUR-MARNE",
+ "nb_users" : 18,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-14T18:54:56.000Z",
+ "derniere_operation" : "2024-10-14T18:54:56.000Z",
+ "nb_passages" : 2494
+ },
+ {
+ "entity_id" : 1277,
+ "code_postal" : "03470",
+ "nom" : "2025 AMICALE 71 DIGOIN SECTEUR 03 ",
+ "ville" : "COULANGES",
+ "nb_users" : 18,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-08-28T09:53:02.000Z",
+ "derniere_operation" : "2025-08-28T09:53:02.000Z",
+ "nb_passages" : 1033
+ },
+ {
+ "entity_id" : 1133,
+ "code_postal" : "03700",
+ "nom" : "2024 AMICALE 03 BELLERIVE SUR ALLIER",
+ "ville" : "BELLERIVE SUR ALLIER",
+ "nb_users" : 9,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-14T18:06:09.000Z",
+ "derniere_operation" : "2024-10-14T18:06:09.000Z",
+ "nb_passages" : 4341
+ },
+ {
+ "entity_id" : 1086,
+ "code_postal" : "04000",
+ "nom" : "2025 AMICALE 04 DIGNE LES BAINS",
+ "ville" : "DIGNE LES BAINS",
+ "nb_users" : 78,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-16T14:19:24.000Z",
+ "derniere_operation" : "2024-09-16T14:19:24.000Z",
+ "nb_passages" : 8988
+ },
+ {
+ "entity_id" : 734,
+ "code_postal" : "04200",
+ "nom" : "2024 AMICALE 04 SISTERON",
+ "ville" : "SISTERON",
+ "nb_users" : 33,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-16T12:06:20.000Z",
+ "derniere_operation" : "2024-10-16T12:06:20.000Z",
+ "nb_passages" : 16346
+ },
+ {
+ "entity_id" : 980,
+ "code_postal" : "04230",
+ "nom" : "2023 AMICALE 04 ST ETIENNE LES ORGES",
+ "ville" : "ST ETIENNE LES ORGUES",
+ "nb_users" : 8,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-11-09T14:09:18.000Z",
+ "derniere_operation" : "2023-11-09T14:09:18.000Z",
+ "nb_passages" : 383
+ },
+ {
+ "entity_id" : 562,
+ "code_postal" : "04300",
+ "nom" : "2025 AMICALE 04 FORCALQUIER",
+ "ville" : "FORCALQUIER",
+ "nb_users" : 45,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-08T16:00:54.000Z",
+ "derniere_operation" : "2024-10-08T16:00:54.000Z",
+ "nb_passages" : 21784
+ },
+ {
+ "entity_id" : 1146,
+ "code_postal" : "05560",
+ "nom" : "2024 AMICALE 05 VARS",
+ "ville" : "VARS",
+ "nb_users" : 7,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-24T12:26:37.000Z",
+ "derniere_operation" : "2024-10-24T12:26:37.000Z",
+ "nb_passages" : 833
+ },
+ {
+ "entity_id" : 1129,
+ "code_postal" : "06390",
+ "nom" : "2024 AMICALE 06 CONTES",
+ "ville" : "CONTES",
+ "nb_users" : 68,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-10T14:27:59.000Z",
+ "derniere_operation" : "2024-10-10T14:27:59.000Z",
+ "nb_passages" : 3509
+ },
+ {
+ "entity_id" : 1158,
+ "code_postal" : "07200",
+ "nom" : "2024 AMICALE 07 VIVIERS ",
+ "ville" : "VIVIERS",
+ "nb_users" : 40,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-06T20:25:28.000Z",
+ "derniere_operation" : "2024-11-06T20:25:28.000Z",
+ "nb_passages" : 3805
+ },
+ {
+ "entity_id" : 1140,
+ "code_postal" : "07400",
+ "nom" : "2024 AMICALE 07 LE TEIL",
+ "ville" : "LE TEIL",
+ "nb_users" : 28,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-19T17:24:11.000Z",
+ "derniere_operation" : "2024-10-19T17:24:11.000Z",
+ "nb_passages" : 5576
+ },
+ {
+ "entity_id" : 1175,
+ "code_postal" : "07700",
+ "nom" : "2025 AMICALE 07 SAINT REMEZE",
+ "ville" : "SAINT REMEZE",
+ "nb_users" : 25,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-03-07T16:48:05.000Z",
+ "derniere_operation" : "2025-03-07T16:48:05.000Z",
+ "nb_passages" : 1159
+ },
+ {
+ "entity_id" : 1248,
+ "code_postal" : "07700",
+ "nom" : "2025 AMICALE 07 Sud Vallée du Rhône Ardèche",
+ "ville" : "SAINT MARCEL D'ARDECHE",
+ "nb_users" : 97,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-05-25T20:16:37.000Z",
+ "derniere_operation" : "2025-05-25T20:16:37.000Z",
+ "nb_passages" : 7162
+ },
+ {
+ "entity_id" : 1020,
+ "code_postal" : "08330",
+ "nom" : "2024 AMICALE 08 VRIGNE AUX BOIS",
+ "ville" : "VRIGNE AUX BOIS",
+ "nb_users" : 15,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-18T15:44:56.000Z",
+ "derniere_operation" : "2024-10-18T15:44:56.000Z",
+ "nb_passages" : 3234
+ },
+ {
+ "entity_id" : 167,
+ "code_postal" : "09300",
+ "nom" : "2024 AMICALE 09 LAVELANET",
+ "ville" : "LAVELANET",
+ "nb_users" : 32,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-10-03T13:37:59.000Z",
+ "derniere_operation" : "2025-10-03T13:37:59.000Z",
+ "nb_passages" : 5072
+ },
+ {
+ "entity_id" : 809,
+ "code_postal" : "10130",
+ "nom" : "2025 AMICALE 10 ERVY LE CHATEL ",
+ "ville" : "ERVY LE CHATEL ",
+ "nb_users" : 26,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-27T10:11:38.000Z",
+ "derniere_operation" : "2024-09-27T10:11:38.000Z",
+ "nb_passages" : 2097
+ },
+ {
+ "entity_id" : 614,
+ "code_postal" : "11120",
+ "nom" : "2023 AMICALE 11 BIZE MINERVOIS OK",
+ "ville" : "BIZE MINERVOIS",
+ "nb_users" : 30,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-11-24T15:57:55.000Z",
+ "derniere_operation" : "2023-11-24T15:57:55.000Z",
+ "nb_passages" : 3605
+ },
+ {
+ "entity_id" : 1137,
+ "code_postal" : "11210",
+ "nom" : "2024 AMICALE 11 PORT LA NOUVELLE",
+ "ville" : "PORT LA NOUVELLE",
+ "nb_users" : 50,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-15T09:24:39.000Z",
+ "derniere_operation" : "2024-10-15T09:24:39.000Z",
+ "nb_passages" : 7175
+ },
+ {
+ "entity_id" : 753,
+ "code_postal" : "11400",
+ "nom" : "2024 AMICALE 11 CASTELNAUDARY",
+ "ville" : "CASTELNAUDARY",
+ "nb_users" : 37,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-10-03T09:44:36.000Z",
+ "derniere_operation" : "2025-10-03T09:44:36.000Z",
+ "nb_passages" : 32473
+ },
+ {
+ "entity_id" : 1033,
+ "code_postal" : "12000",
+ "nom" : "2024 AMICALE 12 RODEZ",
+ "ville" : "RODEZ",
+ "nb_users" : 118,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-03-04T15:47:56.000Z",
+ "derniere_operation" : "2025-03-04T15:47:56.000Z",
+ "nb_passages" : 86905
+ },
+ {
+ "entity_id" : 1021,
+ "code_postal" : "12700",
+ "nom" : "2024 AMICALE 12 CAPDENAC-GARE",
+ "ville" : "CAPDENAC-GARE",
+ "nb_users" : 116,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-26T14:05:57.000Z",
+ "derniere_operation" : "2025-09-26T14:05:57.000Z",
+ "nb_passages" : 10709
+ },
+ {
+ "entity_id" : 954,
+ "code_postal" : "13170",
+ "nom" : "2024 AMICALE 13 LES PENNES MIRABEAU",
+ "ville" : "LES PENNES MIRABEAU",
+ "nb_users" : 137,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-04T09:00:31.000Z",
+ "derniere_operation" : "2024-11-04T09:00:31.000Z",
+ "nb_passages" : 18532
+ },
+ {
+ "entity_id" : 326,
+ "code_postal" : "135",
+ "nom" : "2023 AMICALE 01 CULOZ",
+ "ville" : "CULOZ",
+ "nb_users" : 58,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-10-21T11:06:36.000Z",
+ "derniere_operation" : "2023-10-21T11:06:36.000Z",
+ "nb_passages" : 4094
+ },
+ {
+ "entity_id" : 747,
+ "code_postal" : "13830",
+ "nom" : "2023 AMICALE 13 ROQUEFORT LA BEDOULE",
+ "ville" : "ROQUEFORT LA BEDOULE",
+ "nb_users" : 67,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-10-17T13:58:25.000Z",
+ "derniere_operation" : "2023-10-17T13:58:25.000Z",
+ "nb_passages" : 3624
+ },
+ {
+ "entity_id" : 673,
+ "code_postal" : "14100",
+ "nom" : "2024 AMICALE 14 LISIEUX",
+ "ville" : "LISIEUX",
+ "nb_users" : 59,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-18T15:24:58.000Z",
+ "derniere_operation" : "2024-10-18T15:24:58.000Z",
+ "nb_passages" : 30297
+ },
+ {
+ "entity_id" : 1015,
+ "code_postal" : "14150",
+ "nom" : "2024 AMICALE 14 OUISTREHAM",
+ "ville" : "OUISTREHAM",
+ "nb_users" : 72,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-08-11T10:00:25.000Z",
+ "derniere_operation" : "2025-08-11T10:00:25.000Z",
+ "nb_passages" : 194463
+ },
+ {
+ "entity_id" : 627,
+ "code_postal" : "14220",
+ "nom" : "2024 AMICALE 14 THURY-HARCOURT",
+ "ville" : "ESSON",
+ "nb_users" : 16,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-21T17:22:40.000Z",
+ "derniere_operation" : "2025-09-21T17:22:40.000Z",
+ "nb_passages" : 6367
+ },
+ {
+ "entity_id" : 891,
+ "code_postal" : "14240",
+ "nom" : "2024 AMICALE 14 CAUMONT SUR AURE ",
+ "ville" : "CAUMONT SUR AURE",
+ "nb_users" : 13,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-01T19:40:45.000Z",
+ "derniere_operation" : "2024-09-01T19:40:45.000Z",
+ "nb_passages" : 2762
+ },
+ {
+ "entity_id" : 1005,
+ "code_postal" : "14310",
+ "nom" : "2024 AMICALE 14 VILLY BOCAGE",
+ "ville" : "VILLY BOCAGE",
+ "nb_users" : 19,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-08-16T13:05:17.000Z",
+ "derniere_operation" : "2025-08-16T13:05:17.000Z",
+ "nb_passages" : 44355
+ },
+ {
+ "entity_id" : 965,
+ "code_postal" : "14340",
+ "nom" : "2024 AMICALE 14 CAMBREMER",
+ "ville" : "CAMBREMER",
+ "nb_users" : 11,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-10-24T10:37:34.000Z",
+ "derniere_operation" : "2023-10-24T10:37:34.000Z",
+ "nb_passages" : 0
+ },
+ {
+ "entity_id" : 842,
+ "code_postal" : "14370",
+ "nom" : "2023 TEST AMICALE 14 ARGENCES",
+ "ville" : "ARGENCES",
+ "nb_users" : 47,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-08T11:33:16.000Z",
+ "derniere_operation" : "2025-09-08T11:33:16.000Z",
+ "nb_passages" : 6362
+ },
+ {
+ "entity_id" : 1236,
+ "code_postal" : "14370",
+ "nom" : "2025 AMICALE 14 ARGENCES",
+ "ville" : "ARGENCES",
+ "nb_users" : 8,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-04-17T10:55:44.000Z",
+ "derniere_operation" : "2025-04-17T10:55:44.000Z",
+ "nb_passages" : 436
+ },
+ {
+ "entity_id" : 1159,
+ "code_postal" : "14410",
+ "nom" : "2024 AMICALE 14 VASSY-VALDALLIERE",
+ "ville" : "VASSY-VALDALLIERE",
+ "nb_users" : 9,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-10-06T17:02:45.000Z",
+ "derniere_operation" : "2025-10-06T17:02:45.000Z",
+ "nb_passages" : 1736
+ },
+ {
+ "entity_id" : 56,
+ "code_postal" : "14490",
+ "nom" : "2024 AMICALE 14 BALLEROY SUR DROME",
+ "ville" : "BALLEROY SUR DROME",
+ "nb_users" : 34,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-09T17:19:34.000Z",
+ "derniere_operation" : "2024-10-09T17:19:34.000Z",
+ "nb_passages" : 10916
+ },
+ {
+ "entity_id" : 559,
+ "code_postal" : "14500",
+ "nom" : "2024 AMICALE 14 VIRE ",
+ "ville" : "VIRE",
+ "nb_users" : 57,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-08-13T16:12:58.000Z",
+ "derniere_operation" : "2025-08-13T16:12:58.000Z",
+ "nb_passages" : 83313
+ },
+ {
+ "entity_id" : 539,
+ "code_postal" : "14700",
+ "nom" : "2024 AMICALE 14 FALAISE ",
+ "ville" : "SAINT-MARTIN-DE-MIEUX",
+ "nb_users" : 33,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-23T13:08:15.000Z",
+ "derniere_operation" : "2024-09-23T13:08:15.000Z",
+ "nb_passages" : 21622
+ },
+ {
+ "entity_id" : 1162,
+ "code_postal" : "16150",
+ "nom" : "2024 AMICALE 16 CHABANAIS",
+ "ville" : "CHABANAIS",
+ "nb_users" : 7,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-11T21:28:17.000Z",
+ "derniere_operation" : "2024-11-11T21:28:17.000Z",
+ "nb_passages" : 1712
+ },
+ {
+ "entity_id" : 1071,
+ "code_postal" : "16270",
+ "nom" : "2024 AMICALE 16 TERRES DE HAUTE CHARENTE",
+ "ville" : "ROUMAZIERE",
+ "nb_users" : 20,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-08-30T02:02:42.000Z",
+ "derniere_operation" : "2024-08-30T02:02:42.000Z",
+ "nb_passages" : 2719
+ },
+ {
+ "entity_id" : 1161,
+ "code_postal" : "16420",
+ "nom" : "2024 TEST AMICALE 16 BRIGUEIL",
+ "ville" : "BRIGUEIL",
+ "nb_users" : 16,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-08T14:30:48.000Z",
+ "derniere_operation" : "2024-11-08T14:30:48.000Z",
+ "nb_passages" : 2051
+ },
+ {
+ "entity_id" : 1076,
+ "code_postal" : "16700",
+ "nom" : "2024 AMICALE 16 RUFFEC",
+ "ville" : "RUFFEC",
+ "nb_users" : 39,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-10-05T12:12:42.000Z",
+ "derniere_operation" : "2025-10-05T12:12:42.000Z",
+ "nb_passages" : 53009
+ },
+ {
+ "entity_id" : 330,
+ "code_postal" : "17100",
+ "nom" : "2024 AMICALE 17 SAINTES",
+ "ville" : "SAINTES",
+ "nb_users" : 93,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-26T16:44:07.000Z",
+ "derniere_operation" : "2024-10-26T16:44:07.000Z",
+ "nb_passages" : 40531
+ },
+ {
+ "entity_id" : 743,
+ "code_postal" : "17132",
+ "nom" : "2024 AMICALE 17 MESCHERS SUR GIRONDE",
+ "ville" : "MESCHERS SUR GIRONDE",
+ "nb_users" : 27,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-07T14:02:55.000Z",
+ "derniere_operation" : "2024-10-07T14:02:55.000Z",
+ "nb_passages" : 1991
+ },
+ {
+ "entity_id" : 990,
+ "code_postal" : "17770",
+ "nom" : "2025 TEST AMICALE 17 MIGRON",
+ "ville" : "MIGRON",
+ "nb_users" : 27,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-11-21T21:15:03.000Z",
+ "derniere_operation" : "2023-11-21T21:15:03.000Z",
+ "nb_passages" : 4058
+ },
+ {
+ "entity_id" : 543,
+ "code_postal" : "18110",
+ "nom" : "2024 AMICALE 18 SAINT-MARTIN D'AUXIGNY",
+ "ville" : "SAINT-MARTIN D'AUXIGNY",
+ "nb_users" : 38,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-08-07T18:37:43.000Z",
+ "derniere_operation" : "2024-08-07T18:37:43.000Z",
+ "nb_passages" : 26517
+ },
+ {
+ "entity_id" : 926,
+ "code_postal" : "18190",
+ "nom" : "2023 AMICALE 18 CHATEAUNEUF SUR CHER ",
+ "ville" : " CHATEAUNEUF SUR CHER ",
+ "nb_users" : 28,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-10-02T15:42:09.000Z",
+ "derniere_operation" : "2023-10-02T15:42:09.000Z",
+ "nb_passages" : 1316
+ },
+ {
+ "entity_id" : 691,
+ "code_postal" : "18270",
+ "nom" : "2024 AMICALE 18 CULAN",
+ "ville" : "CULAN",
+ "nb_users" : 20,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-07T14:04:14.000Z",
+ "derniere_operation" : "2024-10-07T14:04:14.000Z",
+ "nb_passages" : 6047
+ },
+ {
+ "entity_id" : 273,
+ "code_postal" : "18510",
+ "nom" : "2024 AMICALE 18 MENETOU-SALON",
+ "ville" : "MENETOU-SALON",
+ "nb_users" : 16,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-01-05T12:45:59.000Z",
+ "derniere_operation" : "2024-01-05T12:45:59.000Z",
+ "nb_passages" : 1953
+ },
+ {
+ "entity_id" : 1054,
+ "code_postal" : "19260",
+ "nom" : "2024 AMICALE 19 TREIGNAC 19260",
+ "ville" : "TREIGNAC",
+ "nb_users" : 21,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-06-24T08:51:25.000Z",
+ "derniere_operation" : "2024-06-24T08:51:25.000Z",
+ "nb_passages" : 1777
+ },
+ {
+ "entity_id" : 1150,
+ "code_postal" : "19410",
+ "nom" : "2024 AMICALE 19 VIGEOIS",
+ "ville" : "VIGEOIS",
+ "nb_users" : 13,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-28T13:39:23.000Z",
+ "derniere_operation" : "2024-10-28T13:39:23.000Z",
+ "nb_passages" : 2214
+ },
+ {
+ "entity_id" : 1286,
+ "code_postal" : "19450",
+ "nom" : "2025 TEST AMICALE 19 CHAMBOULIVE ",
+ "ville" : "CHAMBOULIVE ",
+ "nb_users" : 7,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-17T08:20:00.000Z",
+ "derniere_operation" : "2025-09-17T08:20:00.000Z",
+ "nb_passages" : 803
+ },
+ {
+ "entity_id" : 595,
+ "code_postal" : "21170",
+ "nom" : "2024 AMICALE 21 SAINT JEAN DE LOSNE",
+ "ville" : "SAINT JEAN DE LOSNE",
+ "nb_users" : 30,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-15T16:45:51.000Z",
+ "derniere_operation" : "2025-09-15T16:45:51.000Z",
+ "nb_passages" : 7466
+ },
+ {
+ "entity_id" : 1127,
+ "code_postal" : "21190",
+ "nom" : "CHAGNY TEST 71150 + département 21",
+ "ville" : "CORPEAU",
+ "nb_users" : 11,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-08T14:42:13.000Z",
+ "derniere_operation" : "2024-10-08T14:42:13.000Z",
+ "nb_passages" : 885
+ },
+ {
+ "entity_id" : 1138,
+ "code_postal" : "21250",
+ "nom" : "2024 AMICALE 21 SEURRE",
+ "ville" : "SEURRE",
+ "nb_users" : 66,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-18T11:02:26.000Z",
+ "derniere_operation" : "2024-10-18T11:02:26.000Z",
+ "nb_passages" : 8278
+ },
+ {
+ "entity_id" : 973,
+ "code_postal" : "21260",
+ "nom" : "2024 AMICALE 21 SELONGEY",
+ "ville" : "SELONGEY",
+ "nb_users" : 20,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-03-26T08:59:13.000Z",
+ "derniere_operation" : "2025-03-26T08:59:13.000Z",
+ "nb_passages" : 2735
+ },
+ {
+ "entity_id" : 55,
+ "code_postal" : "22130",
+ "nom" : "2024 AMICALE 22 PLANCOET",
+ "ville" : "PLANCOET",
+ "nb_users" : 55,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-17T12:25:55.000Z",
+ "derniere_operation" : "2024-10-17T12:25:55.000Z",
+ "nb_passages" : 24471
+ },
+ {
+ "entity_id" : 1188,
+ "code_postal" : "22160",
+ "nom" : "2025 TEST AMICALE 22 CALLAC",
+ "ville" : "CALLAC",
+ "nb_users" : 6,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-01-28T16:17:19.000Z",
+ "derniere_operation" : "2025-01-28T16:17:19.000Z",
+ "nb_passages" : 0
+ },
+ {
+ "entity_id" : 89,
+ "code_postal" : "22200",
+ "nom" : "2024 AMICALE 22 GUINGAMP",
+ "ville" : "GUINGAMP",
+ "nb_users" : 66,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-27T17:21:17.000Z",
+ "derniere_operation" : "2024-10-27T17:21:17.000Z",
+ "nb_passages" : 16813
+ },
+ {
+ "entity_id" : 937,
+ "code_postal" : "22290",
+ "nom" : "2024 AMICALE 22 LANVOLLON",
+ "ville" : "LANVOLLON",
+ "nb_users" : 33,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-14T08:35:25.000Z",
+ "derniere_operation" : "2024-11-14T08:35:25.000Z",
+ "nb_passages" : 4941
+ },
+ {
+ "entity_id" : 707,
+ "code_postal" : "22310",
+ "nom" : "2024 AMICALE 22 PLESTIN LES GREVES",
+ "ville" : "PLESTIN LES GREVES",
+ "nb_users" : 22,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-10T15:56:07.000Z",
+ "derniere_operation" : "2024-09-10T15:56:07.000Z",
+ "nb_passages" : 7773
+ },
+ {
+ "entity_id" : 911,
+ "code_postal" : "22330",
+ "nom" : "2024 AMICALE 22 PLESSALA",
+ "ville" : "PLESSALA",
+ "nb_users" : 12,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-15T17:35:20.000Z",
+ "derniere_operation" : "2024-10-15T17:35:20.000Z",
+ "nb_passages" : 1899
+ },
+ {
+ "entity_id" : 704,
+ "code_postal" : "22350",
+ "nom" : "2024 AMICALE 22 YVIGNAC LA TOUR",
+ "ville" : "YVIGNAC LA TOUR",
+ "nb_users" : 27,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-04T17:59:46.000Z",
+ "derniere_operation" : "2024-11-04T17:59:46.000Z",
+ "nb_passages" : 10753
+ },
+ {
+ "entity_id" : 130,
+ "code_postal" : "22350",
+ "nom" : "2025 AMICALE 22 CAULNES",
+ "ville" : "CAULNES",
+ "nb_users" : 34,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-26T18:10:21.000Z",
+ "derniere_operation" : "2025-09-26T18:10:21.000Z",
+ "nb_passages" : 4726
+ },
+ {
+ "entity_id" : 51,
+ "code_postal" : "22400",
+ "nom" : "2024 AMICALE 22 LAMBALLE",
+ "ville" : "LAMBALLE",
+ "nb_users" : 81,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-15T20:43:41.000Z",
+ "derniere_operation" : "2024-10-15T20:43:41.000Z",
+ "nb_passages" : 14218
+ },
+ {
+ "entity_id" : 102,
+ "code_postal" : "22510",
+ "nom" : "2024 AMICALE 22 MONCONTOUR",
+ "ville" : "MONCONTOUR",
+ "nb_users" : 37,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-01T18:13:40.000Z",
+ "derniere_operation" : "2025-09-01T18:13:40.000Z",
+ "nb_passages" : 29184
+ },
+ {
+ "entity_id" : 454,
+ "code_postal" : "22550",
+ "nom" : "2024 AMICALE 22 HENANBIHEN",
+ "ville" : "HENANBIHEN",
+ "nb_users" : 30,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-02T16:39:41.000Z",
+ "derniere_operation" : "2025-09-02T16:39:41.000Z",
+ "nb_passages" : 15504
+ },
+ {
+ "entity_id" : 1298,
+ "code_postal" : "22590",
+ "nom" : "2025 AMICALE PORDIC 22590",
+ "ville" : "PORDIC",
+ "nb_users" : 43,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-10-01T15:04:30.000Z",
+ "derniere_operation" : "2025-10-01T15:04:30.000Z",
+ "nb_passages" : 4332
+ },
+ {
+ "entity_id" : 651,
+ "code_postal" : "22610",
+ "nom" : "2024 AMICALE 22 PLEUBIAN",
+ "ville" : "PLEUBIAN",
+ "nb_users" : 27,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-13T09:11:55.000Z",
+ "derniere_operation" : "2024-10-13T09:11:55.000Z",
+ "nb_passages" : 4526
+ },
+ {
+ "entity_id" : 893,
+ "code_postal" : "22640",
+ "nom" : "2024 AMICALE 22 PLÉNÉE JUGON",
+ "ville" : "PLÉNÉE JUGON",
+ "nb_users" : 30,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-03-18T20:45:21.000Z",
+ "derniere_operation" : "2024-03-18T20:45:21.000Z",
+ "nb_passages" : 11526
+ },
+ {
+ "entity_id" : 935,
+ "code_postal" : "23000",
+ "nom" : "2024 AMICALE 23 GUERET",
+ "ville" : "GUERET",
+ "nb_users" : 82,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-09T17:15:44.000Z",
+ "derniere_operation" : "2024-11-09T17:15:44.000Z",
+ "nb_passages" : 684
+ },
+ {
+ "entity_id" : 977,
+ "code_postal" : "23200",
+ "nom" : "2024 AMICALE 23 BONNAT",
+ "ville" : "BONNAT",
+ "nb_users" : 28,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-21T15:43:03.000Z",
+ "derniere_operation" : "2024-11-21T15:43:03.000Z",
+ "nb_passages" : 3522
+ },
+ {
+ "entity_id" : 757,
+ "code_postal" : "24100",
+ "nom" : "2024 AMICALE 24 BERGERAC ",
+ "ville" : "BERGERAC",
+ "nb_users" : 79,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-30T11:58:33.000Z",
+ "derniere_operation" : "2024-10-30T11:58:33.000Z",
+ "nb_passages" : 56229
+ },
+ {
+ "entity_id" : 867,
+ "code_postal" : "24220",
+ "nom" : "2023 AMICALE 24 SAINT CYPRIEN ",
+ "ville" : "SAINT CYPRIEN ",
+ "nb_users" : 40,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-13T14:36:11.000Z",
+ "derniere_operation" : "2025-09-13T14:36:11.000Z",
+ "nb_passages" : 23706
+ },
+ {
+ "entity_id" : 1183,
+ "code_postal" : "24360",
+ "nom" : "2025 TEST AMICALE 24 PIEGUT-PLUVIERS",
+ "ville" : "PIEGUT-PLUVIERS",
+ "nb_users" : 21,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-04-16T07:37:17.000Z",
+ "derniere_operation" : "2025-04-16T07:37:17.000Z",
+ "nb_passages" : 0
+ },
+ {
+ "entity_id" : 1202,
+ "code_postal" : "24410",
+ "nom" : "2025 TEST AMICALE 24 ST AULAYE PUYMANGOU",
+ "ville" : "ST AULAYE PUYMANGOU",
+ "nb_users" : 11,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-02-17T18:08:58.000Z",
+ "derniere_operation" : "2025-02-17T18:08:58.000Z",
+ "nb_passages" : 0
+ },
+ {
+ "entity_id" : 1030,
+ "code_postal" : "24800",
+ "nom" : "2025 AMICALE 24 THIVIERS",
+ "ville" : "THIVIERS",
+ "nb_users" : 54,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-08-06T18:44:32.000Z",
+ "derniere_operation" : "2025-08-06T18:44:32.000Z",
+ "nb_passages" : 54033
+ },
+ {
+ "entity_id" : 264,
+ "code_postal" : "25250",
+ "nom" : "2024 AMICALE 25 L'ISLE SUR LE DOUBS ",
+ "ville" : "L'ISLE SUR LE DOUBS",
+ "nb_users" : 27,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-09T12:08:22.000Z",
+ "derniere_operation" : "2024-09-09T12:08:22.000Z",
+ "nb_passages" : 4154
+ },
+ {
+ "entity_id" : 693,
+ "code_postal" : "25440",
+ "nom" : "2024 AMICALE 25 SAPEURS POMPIERS QUINGEY",
+ "ville" : "QUINGEY",
+ "nb_users" : 16,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-10T19:41:19.000Z",
+ "derniere_operation" : "2024-10-10T19:41:19.000Z",
+ "nb_passages" : 8210
+ },
+ {
+ "entity_id" : 182,
+ "code_postal" : "26130",
+ "nom" : "2024 AMICALE 26 SAINT PAUL TROIS CHATEAU ",
+ "ville" : "ST PAUL TROIS CHATEAUX",
+ "nb_users" : 84,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-24T10:55:21.000Z",
+ "derniere_operation" : "2024-10-24T10:55:21.000Z",
+ "nb_passages" : 23650
+ },
+ {
+ "entity_id" : 1267,
+ "code_postal" : "26160",
+ "nom" : "2025 TEST AMICALE 26 SAINT GERVAIS SUR ROUBIO",
+ "ville" : "SAINT GERVAIS SUR ROUBION",
+ "nb_users" : 40,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-21T15:45:32.000Z",
+ "derniere_operation" : "2025-09-21T15:45:32.000Z",
+ "nb_passages" : 1537
+ },
+ {
+ "entity_id" : 952,
+ "code_postal" : "26700",
+ "nom" : "PAS 2023 AMICALE 26 PIERRELATTE",
+ "ville" : "PIERRELATTE",
+ "nb_users" : 21,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-10-14T06:49:48.000Z",
+ "derniere_operation" : "2023-10-14T06:49:48.000Z",
+ "nb_passages" : 3
+ },
+ {
+ "entity_id" : 392,
+ "code_postal" : "26740",
+ "nom" : "2025 AMICALE 26 MARSANNE",
+ "ville" : "MARSANNE",
+ "nb_users" : 17,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-12T12:02:48.000Z",
+ "derniere_operation" : "2024-10-12T12:02:48.000Z",
+ "nb_passages" : 5890
+ },
+ {
+ "entity_id" : 310,
+ "code_postal" : "27140",
+ "nom" : "2024 AMICALE 27 GISORS",
+ "ville" : "GISORS",
+ "nb_users" : 40,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-10T19:49:20.000Z",
+ "derniere_operation" : "2024-10-10T19:49:20.000Z",
+ "nb_passages" : 12801
+ },
+ {
+ "entity_id" : 1081,
+ "code_postal" : "27150",
+ "nom" : "2024 AMICALE 27 ETREPAGNY",
+ "ville" : "ETREPAGNY",
+ "nb_users" : 36,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-12T15:58:01.000Z",
+ "derniere_operation" : "2024-09-12T15:58:01.000Z",
+ "nb_passages" : 7833
+ },
+ {
+ "entity_id" : 674,
+ "code_postal" : "27170",
+ "nom" : "2024 AMICALE 27 BEAUMONT LE ROGER",
+ "ville" : "BEAUMONT LE ROGER",
+ "nb_users" : 27,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-20T10:59:33.000Z",
+ "derniere_operation" : "2024-09-20T10:59:33.000Z",
+ "nb_passages" : 1946
+ },
+ {
+ "entity_id" : 1112,
+ "code_postal" : "27300",
+ "nom" : "2024 AMICALE 27 BERNAY",
+ "ville" : "BERNAY",
+ "nb_users" : 73,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-07-29T12:36:27.000Z",
+ "derniere_operation" : "2025-07-29T12:36:27.000Z",
+ "nb_passages" : 24587
+ },
+ {
+ "entity_id" : 897,
+ "code_postal" : "27310",
+ "nom" : "2024 AMICALE 27 BOURG-ACHARD",
+ "ville" : "BOURG-ACHARD",
+ "nb_users" : 37,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-08-27T15:32:20.000Z",
+ "derniere_operation" : "2025-08-27T15:32:20.000Z",
+ "nb_passages" : 16617
+ },
+ {
+ "entity_id" : 536,
+ "code_postal" : "27380",
+ "nom" : "2024 AMICALE 27 FLEURY-CHARLEVAL",
+ "ville" : "FLEURY-CHARLEVAL",
+ "nb_users" : 25,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-10-03T10:41:41.000Z",
+ "derniere_operation" : "2025-10-03T10:41:41.000Z",
+ "nb_passages" : 10695
+ },
+ {
+ "entity_id" : 924,
+ "code_postal" : "27500",
+ "nom" : "2024 AMICALE 27 PONT AUDEMER",
+ "ville" : "PONT AUDEMER",
+ "nb_users" : 50,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-12T09:59:12.000Z",
+ "derniere_operation" : "2025-09-12T09:59:12.000Z",
+ "nb_passages" : 36674
+ },
+ {
+ "entity_id" : 1203,
+ "code_postal" : "27800",
+ "nom" : "2025 AMICALE 27 BRIONNE",
+ "ville" : "BRIONNE",
+ "nb_users" : 28,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-02-19T13:50:18.000Z",
+ "derniere_operation" : "2025-02-19T13:50:18.000Z",
+ "nb_passages" : 5860
+ },
+ {
+ "entity_id" : 1111,
+ "code_postal" : "28140",
+ "nom" : "2024 AMICALE 28 ORGERES EN BEAUCE ",
+ "ville" : "ORGERES EN BEAUCE",
+ "nb_users" : 8,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-10-07T10:15:47.000Z",
+ "derniere_operation" : "2025-10-07T10:15:47.000Z",
+ "nb_passages" : 2503
+ },
+ {
+ "entity_id" : 964,
+ "code_postal" : "28310",
+ "nom" : "2023 AMICALE 28 BAUDREVILLE ",
+ "ville" : "BAUDREVILLE ",
+ "nb_users" : 11,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-11T21:00:30.000Z",
+ "derniere_operation" : "2024-11-11T21:00:30.000Z",
+ "nb_passages" : 2882
+ },
+ {
+ "entity_id" : 229,
+ "code_postal" : "29120",
+ "nom" : "2024 AMICALE 29 PONT L'ABBE",
+ "ville" : "PONT L'ABBE",
+ "nb_users" : 54,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-30T20:22:43.000Z",
+ "derniere_operation" : "2025-09-30T20:22:43.000Z",
+ "nb_passages" : 2223
+ },
+ {
+ "entity_id" : 711,
+ "code_postal" : "29140",
+ "nom" : "2024 AMICALE 29 ROSPORDEN",
+ "ville" : "ROSPORDEN",
+ "nb_users" : 35,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-08-05T10:49:40.000Z",
+ "derniere_operation" : "2025-08-05T10:49:40.000Z",
+ "nb_passages" : 19124
+ },
+ {
+ "entity_id" : 678,
+ "code_postal" : "29300",
+ "nom" : "2024 AMICALE 29 QUIMPERLE",
+ "ville" : "QUIMPERLE",
+ "nb_users" : 54,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-19T07:20:38.000Z",
+ "derniere_operation" : "2024-10-19T07:20:38.000Z",
+ "nb_passages" : 25542
+ },
+ {
+ "entity_id" : 894,
+ "code_postal" : "29390",
+ "nom" : "2024 AMICALE 29 SCAER",
+ "ville" : "SCAER",
+ "nb_users" : 27,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-08-25T13:24:40.000Z",
+ "derniere_operation" : "2025-08-25T13:24:40.000Z",
+ "nb_passages" : 27015
+ },
+ {
+ "entity_id" : 698,
+ "code_postal" : "29520",
+ "nom" : "2023 AMICALE 29 CHATEAUNEUF DU FAOU",
+ "ville" : "CHATEAUNEUF DU FAOU",
+ "nb_users" : 34,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-10-23T16:26:03.000Z",
+ "derniere_operation" : "2023-10-23T16:26:03.000Z",
+ "nb_passages" : 4742
+ },
+ {
+ "entity_id" : 203,
+ "code_postal" : "29600",
+ "nom" : "2024 AMICALE 29 MORLAIX",
+ "ville" : "MORLAIX",
+ "nb_users" : 58,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-19T21:50:03.000Z",
+ "derniere_operation" : "2025-09-19T21:50:03.000Z",
+ "nb_passages" : 37047
+ },
+ {
+ "entity_id" : 98,
+ "code_postal" : "29720",
+ "nom" : "2024 AMICALE 29 PLONEOUR-LANVERN",
+ "ville" : "PLONEOUR-LANVERN",
+ "nb_users" : 25,
+ "nb_operations" : 4,
+ "premiere_operation" : "2023-10-24T14:55:33.000Z",
+ "derniere_operation" : "2024-10-12T12:06:55.000Z",
+ "nb_passages" : 10304
+ },
+ {
+ "entity_id" : 1147,
+ "code_postal" : "29720",
+ "nom" : "2024 AMICALE 29 POULDREUZIC",
+ "ville" : "POULDREUZIC",
+ "nb_users" : 15,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-24T13:11:23.000Z",
+ "derniere_operation" : "2024-10-24T13:11:23.000Z",
+ "nb_passages" : 2962
+ },
+ {
+ "entity_id" : 1006,
+ "code_postal" : "30320",
+ "nom" : "2024 AMICALE 30 MARGUERITTES ",
+ "ville" : "MARGUERITTES",
+ "nb_users" : 1116,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-21T07:00:07.000Z",
+ "derniere_operation" : "2025-09-21T07:00:07.000Z",
+ "nb_passages" : 6737
+ },
+ {
+ "entity_id" : 953,
+ "code_postal" : "31390",
+ "nom" : "2024 AMICALE 31 CARBONNE",
+ "ville" : "CARBONNE",
+ "nb_users" : 38,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-20T07:59:33.000Z",
+ "derniere_operation" : "2025-09-20T07:59:33.000Z",
+ "nb_passages" : 10516
+ },
+ {
+ "entity_id" : 307,
+ "code_postal" : "31470",
+ "nom" : "2023 TEST AMICALE 31 SAINT LYS",
+ "ville" : "SAINT LYS",
+ "nb_users" : 11,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-09-25T15:38:03.000Z",
+ "derniere_operation" : "2023-09-25T15:38:03.000Z",
+ "nb_passages" : 0
+ },
+ {
+ "entity_id" : 268,
+ "code_postal" : "31790",
+ "nom" : "2024 AMICALE 31 SAINT JORY",
+ "ville" : "SAINT JORY",
+ "nb_users" : 70,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-07T16:14:42.000Z",
+ "derniere_operation" : "2024-11-07T16:14:42.000Z",
+ "nb_passages" : 22220
+ },
+ {
+ "entity_id" : 640,
+ "code_postal" : "32130",
+ "nom" : "2024 AMICALE 32 SAMATAN",
+ "ville" : "SAMATAN",
+ "nb_users" : 25,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-19T09:38:38.000Z",
+ "derniere_operation" : "2024-11-19T09:38:38.000Z",
+ "nb_passages" : 9470
+ },
+ {
+ "entity_id" : 1231,
+ "code_postal" : "32190",
+ "nom" : "2025 AMICALE 32 VIC-FEZENSAC",
+ "ville" : "VIC-FEZENSAC",
+ "nb_users" : 34,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-08T10:57:59.000Z",
+ "derniere_operation" : "2025-09-08T10:57:59.000Z",
+ "nb_passages" : 4641
+ },
+ {
+ "entity_id" : 1164,
+ "code_postal" : "32450",
+ "nom" : "2024 AMICALE 32 SARAMON",
+ "ville" : "SARAMON",
+ "nb_users" : 18,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-16T14:19:31.000Z",
+ "derniere_operation" : "2024-11-16T14:19:31.000Z",
+ "nb_passages" : 2100
+ },
+ {
+ "entity_id" : 817,
+ "code_postal" : "32800",
+ "nom" : "2024 AMICALE 32 EAUZE",
+ "ville" : "EAUZE",
+ "nb_users" : 22,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-25T15:21:11.000Z",
+ "derniere_operation" : "2025-09-25T15:21:11.000Z",
+ "nb_passages" : 13922
+ },
+ {
+ "entity_id" : 797,
+ "code_postal" : "33240",
+ "nom" : "2024 AMICALE 33 SAINT ANDRé DE CUBZA",
+ "ville" : "SAINT ANDRÉ DE CUBZAC",
+ "nb_users" : 67,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-14T08:09:21.000Z",
+ "derniere_operation" : "2024-09-14T08:09:21.000Z",
+ "nb_passages" : 23689
+ },
+ {
+ "entity_id" : 996,
+ "code_postal" : "33650",
+ "nom" : "2024 AMICALE 33 CABANAC",
+ "ville" : "CABANAC",
+ "nb_users" : 38,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-12-23T13:59:52.000Z",
+ "derniere_operation" : "2023-12-23T13:59:52.000Z",
+ "nb_passages" : 6194
+ },
+ {
+ "entity_id" : 5,
+ "code_postal" : "35000",
+ "nom" : "AMICALE TEST DEV PIERRE",
+ "ville" : "RENNES",
+ "nb_users" : 14,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-24T09:25:59.000Z",
+ "derniere_operation" : "2024-09-24T09:25:59.000Z",
+ "nb_passages" : 2126
+ },
+ {
+ "entity_id" : 172,
+ "code_postal" : "35120",
+ "nom" : "2024 AMICALE 35 DOL DE BRETAGNE",
+ "ville" : "DOL DE BRETAGNE",
+ "nb_users" : 62,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-04T20:38:15.000Z",
+ "derniere_operation" : "2024-10-04T20:38:15.000Z",
+ "nb_passages" : 35217
+ },
+ {
+ "entity_id" : 1051,
+ "code_postal" : "35133",
+ "nom" : "2024 AMICALE 35 SAINT GERMAIN EN COGLES",
+ "ville" : "SAINT GERMAIN EN COGLES",
+ "nb_users" : 12,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-07-15T19:47:40.000Z",
+ "derniere_operation" : "2025-07-15T19:47:40.000Z",
+ "nb_passages" : 3750
+ },
+ {
+ "entity_id" : 1069,
+ "code_postal" : "35150",
+ "nom" : "2024 AMICALE 35 JANZÉ 35150",
+ "ville" : "JANZÉ",
+ "nb_users" : 47,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-08-22T15:09:43.000Z",
+ "derniere_operation" : "2024-08-22T15:09:43.000Z",
+ "nb_passages" : 5830
+ },
+ {
+ "entity_id" : 716,
+ "code_postal" : "35220",
+ "nom" : "2023 AMICALE 35 CHATEAUBOURG",
+ "ville" : "CHATEAUBOURG",
+ "nb_users" : 49,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-09-29T13:23:52.000Z",
+ "derniere_operation" : "2023-09-29T13:23:52.000Z",
+ "nb_passages" : 11766
+ },
+ {
+ "entity_id" : 545,
+ "code_postal" : "35250",
+ "nom" : "2024 AMICALE 35 SAINT AUBIN D'AUBIGNÉ",
+ "ville" : "SAINT AUBIN D'AUBIGNÉ",
+ "nb_users" : 26,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-05-13T13:19:01.000Z",
+ "derniere_operation" : "2025-05-13T13:19:01.000Z",
+ "nb_passages" : 21923
+ },
+ {
+ "entity_id" : 969,
+ "code_postal" : "35290",
+ "nom" : "2024 AMICALE 35 GAEL \/ MUEL",
+ "ville" : "GAEL \/ MUEL",
+ "nb_users" : 15,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-06T17:39:39.000Z",
+ "derniere_operation" : "2024-11-06T17:39:39.000Z",
+ "nb_passages" : 2468
+ },
+ {
+ "entity_id" : 722,
+ "code_postal" : "35290",
+ "nom" : "2024 AMICALE 35 SAINT MEEN LE GRAND",
+ "ville" : "SAINT MEEN LE GRAND",
+ "nb_users" : 45,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-21T17:21:57.000Z",
+ "derniere_operation" : "2024-09-21T17:21:57.000Z",
+ "nb_passages" : 7142
+ },
+ {
+ "entity_id" : 511,
+ "code_postal" : "35310",
+ "nom" : "2023 AMICALE 35 MORDELLES",
+ "ville" : "MORDELLES",
+ "nb_users" : 38,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-10-12T17:45:20.000Z",
+ "derniere_operation" : "2023-10-12T17:45:20.000Z",
+ "nb_passages" : 11529
+ },
+ {
+ "entity_id" : 689,
+ "code_postal" : "35370",
+ "nom" : "2024 AMICALE 35 ARGENTRE ETRELLES",
+ "ville" : "ARGENTRE DU PLESSIS",
+ "nb_users" : 39,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-24T14:43:12.000Z",
+ "derniere_operation" : "2024-09-24T14:43:12.000Z",
+ "nb_passages" : 26655
+ },
+ {
+ "entity_id" : 1220,
+ "code_postal" : "35440",
+ "nom" : "2025 TEST AMICALE 35 ILLE ET RANCE",
+ "ville" : "MONTREUIL SUR ILLE",
+ "nb_users" : 29,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-03-19T18:46:49.000Z",
+ "derniere_operation" : "2025-03-19T18:46:49.000Z",
+ "nb_passages" : 2689
+ },
+ {
+ "entity_id" : 1089,
+ "code_postal" : "35470",
+ "nom" : "2024 AMICALE 35 BAIN DE BRETAGNE ",
+ "ville" : "BAIN DE BRETAGNE",
+ "nb_users" : 58,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-13T12:27:52.000Z",
+ "derniere_operation" : "2025-09-13T12:27:52.000Z",
+ "nb_passages" : 96220
+ },
+ {
+ "entity_id" : 1060,
+ "code_postal" : "35500",
+ "nom" : "2024 AMICALE 35 SAINT M'HERVE",
+ "ville" : "SAINT M'HERVE",
+ "nb_users" : 13,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-02-11T18:12:39.000Z",
+ "derniere_operation" : "2025-02-11T18:12:39.000Z",
+ "nb_passages" : 3989
+ },
+ {
+ "entity_id" : 1295,
+ "code_postal" : "35500",
+ "nom" : "2025 AMICALE 35 ERBREE",
+ "ville" : "HERBREE",
+ "nb_users" : 22,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-24T13:30:58.000Z",
+ "derniere_operation" : "2025-09-24T13:30:58.000Z",
+ "nb_passages" : 3
+ },
+ {
+ "entity_id" : 232,
+ "code_postal" : "35520",
+ "nom" : "2024 AMICALE 35 MELESSE",
+ "ville" : "MELESSE",
+ "nb_users" : 23,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-10-06T16:35:45.000Z",
+ "derniere_operation" : "2025-10-06T16:35:45.000Z",
+ "nb_passages" : 16596
+ },
+ {
+ "entity_id" : 1259,
+ "code_postal" : "35580",
+ "nom" : "2025 AMICALE 35 BAULON ",
+ "ville" : "BAULON",
+ "nb_users" : 22,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-06-13T11:17:06.000Z",
+ "derniere_operation" : "2025-06-13T11:17:06.000Z",
+ "nb_passages" : 2584
+ },
+ {
+ "entity_id" : 1036,
+ "code_postal" : "35690",
+ "nom" : "2024 AMICALE ACIGNé 35",
+ "ville" : "ACIGNé",
+ "nb_users" : 28,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-08-25T08:25:59.000Z",
+ "derniere_operation" : "2025-08-25T08:25:59.000Z",
+ "nb_passages" : 10926
+ },
+ {
+ "entity_id" : 1017,
+ "code_postal" : "35800",
+ "nom" : "2024 AMICALE 35 DINARD ",
+ "ville" : "DINARD",
+ "nb_users" : 6,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-02-29T08:27:06.000Z",
+ "derniere_operation" : "2024-02-29T08:27:06.000Z",
+ "nb_passages" : 16
+ },
+ {
+ "entity_id" : 557,
+ "code_postal" : "35850",
+ "nom" : "2024 AMICALE 35 IRODOUER",
+ "ville" : "IRODOUER",
+ "nb_users" : 17,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-15T17:09:25.000Z",
+ "derniere_operation" : "2024-10-15T17:09:25.000Z",
+ "nb_passages" : 4205
+ },
+ {
+ "entity_id" : 537,
+ "code_postal" : "35850",
+ "nom" : "2024 AMICALE 35 ROMILLE",
+ "ville" : "ROMILLE",
+ "nb_users" : 30,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-18T16:43:03.000Z",
+ "derniere_operation" : "2025-09-18T16:43:03.000Z",
+ "nb_passages" : 12680
+ },
+ {
+ "entity_id" : 1139,
+ "code_postal" : "36000",
+ "nom" : "2024 AMICALE 36 CHATEAUROUX",
+ "ville" : "CHATEAUROUX",
+ "nb_users" : 76,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-19T16:51:40.000Z",
+ "derniere_operation" : "2024-10-19T16:51:40.000Z",
+ "nb_passages" : 16673
+ },
+ {
+ "entity_id" : 920,
+ "code_postal" : "37120",
+ "nom" : "2024 AMICALE 37 CHAMPIGNY SUR VEUDE",
+ "ville" : " CHAMPIGNY SUR VEUDE",
+ "nb_users" : 22,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-06T20:23:15.000Z",
+ "derniere_operation" : "2024-10-06T20:23:15.000Z",
+ "nb_passages" : 7781
+ },
+ {
+ "entity_id" : 157,
+ "code_postal" : "38190",
+ "nom" : "2024 AMICALE 38 VILLARD-BONNOT",
+ "ville" : "VILLARD-BONNOT",
+ "nb_users" : 20,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-05-07T17:41:13.000Z",
+ "derniere_operation" : "2025-05-07T17:41:13.000Z",
+ "nb_passages" : 28736
+ },
+ {
+ "entity_id" : 1225,
+ "code_postal" : "38210",
+ "nom" : "2025 AMICALE 38 SAINT QUENTINSUR ISÈRE",
+ "ville" : "ST QUENTIN SUR ISÈRE",
+ "nb_users" : 8,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-03-24T17:32:35.000Z",
+ "derniere_operation" : "2025-03-24T17:32:35.000Z",
+ "nb_passages" : 2200
+ },
+ {
+ "entity_id" : 1008,
+ "code_postal" : "38300",
+ "nom" : "2024 TEST AMICALE 38 NIVOLAS-VERMELLE",
+ "ville" : "NIVOLAS-VERMELLE ",
+ "nb_users" : 16,
+ "nb_operations" : 5,
+ "premiere_operation" : "2024-02-13T09:18:59.000Z",
+ "derniere_operation" : "2024-02-13T09:19:51.000Z",
+ "nb_passages" : 0
+ },
+ {
+ "entity_id" : 164,
+ "code_postal" : "38440",
+ "nom" : "2024 AMICALE 38 ST JEAN DE BOURNAY",
+ "ville" : "ST JEAN DE BOURNAY",
+ "nb_users" : 26,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-10-03T06:05:54.000Z",
+ "derniere_operation" : "2025-10-03T06:05:54.000Z",
+ "nb_passages" : 18780
+ },
+ {
+ "entity_id" : 1152,
+ "code_postal" : "38530",
+ "nom" : "2024 AMICALE 38 CHAPAREILLAN ",
+ "ville" : "CHAPAREILLAN ",
+ "nb_users" : 45,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-28T16:54:08.000Z",
+ "derniere_operation" : "2024-10-28T16:54:08.000Z",
+ "nb_passages" : 2376
+ },
+ {
+ "entity_id" : 852,
+ "code_postal" : "38630",
+ "nom" : "2024 AMICALE 38 DES 3 SIRèNES",
+ "ville" : "CORBELIN",
+ "nb_users" : 36,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-25T16:40:15.000Z",
+ "derniere_operation" : "2024-09-25T16:40:15.000Z",
+ "nb_passages" : 4337
+ },
+ {
+ "entity_id" : 933,
+ "code_postal" : "38650",
+ "nom" : "2024 AMICALE 38 MONESTIER DE CLERMONT",
+ "ville" : "MONESTIER DE CLERMONT",
+ "nb_users" : 13,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-25T13:51:01.000Z",
+ "derniere_operation" : "2024-10-25T13:51:01.000Z",
+ "nb_passages" : 3824
+ },
+ {
+ "entity_id" : 1082,
+ "code_postal" : "38850",
+ "nom" : "2024 AMICALE 38 CHARAVINES",
+ "ville" : "CHARAVINES",
+ "nb_users" : 14,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-05-28T16:28:05.000Z",
+ "derniere_operation" : "2025-05-28T16:28:05.000Z",
+ "nb_passages" : 4919
+ },
+ {
+ "entity_id" : 921,
+ "code_postal" : "38860",
+ "nom" : "2023 AMICALE 38 LES DEUX ALPES",
+ "ville" : "LES DEUX ALPES",
+ "nb_users" : 24,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-02T10:26:06.000Z",
+ "derniere_operation" : "2024-11-02T10:26:06.000Z",
+ "nb_passages" : 562
+ },
+ {
+ "entity_id" : 999,
+ "code_postal" : "38890",
+ "nom" : "2024 AMICALE 38 Val du Ver ",
+ "ville" : "SAINT CHEF",
+ "nb_users" : 15,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-07-12T16:47:21.000Z",
+ "derniere_operation" : "2024-07-12T16:47:21.000Z",
+ "nb_passages" : 2363
+ },
+ {
+ "entity_id" : 971,
+ "code_postal" : "39100",
+ "nom" : "2024 AMICALE 39 CHOISEY",
+ "ville" : "Choisey",
+ "nb_users" : 93,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-07-09T14:41:38.000Z",
+ "derniere_operation" : "2025-07-09T14:41:38.000Z",
+ "nb_passages" : 45976
+ },
+ {
+ "entity_id" : 752,
+ "code_postal" : "39150",
+ "nom" : "2024 AMICALE 39 SAINT LAURENT EN GRANDVAUX",
+ "ville" : "SAINT LAURENT EN GRANDVAUX ",
+ "nb_users" : 26,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-08-21T06:13:11.000Z",
+ "derniere_operation" : "2024-08-21T06:13:11.000Z",
+ "nb_passages" : 4101
+ },
+ {
+ "entity_id" : 656,
+ "code_postal" : "39210",
+ "nom" : "2024 AMICALE 39 LA MARRE",
+ "ville" : "LA MARRE",
+ "nb_users" : 17,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-12T20:24:21.000Z",
+ "derniere_operation" : "2024-11-12T20:24:21.000Z",
+ "nb_passages" : 1706
+ },
+ {
+ "entity_id" : 880,
+ "code_postal" : "39300",
+ "nom" : "2024 AMICALE 39 CHAMPAGNOLE",
+ "ville" : "CHAMPAGNOLE",
+ "nb_users" : 55,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-23T14:18:21.000Z",
+ "derniere_operation" : "2025-09-23T14:18:21.000Z",
+ "nb_passages" : 11432
+ },
+ {
+ "entity_id" : 756,
+ "code_postal" : "40090",
+ "nom" : "2024 AMICALE 40 SAINT PERDON",
+ "ville" : "SAINT PERDON",
+ "nb_users" : 15,
+ "nb_operations" : 2,
+ "premiere_operation" : "2024-07-09T16:15:39.000Z",
+ "derniere_operation" : "2025-03-15T15:34:25.000Z",
+ "nb_passages" : 7739
+ },
+ {
+ "entity_id" : 936,
+ "code_postal" : "40190",
+ "nom" : "2024 AMICALE 40 VILLENEUVE DE MARSAN ",
+ "ville" : "VILLENEUVE DE MARSAN",
+ "nb_users" : 24,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-08-27T20:09:37.000Z",
+ "derniere_operation" : "2024-08-27T20:09:37.000Z",
+ "nb_passages" : 3480
+ },
+ {
+ "entity_id" : 763,
+ "code_postal" : "40800",
+ "nom" : "2024 AMICALE 40 AIRE SUR L'ADOUR",
+ "ville" : "AIRE SUR L ADOUR",
+ "nb_users" : 34,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-05T14:08:29.000Z",
+ "derniere_operation" : "2024-10-05T14:08:29.000Z",
+ "nb_passages" : 12450
+ },
+ {
+ "entity_id" : 845,
+ "code_postal" : "41110",
+ "nom" : "2023 AMICALE 41 SAINT AIGNAN SUR CHER",
+ "ville" : "SAINT AIGNAN SUR CHER",
+ "nb_users" : 17,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-02-07T13:40:53.000Z",
+ "derniere_operation" : "2024-02-07T13:40:53.000Z",
+ "nb_passages" : 3362
+ },
+ {
+ "entity_id" : 991,
+ "code_postal" : "41130",
+ "nom" : "2024 AMICALE 41 MEUSNES COUFFY",
+ "ville" : "MEUSNES\/COUFFY",
+ "nb_users" : 15,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-12-06T19:53:50.000Z",
+ "derniere_operation" : "2024-12-06T19:53:50.000Z",
+ "nb_passages" : 1799
+ },
+ {
+ "entity_id" : 607,
+ "code_postal" : "41130",
+ "nom" : "2024 AMICALE 41 SELLES SUR CHER",
+ "ville" : "SELLES SUR CHER",
+ "nb_users" : 33,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-07-01T12:03:07.000Z",
+ "derniere_operation" : "2025-07-01T12:03:07.000Z",
+ "nb_passages" : 25861
+ },
+ {
+ "entity_id" : 1108,
+ "code_postal" : "41500",
+ "nom" : "2024 AMICALE 41 LA CHAPELLE SAINT MARTIN",
+ "ville" : "LA CHAPELLE SAINT MARTIN EN PLAINE",
+ "nb_users" : 22,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-30T06:47:51.000Z",
+ "derniere_operation" : "2024-09-30T06:47:51.000Z",
+ "nb_passages" : 1215
+ },
+ {
+ "entity_id" : 592,
+ "code_postal" : "41800",
+ "nom" : "2021 AMICALE 41 VALLéE DE RONSARD ",
+ "ville" : "VALLéE DE RONSARD",
+ "nb_users" : 16,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-19T10:02:10.000Z",
+ "derniere_operation" : "2024-10-19T10:02:10.000Z",
+ "nb_passages" : 3097
+ },
+ {
+ "entity_id" : 1201,
+ "code_postal" : "42110",
+ "nom" : "2025 TEST AMICALE 42 FEURS",
+ "ville" : "FEURS",
+ "nb_users" : 38,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-02-17T14:00:34.000Z",
+ "derniere_operation" : "2025-02-17T14:00:34.000Z",
+ "nb_passages" : 4289
+ },
+ {
+ "entity_id" : 1272,
+ "code_postal" : "42520",
+ "nom" : "2025 TEST AMICALE 42 SAINT PIERRE DE BOEUF",
+ "ville" : "SAINT PIERRE DE BOEUF",
+ "nb_users" : 27,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-08-02T06:09:33.000Z",
+ "derniere_operation" : "2025-08-02T06:09:33.000Z",
+ "nb_passages" : 789
+ },
+ {
+ "entity_id" : 369,
+ "code_postal" : "42630",
+ "nom" : "2024 AMICALE 42 VAL DE RHINS",
+ "ville" : "REGNY",
+ "nb_users" : 12,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-02-16T10:55:59.000Z",
+ "derniere_operation" : "2025-02-16T10:55:59.000Z",
+ "nb_passages" : 7126
+ },
+ {
+ "entity_id" : 986,
+ "code_postal" : "43210",
+ "nom" : "2025 AMICALE 43 BAS EN BASSET",
+ "ville" : "BAS EN BASSET",
+ "nb_users" : 35,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-11-17T08:03:46.000Z",
+ "derniere_operation" : "2023-11-17T08:03:46.000Z",
+ "nb_passages" : 3030
+ },
+ {
+ "entity_id" : 899,
+ "code_postal" : "44130",
+ "nom" : "2024 AMICALE 44 BOUVRON",
+ "ville" : "BOUVRON",
+ "nb_users" : 20,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-10-07T18:07:06.000Z",
+ "derniere_operation" : "2025-10-07T18:07:06.000Z",
+ "nb_passages" : 5253
+ },
+ {
+ "entity_id" : 1016,
+ "code_postal" : "44130",
+ "nom" : "2024 AMICALE 44 FAY DE BRETAGNE",
+ "ville" : "FAY DE BRETAGNE",
+ "nb_users" : 18,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-22T19:11:17.000Z",
+ "derniere_operation" : "2025-09-22T19:11:17.000Z",
+ "nb_passages" : 4005
+ },
+ {
+ "entity_id" : 859,
+ "code_postal" : "44220",
+ "nom" : "2023 AMICALE 44 COUERON",
+ "ville" : "COUERON",
+ "nb_users" : 23,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-07-26T08:29:00.000Z",
+ "derniere_operation" : "2025-07-26T08:29:00.000Z",
+ "nb_passages" : 9986
+ },
+ {
+ "entity_id" : 1269,
+ "code_postal" : "44250",
+ "nom" : "2025 AMICALE 44 SAINT BREVIN LES PINS",
+ "ville" : "SAINT BREVIN LES PINS",
+ "nb_users" : 11,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-07-08T21:08:06.000Z",
+ "derniere_operation" : "2025-07-08T21:08:06.000Z",
+ "nb_passages" : 9438
+ },
+ {
+ "entity_id" : 120,
+ "code_postal" : "44270",
+ "nom" : "2024 AMICALE 44 MACHECOUL",
+ "ville" : "MACHECOUL",
+ "nb_users" : 40,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-29T11:00:54.000Z",
+ "derniere_operation" : "2024-10-29T11:00:54.000Z",
+ "nb_passages" : 19200
+ },
+ {
+ "entity_id" : 1186,
+ "code_postal" : "44370",
+ "nom" : "2025 AMICALE 44 VARADES",
+ "ville" : "LOIREAUXENCE",
+ "nb_users" : 17,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-01-24T13:45:12.000Z",
+ "derniere_operation" : "2025-01-24T13:45:12.000Z",
+ "nb_passages" : 4337
+ },
+ {
+ "entity_id" : 53,
+ "code_postal" : "44390",
+ "nom" : "2024 AMICALE 44 NORT SUR ERDRE",
+ "ville" : "NORT SUR ERDRE",
+ "nb_users" : 34,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-15T08:53:23.000Z",
+ "derniere_operation" : "2025-09-15T08:53:23.000Z",
+ "nb_passages" : 66820
+ },
+ {
+ "entity_id" : 605,
+ "code_postal" : "44430",
+ "nom" : "2025 AMICALE 44 LE LOROUX BOTTEREAU",
+ "ville" : "LE LOROUX BOTTEREAU",
+ "nb_users" : 58,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-30T17:39:33.000Z",
+ "derniere_operation" : "2025-09-30T17:39:33.000Z",
+ "nb_passages" : 32883
+ },
+ {
+ "entity_id" : 1035,
+ "code_postal" : "44490",
+ "nom" : "2024 AMICALE 44 LE CROISIC ",
+ "ville" : "LE CROISIC",
+ "nb_users" : 16,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-08T15:20:02.000Z",
+ "derniere_operation" : "2025-09-08T15:20:02.000Z",
+ "nb_passages" : 7286
+ },
+ {
+ "entity_id" : 1274,
+ "code_postal" : "44570",
+ "nom" : "2025 AMICALE 44 TRIGNAC",
+ "ville" : "TRIGNAC",
+ "nb_users" : 30,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-08-11T13:46:36.000Z",
+ "derniere_operation" : "2025-08-11T13:46:36.000Z",
+ "nb_passages" : 1876
+ },
+ {
+ "entity_id" : 847,
+ "code_postal" : "44590",
+ "nom" : "2024 AMICALE 44 DERVAL",
+ "ville" : "DERVAL",
+ "nb_users" : 18,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-10-07T13:20:06.000Z",
+ "derniere_operation" : "2025-10-07T13:20:06.000Z",
+ "nb_passages" : 9276
+ },
+ {
+ "entity_id" : 622,
+ "code_postal" : "44680",
+ "nom" : "2024 AMICALE 44 SAINT PAZANNE",
+ "ville" : "SAINTE-PAZANNE",
+ "nb_users" : 53,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-07-01T11:53:35.000Z",
+ "derniere_operation" : "2025-07-01T11:53:35.000Z",
+ "nb_passages" : 31765
+ },
+ {
+ "entity_id" : 892,
+ "code_postal" : "44750",
+ "nom" : "2024 AMICALE 44 CAMPBON",
+ "ville" : "CAMPBON",
+ "nb_users" : 41,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-11T16:33:36.000Z",
+ "derniere_operation" : "2024-09-11T16:33:36.000Z",
+ "nb_passages" : 7058
+ },
+ {
+ "entity_id" : 244,
+ "code_postal" : "44810",
+ "nom" : "2024 AMICALE 44 HERIC",
+ "ville" : "HERIC",
+ "nb_users" : 22,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-21T06:21:43.000Z",
+ "derniere_operation" : "2025-09-21T06:21:43.000Z",
+ "nb_passages" : 27059
+ },
+ {
+ "entity_id" : 968,
+ "code_postal" : "44830",
+ "nom" : "2023 AMICALE 44 BRAINS ",
+ "ville" : "BRAINS",
+ "nb_users" : 8,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-10-27T11:22:46.000Z",
+ "derniere_operation" : "2023-10-27T11:22:46.000Z",
+ "nb_passages" : 378
+ },
+ {
+ "entity_id" : 914,
+ "code_postal" : "45110",
+ "nom" : "2025 AMICALE 45 CHATEAUNEUF \/ LOIRE",
+ "ville" : "CHATEAUNEUF \/ LOIRE",
+ "nb_users" : 38,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-10-06T07:23:38.000Z",
+ "derniere_operation" : "2025-10-06T07:23:38.000Z",
+ "nb_passages" : 4100
+ },
+ {
+ "entity_id" : 1209,
+ "code_postal" : "45150",
+ "nom" : "2025 TEST AMICALE 45 JARGEAU ",
+ "ville" : "JARGEAU",
+ "nb_users" : 7,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-03-01T12:23:36.000Z",
+ "derniere_operation" : "2025-03-01T12:23:36.000Z",
+ "nb_passages" : 3181
+ },
+ {
+ "entity_id" : 1136,
+ "code_postal" : "45170",
+ "nom" : "2024 TEST AMICALE 45 ASCHERES LE MARCHE",
+ "ville" : "ASCHERES LE MARCHE",
+ "nb_users" : 11,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-15T08:38:26.000Z",
+ "derniere_operation" : "2024-10-15T08:38:26.000Z",
+ "nb_passages" : 246
+ },
+ {
+ "entity_id" : 774,
+ "code_postal" : "45380",
+ "nom" : "2024 AMICALE 45 CHAINGY",
+ "ville" : "CHAINGY",
+ "nb_users" : 17,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-26T09:52:02.000Z",
+ "derniere_operation" : "2024-09-26T09:52:02.000Z",
+ "nb_passages" : 4976
+ },
+ {
+ "entity_id" : 1184,
+ "code_postal" : "45380",
+ "nom" : "2025 TEST AMICALE 45 LA CHAPELLE SAINT MESMIN",
+ "ville" : "LA CHAPELLE SAINT MESMIN",
+ "nb_users" : 69,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-01-23T13:27:52.000Z",
+ "derniere_operation" : "2025-01-23T13:27:52.000Z",
+ "nb_passages" : 4026
+ },
+ {
+ "entity_id" : 685,
+ "code_postal" : "45500",
+ "nom" : "2024 AMICALE 45 GIEN",
+ "ville" : "GIEN",
+ "nb_users" : 40,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-08-14T15:19:01.000Z",
+ "derniere_operation" : "2024-08-14T15:19:01.000Z",
+ "nb_passages" : 16413
+ },
+ {
+ "entity_id" : 1268,
+ "code_postal" : "45700",
+ "nom" : "2025 TEST AMICALE MONTARGIS",
+ "ville" : "VILLEMANDEUR",
+ "nb_users" : 63,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-07-08T09:25:59.000Z",
+ "derniere_operation" : "2025-07-08T09:25:59.000Z",
+ "nb_passages" : 23347
+ },
+ {
+ "entity_id" : 1192,
+ "code_postal" : "46100",
+ "nom" : "2025 TEST AMICALE 46 FIGEAC",
+ "ville" : "FIGEAC",
+ "nb_users" : 40,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-02-06T21:11:19.000Z",
+ "derniere_operation" : "2025-02-06T21:11:19.000Z",
+ "nb_passages" : 0
+ },
+ {
+ "entity_id" : 532,
+ "code_postal" : "46110",
+ "nom" : "2023 AMICALE 46 VAYRAC-BETAILLE",
+ "ville" : "VAYRAC-BETAILLE",
+ "nb_users" : 46,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-10-07T08:42:33.000Z",
+ "derniere_operation" : "2023-10-07T08:42:33.000Z",
+ "nb_passages" : 41
+ },
+ {
+ "entity_id" : 1250,
+ "code_postal" : "46110",
+ "nom" : "2025 TEST AMICALE 46 QUATRE ROUTES",
+ "ville" : "QUATRE ROUTES",
+ "nb_users" : 32,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-05-27T10:10:00.000Z",
+ "derniere_operation" : "2025-05-27T10:10:00.000Z",
+ "nb_passages" : 2117
+ },
+ {
+ "entity_id" : 484,
+ "code_postal" : "46130",
+ "nom" : "2024 AMICALE 46 BRETENOUX",
+ "ville" : "BRETENOUX",
+ "nb_users" : 57,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-18T16:55:12.000Z",
+ "derniere_operation" : "2025-09-18T16:55:12.000Z",
+ "nb_passages" : 33901
+ },
+ {
+ "entity_id" : 531,
+ "code_postal" : "46140",
+ "nom" : "2023 AMICALE 46 LUZECH",
+ "ville" : "LUZECH",
+ "nb_users" : 13,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-31T17:46:01.000Z",
+ "derniere_operation" : "2024-10-31T17:46:01.000Z",
+ "nb_passages" : 11600
+ },
+ {
+ "entity_id" : 1189,
+ "code_postal" : "46150",
+ "nom" : "2025 TEST AMICALE 46 CATUS ",
+ "ville" : "CATUS",
+ "nb_users" : 14,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-01-29T09:23:07.000Z",
+ "derniere_operation" : "2025-01-29T09:23:07.000Z",
+ "nb_passages" : 991
+ },
+ {
+ "entity_id" : 946,
+ "code_postal" : "46160",
+ "nom" : "2024 AMICALE 46 CAJARC",
+ "ville" : "CAJARC",
+ "nb_users" : 41,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-03T08:28:20.000Z",
+ "derniere_operation" : "2024-11-03T08:28:20.000Z",
+ "nb_passages" : 6071
+ },
+ {
+ "entity_id" : 590,
+ "code_postal" : "46190",
+ "nom" : "2023 AMICALE 46 SOUSCEYRAC EN QUERCY",
+ "ville" : "SOUSCEYRAC EN QUERCY",
+ "nb_users" : 16,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-10-30T20:27:32.000Z",
+ "derniere_operation" : "2023-10-30T20:27:32.000Z",
+ "nb_passages" : 3711
+ },
+ {
+ "entity_id" : 296,
+ "code_postal" : "46200",
+ "nom" : "2024 AMICALE 46 SOUILLAC",
+ "ville" : "SOUILLAC",
+ "nb_users" : 16,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-10T12:42:46.000Z",
+ "derniere_operation" : "2024-09-10T12:42:46.000Z",
+ "nb_passages" : 17609
+ },
+ {
+ "entity_id" : 530,
+ "code_postal" : "46240",
+ "nom" : "2024 AMICALE 46 LABASTIDE-MURAT",
+ "ville" : "LABASTIDE MURAT",
+ "nb_users" : 13,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-08-25T20:52:43.000Z",
+ "derniere_operation" : "2025-08-25T20:52:43.000Z",
+ "nb_passages" : 23547
+ },
+ {
+ "entity_id" : 529,
+ "code_postal" : "46260",
+ "nom" : "2024 AMICALE 46 LIMOGNE EN QUERCY",
+ "ville" : "LIMOGNE EN QUERCY",
+ "nb_users" : 18,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-03T14:07:41.000Z",
+ "derniere_operation" : "2024-10-03T14:07:41.000Z",
+ "nb_passages" : 3383
+ },
+ {
+ "entity_id" : 487,
+ "code_postal" : "46300",
+ "nom" : "2024 AMICALE 46 GOURDON",
+ "ville" : "GOURDON",
+ "nb_users" : 72,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-24T11:06:31.000Z",
+ "derniere_operation" : "2024-10-24T11:06:31.000Z",
+ "nb_passages" : 34672
+ },
+ {
+ "entity_id" : 533,
+ "code_postal" : "46320",
+ "nom" : "2023 AMICALE 46 LIVERNON",
+ "ville" : "LIVERNON",
+ "nb_users" : 40,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-01T12:55:39.000Z",
+ "derniere_operation" : "2024-10-01T12:55:39.000Z",
+ "nb_passages" : 10584
+ },
+ {
+ "entity_id" : 528,
+ "code_postal" : "46340",
+ "nom" : "2024 AMICALE 46 SALVIAC",
+ "ville" : "SALVIAC",
+ "nb_users" : 9,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-11T09:10:51.000Z",
+ "derniere_operation" : "2024-09-11T09:10:51.000Z",
+ "nb_passages" : 2344
+ },
+ {
+ "entity_id" : 450,
+ "code_postal" : "46500",
+ "nom" : "2023 AMICALE 46 GRAMAT ",
+ "ville" : "GRAMAT",
+ "nb_users" : 49,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-10-02T14:00:00.000Z",
+ "derniere_operation" : "2025-10-02T14:00:00.000Z",
+ "nb_passages" : 27032
+ },
+ {
+ "entity_id" : 634,
+ "code_postal" : "46600",
+ "nom" : "2024 AMICALE 46 MARTEL",
+ "ville" : "MARTEL",
+ "nb_users" : 13,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-18T17:25:42.000Z",
+ "derniere_operation" : "2025-09-18T17:25:42.000Z",
+ "nb_passages" : 4907
+ },
+ {
+ "entity_id" : 527,
+ "code_postal" : "46800",
+ "nom" : "2023 AMICALE 46 MONTCUQ",
+ "ville" : "MONTCUQ",
+ "nb_users" : 15,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-14T20:28:16.000Z",
+ "derniere_operation" : "2024-09-14T20:28:16.000Z",
+ "nb_passages" : 8345
+ },
+ {
+ "entity_id" : 1050,
+ "code_postal" : "47520",
+ "nom" : "2024 AMICALE 47 LE PASSAGE D'AGEN ",
+ "ville" : "LE PASSAGE D'AGEN",
+ "nb_users" : 49,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-07-23T19:44:03.000Z",
+ "derniere_operation" : "2025-07-23T19:44:03.000Z",
+ "nb_passages" : 21182
+ },
+ {
+ "entity_id" : 1235,
+ "code_postal" : "47800",
+ "nom" : "2025 TEST AMICALE 47 MIRAMONT DE GUYENNE",
+ "ville" : "MIRAMONT",
+ "nb_users" : 25,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-04-15T12:54:20.000Z",
+ "derniere_operation" : "2025-04-15T12:54:20.000Z",
+ "nb_passages" : 3674
+ },
+ {
+ "entity_id" : 599,
+ "code_postal" : "49123",
+ "nom" : "2024 AMICALE 49 LOIRE ET AUXENCE",
+ "ville" : "CHAMPTOCE SUR LOIRE",
+ "nb_users" : 55,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-18T17:13:27.000Z",
+ "derniere_operation" : "2025-09-18T17:13:27.000Z",
+ "nb_passages" : 15662
+ },
+ {
+ "entity_id" : 160,
+ "code_postal" : "49150",
+ "nom" : "2024 AMICALE 49 BAUGé EN ANJOU",
+ "ville" : "BAUGE EN ANJOU",
+ "nb_users" : 50,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-05-10T13:45:58.000Z",
+ "derniere_operation" : "2025-05-10T13:45:58.000Z",
+ "nb_passages" : 44781
+ },
+ {
+ "entity_id" : 723,
+ "code_postal" : "49160",
+ "nom" : "2024 AMICALE 49 LONGUé-JUMELLES",
+ "ville" : "LONGUé-JUMELLES",
+ "nb_users" : 34,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-08-28T15:59:56.000Z",
+ "derniere_operation" : "2025-08-28T15:59:56.000Z",
+ "nb_passages" : 12989
+ },
+ {
+ "entity_id" : 281,
+ "code_postal" : "49220",
+ "nom" : "2024 AMICALE 49 LE LION D ANGERS ",
+ "ville" : "LE LION D ANGERS",
+ "nb_users" : 33,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-02-25T23:08:34.000Z",
+ "derniere_operation" : "2024-02-25T23:08:34.000Z",
+ "nb_passages" : 21669
+ },
+ {
+ "entity_id" : 455,
+ "code_postal" : "49220",
+ "nom" : "2024 AMICALE 49 VERN D'ANJOU ",
+ "ville" : "VERN D'ANJOU",
+ "nb_users" : 23,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-05T09:19:14.000Z",
+ "derniere_operation" : "2024-09-05T09:19:14.000Z",
+ "nb_passages" : 9913
+ },
+ {
+ "entity_id" : 522,
+ "code_postal" : "49250",
+ "nom" : "2024 AMICALE 49 BEAUFORT-EN-ANJOU PAS 2024 ",
+ "ville" : "BEAUFORT-EN-ANJOU",
+ "nb_users" : 50,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-08-07T12:50:43.000Z",
+ "derniere_operation" : "2023-08-07T12:50:43.000Z",
+ "nb_passages" : 6147
+ },
+ {
+ "entity_id" : 811,
+ "code_postal" : "49250",
+ "nom" : "2024 AMICALE 49 SAINT MATHURIN SUR LOIRE",
+ "ville" : "SAINT MATHURIN SUR LOIRE",
+ "nb_users" : 27,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-21T18:06:03.000Z",
+ "derniere_operation" : "2024-10-21T18:06:03.000Z",
+ "nb_passages" : 4897
+ },
+ {
+ "entity_id" : 687,
+ "code_postal" : "49330",
+ "nom" : "2024 AMICALE 49 CHAMPIGNE",
+ "ville" : "CHAMPIGNE",
+ "nb_users" : 39,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-05-15T12:47:53.000Z",
+ "derniere_operation" : "2025-05-15T12:47:53.000Z",
+ "nb_passages" : 1888
+ },
+ {
+ "entity_id" : 736,
+ "code_postal" : "49330",
+ "nom" : "2024 AMICALE 49 CHATEAUNEUF SUR SARTHE",
+ "ville" : "LES HAUTS D'ANJOU",
+ "nb_users" : 34,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-10-04T19:38:26.000Z",
+ "derniere_operation" : "2025-10-04T19:38:26.000Z",
+ "nb_passages" : 14752
+ },
+ {
+ "entity_id" : 710,
+ "code_postal" : "49420",
+ "nom" : "2022 AMICALE 49 POUANCE",
+ "ville" : "POUANCE",
+ "nb_users" : 43,
+ "nb_operations" : 3,
+ "premiere_operation" : "2023-10-06T14:33:47.000Z",
+ "derniere_operation" : "2023-10-06T14:38:42.000Z",
+ "nb_passages" : 12660
+ },
+ {
+ "entity_id" : 724,
+ "code_postal" : "49490",
+ "nom" : "2022 AMICALE 49 NOYANT",
+ "ville" : "NOYANT",
+ "nb_users" : 23,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-11-04T10:23:28.000Z",
+ "derniere_operation" : "2023-11-04T10:23:28.000Z",
+ "nb_passages" : 4613
+ },
+ {
+ "entity_id" : 740,
+ "code_postal" : "49500",
+ "nom" : "2024 AMICALE 49 SAINT MARTIN DU BOIS",
+ "ville" : "SAINT MARTIN DU BOIS",
+ "nb_users" : 11,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-17T18:11:50.000Z",
+ "derniere_operation" : "2025-09-17T18:11:50.000Z",
+ "nb_passages" : 2406
+ },
+ {
+ "entity_id" : 732,
+ "code_postal" : "49500",
+ "nom" : "2024 AMICALE 49 SEGRE",
+ "ville" : "SEGRE",
+ "nb_users" : 65,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-11T10:57:20.000Z",
+ "derniere_operation" : "2024-10-11T10:57:20.000Z",
+ "nb_passages" : 5413
+ },
+ {
+ "entity_id" : 729,
+ "code_postal" : "49520",
+ "nom" : "2024 AMICALE 49 L'ARAIZE",
+ "ville" : "BOUILLE MENARD",
+ "nb_users" : 19,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-05T19:37:35.000Z",
+ "derniere_operation" : "2024-11-05T19:37:35.000Z",
+ "nb_passages" : 2075
+ },
+ {
+ "entity_id" : 1265,
+ "code_postal" : "49590",
+ "nom" : "2025 TEST AMICALE ",
+ "ville" : "FONTEVRAUD-L'ABBAYE",
+ "nb_users" : 17,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-07-01T02:39:42.000Z",
+ "derniere_operation" : "2025-07-01T02:39:42.000Z",
+ "nb_passages" : 611
+ },
+ {
+ "entity_id" : 514,
+ "code_postal" : "49620",
+ "nom" : "2025 AMICALE 49 LE PÉLICAN",
+ "ville" : "LE PÉLICAN",
+ "nb_users" : 68,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-07-10T19:01:34.000Z",
+ "derniere_operation" : "2025-07-10T19:01:34.000Z",
+ "nb_passages" : 17037
+ },
+ {
+ "entity_id" : 978,
+ "code_postal" : "50140",
+ "nom" : "2024 AMICALE 50 MORTAIN BOCAGE",
+ "ville" : "MORTAIN BOCAGE",
+ "nb_users" : 7,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-10T07:29:58.000Z",
+ "derniere_operation" : "2024-10-10T07:29:58.000Z",
+ "nb_passages" : 1019
+ },
+ {
+ "entity_id" : 855,
+ "code_postal" : "50160",
+ "nom" : "2024 AMICALE 50 TORIGNY LES VILLES",
+ "ville" : "TORIGNY LES VILLES",
+ "nb_users" : 25,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-02T14:31:21.000Z",
+ "derniere_operation" : "2024-10-02T14:31:21.000Z",
+ "nb_passages" : 13070
+ },
+ {
+ "entity_id" : 69,
+ "code_postal" : "50190",
+ "nom" : "2024 AMICALE 50 PERIERS",
+ "ville" : "PERIERS",
+ "nb_users" : 36,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-11T17:44:47.000Z",
+ "derniere_operation" : "2025-09-11T17:44:47.000Z",
+ "nb_passages" : 9608
+ },
+ {
+ "entity_id" : 950,
+ "code_postal" : "50200",
+ "nom" : "2024 AMICALE 50 COUTANCES",
+ "ville" : "COUTANCES",
+ "nb_users" : 58,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-21T12:19:53.000Z",
+ "derniere_operation" : "2025-09-21T12:19:53.000Z",
+ "nb_passages" : 23682
+ },
+ {
+ "entity_id" : 850,
+ "code_postal" : "50340",
+ "nom" : "2024 AMICALE 50 LES PIEUX",
+ "ville" : "LES PIEUX",
+ "nb_users" : 52,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-06-24T07:55:32.000Z",
+ "derniere_operation" : "2025-06-24T07:55:32.000Z",
+ "nb_passages" : 26304
+ },
+ {
+ "entity_id" : 1285,
+ "code_postal" : "50410",
+ "nom" : "2025 AMICALE 50 PERCY",
+ "ville" : "PERCY",
+ "nb_users" : 30,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-10T15:34:37.000Z",
+ "derniere_operation" : "2025-09-10T15:34:37.000Z",
+ "nb_passages" : 2633
+ },
+ {
+ "entity_id" : 830,
+ "code_postal" : "50420",
+ "nom" : "2024 AMICALE 50 TESSY-BOCAGE",
+ "ville" : "TESSY-BOCAGE",
+ "nb_users" : 23,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-26T19:16:28.000Z",
+ "derniere_operation" : "2024-10-26T19:16:28.000Z",
+ "nb_passages" : 5022
+ },
+ {
+ "entity_id" : 1068,
+ "code_postal" : "50430",
+ "nom" : "2024 AMICALE 50 LESSAY ET DE PIROU",
+ "ville" : "LESSAY ET DE PIROU",
+ "nb_users" : 40,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-02-12T08:36:23.000Z",
+ "derniere_operation" : "2025-02-12T08:36:23.000Z",
+ "nb_passages" : 17222
+ },
+ {
+ "entity_id" : 682,
+ "code_postal" : "50440",
+ "nom" : "2024 AMICALE 50 LA HAGUE",
+ "ville" : "LA HAGUE",
+ "nb_users" : 26,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-10T14:04:45.000Z",
+ "derniere_operation" : "2024-10-10T14:04:45.000Z",
+ "nb_passages" : 11284
+ },
+ {
+ "entity_id" : 1212,
+ "code_postal" : "50570",
+ "nom" : "2025 AMICALE 50 MARIGNY",
+ "ville" : "MARIGNY",
+ "nb_users" : 28,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-03-09T14:29:44.000Z",
+ "derniere_operation" : "2025-03-09T14:29:44.000Z",
+ "nb_passages" : 4313
+ },
+ {
+ "entity_id" : 1094,
+ "code_postal" : "51800",
+ "nom" : "2024 TEST AMICALE SAINTE MENEHOULD",
+ "ville" : "SAINTE MENEHOULD",
+ "nb_users" : 8,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-26T06:03:07.000Z",
+ "derniere_operation" : "2024-09-26T06:03:07.000Z",
+ "nb_passages" : 807
+ },
+ {
+ "entity_id" : 802,
+ "code_postal" : "52290",
+ "nom" : "2024 AMICALE 52 ECLARON",
+ "ville" : "ECLARON ",
+ "nb_users" : 30,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-03T19:48:46.000Z",
+ "derniere_operation" : "2024-11-03T19:48:46.000Z",
+ "nb_passages" : 5559
+ },
+ {
+ "entity_id" : 721,
+ "code_postal" : "53140",
+ "nom" : "2024 AMICALE 53 PRE EN PAIL",
+ "ville" : "PRE EN PAIL",
+ "nb_users" : 30,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-07-15T07:25:23.000Z",
+ "derniere_operation" : "2025-07-15T07:25:23.000Z",
+ "nb_passages" : 14283
+ },
+ {
+ "entity_id" : 987,
+ "code_postal" : "53200",
+ "nom" : "2024 AMICALE 53 CHâTEAU-GONTIER",
+ "ville" : "CHÂTEAU-GONTIER",
+ "nb_users" : 11,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-30T15:22:38.000Z",
+ "derniere_operation" : "2024-10-30T15:22:38.000Z",
+ "nb_passages" : 3248
+ },
+ {
+ "entity_id" : 1002,
+ "code_postal" : "53240",
+ "nom" : "2024 AMICALE 53 LA BACONNIèRE",
+ "ville" : "LA BACONNIèRE",
+ "nb_users" : 15,
+ "nb_operations" : 2,
+ "premiere_operation" : "2024-04-08T19:42:37.000Z",
+ "derniere_operation" : "2025-09-24T18:48:26.000Z",
+ "nb_passages" : 1814
+ },
+ {
+ "entity_id" : 1065,
+ "code_postal" : "53240",
+ "nom" : "2024 association Sportive d'Andouillé",
+ "ville" : "ANDOUILLE",
+ "nb_users" : 33,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-07-17T13:05:50.000Z",
+ "derniere_operation" : "2024-07-17T13:05:50.000Z",
+ "nb_passages" : 1635
+ },
+ {
+ "entity_id" : 1042,
+ "code_postal" : "53360",
+ "nom" : "2024 AMICALE 53 QUELAINES SAINT GAULT",
+ "ville" : "QUELAINES SAINT GAULT",
+ "nb_users" : 38,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-25T20:50:25.000Z",
+ "derniere_operation" : "2024-09-25T20:50:25.000Z",
+ "nb_passages" : 1616
+ },
+ {
+ "entity_id" : 856,
+ "code_postal" : "53410",
+ "nom" : "2024 AMICALE 53 PORT BRILLET",
+ "ville" : "PORT BRILLET",
+ "nb_users" : 35,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-03-21T18:48:26.000Z",
+ "derniere_operation" : "2025-03-21T18:48:26.000Z",
+ "nb_passages" : 12969
+ },
+ {
+ "entity_id" : 981,
+ "code_postal" : "53540",
+ "nom" : "2024 AMICALE 53 LAUBRIERE",
+ "ville" : "LAUBRIERE",
+ "nb_users" : 17,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-03-29T12:51:34.000Z",
+ "derniere_operation" : "2025-03-29T12:51:34.000Z",
+ "nb_passages" : 20953
+ },
+ {
+ "entity_id" : 1119,
+ "code_postal" : "53540",
+ "nom" : "2024 AMICALE 61 COUTERNE (secteur Mayenne)",
+ "ville" : "LAUBRIERE",
+ "nb_users" : 17,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-04T15:08:18.000Z",
+ "derniere_operation" : "2024-10-04T15:08:18.000Z",
+ "nb_passages" : 1340
+ },
+ {
+ "entity_id" : 11,
+ "code_postal" : "56000",
+ "nom" : "2023 AMICALE 56 VANNES",
+ "ville" : "VANNES",
+ "nb_users" : 99,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-14T09:16:33.000Z",
+ "derniere_operation" : "2025-09-14T09:16:33.000Z",
+ "nb_passages" : 9638
+ },
+ {
+ "entity_id" : 28,
+ "code_postal" : "56120",
+ "nom" : "2024 AMICALE 56 JOSSELIN ",
+ "ville" : "JOSSELIN",
+ "nb_users" : 38,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-26T08:22:17.000Z",
+ "derniere_operation" : "2024-09-26T08:22:17.000Z",
+ "nb_passages" : 7664
+ },
+ {
+ "entity_id" : 870,
+ "code_postal" : "56130",
+ "nom" : "2023 AMICALE 56 LA ROCHE BERNARD",
+ "ville" : "NIVILLAC",
+ "nb_users" : 49,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-10-20T15:14:36.000Z",
+ "derniere_operation" : "2023-10-20T15:14:36.000Z",
+ "nb_passages" : 3922
+ },
+ {
+ "entity_id" : 903,
+ "code_postal" : "56160",
+ "nom" : "2024 AMICALE 56 GUEMENE SUR SCORFF",
+ "ville" : "GUEMENE SUR SCORFF",
+ "nb_users" : 38,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-30T10:42:03.000Z",
+ "derniere_operation" : "2024-11-30T10:42:03.000Z",
+ "nb_passages" : 29578
+ },
+ {
+ "entity_id" : 411,
+ "code_postal" : "56200",
+ "nom" : "2021 AMICALE 56 LA GACILLY ",
+ "ville" : "LA GACILLY",
+ "nb_users" : 46,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-29T07:06:37.000Z",
+ "derniere_operation" : "2024-10-29T07:06:37.000Z",
+ "nb_passages" : 15496
+ },
+ {
+ "entity_id" : 133,
+ "code_postal" : "56240",
+ "nom" : "2024 AMICALE 56 PLOUAY",
+ "ville" : "PLOUAY",
+ "nb_users" : 21,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-02T09:06:27.000Z",
+ "derniere_operation" : "2025-09-02T09:06:27.000Z",
+ "nb_passages" : 11729
+ },
+ {
+ "entity_id" : 1084,
+ "code_postal" : "56250",
+ "nom" : "2024 AMICALE 56 ELVEN",
+ "ville" : "ELVEN",
+ "nb_users" : 46,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-14T20:00:58.000Z",
+ "derniere_operation" : "2024-09-14T20:00:58.000Z",
+ "nb_passages" : 8821
+ },
+ {
+ "entity_id" : 1208,
+ "code_postal" : "56300",
+ "nom" : "2025 AMICALE 56 PONTIVY ",
+ "ville" : "PONTIVY",
+ "nb_users" : 63,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-02-27T07:26:46.000Z",
+ "derniere_operation" : "2025-02-27T07:26:46.000Z",
+ "nb_passages" : 3840
+ },
+ {
+ "entity_id" : 1080,
+ "code_postal" : "56330",
+ "nom" : "2024 AMICALE 56 PLUVIGNER",
+ "ville" : "PLUVIGNER",
+ "nb_users" : 41,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-07-26T12:21:11.000Z",
+ "derniere_operation" : "2025-07-26T12:21:11.000Z",
+ "nb_passages" : 22167
+ },
+ {
+ "entity_id" : 563,
+ "code_postal" : "56390",
+ "nom" : "COMPTE GRAND-CHAMP FRED",
+ "ville" : "GRAND-CHAMP",
+ "nb_users" : 10,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-10T16:23:22.000Z",
+ "derniere_operation" : "2025-09-10T16:23:22.000Z",
+ "nb_passages" : 1443
+ },
+ {
+ "entity_id" : 1153,
+ "code_postal" : "56420",
+ "nom" : "2024 AMICALE 56 PLUMELEC",
+ "ville" : "PLUMELEC",
+ "nb_users" : 35,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-29T10:49:18.000Z",
+ "derniere_operation" : "2024-10-29T10:49:18.000Z",
+ "nb_passages" : 3800
+ },
+ {
+ "entity_id" : 1227,
+ "code_postal" : "56440",
+ "nom" : "2025 TEST AMICALE 56 LANGUDIC ",
+ "ville" : "LANGUIDIC",
+ "nb_users" : 32,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-03-26T15:30:08.000Z",
+ "derniere_operation" : "2025-03-26T15:30:08.000Z",
+ "nb_passages" : 3898
+ },
+ {
+ "entity_id" : 1123,
+ "code_postal" : "56490",
+ "nom" : "2024 AMICALE 56 MENEAC",
+ "ville" : "MENEAC",
+ "nb_users" : 9,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-07T16:35:04.000Z",
+ "derniere_operation" : "2024-10-07T16:35:04.000Z",
+ "nb_passages" : 1432
+ },
+ {
+ "entity_id" : 591,
+ "code_postal" : "56490",
+ "nom" : "PAS 2023 AMICALE 56 LA TRINITE PORHOET",
+ "ville" : "LA TRINITE PORHOET",
+ "nb_users" : 23,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-10T12:53:00.000Z",
+ "derniere_operation" : "2024-10-10T12:53:00.000Z",
+ "nb_passages" : 3671
+ },
+ {
+ "entity_id" : 1261,
+ "code_postal" : "56510",
+ "nom" : "2025 AMICALE 56 QUIBERON",
+ "ville" : "QUIBERON",
+ "nb_users" : 10,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-06-17T07:06:02.000Z",
+ "derniere_operation" : "2025-06-17T07:06:02.000Z",
+ "nb_passages" : 4249
+ },
+ {
+ "entity_id" : 423,
+ "code_postal" : "56520",
+ "nom" : "2023 AMICALE 56 GUIDEL",
+ "ville" : "GUIDEL",
+ "nb_users" : 35,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-10-05T09:15:14.000Z",
+ "derniere_operation" : "2023-10-05T09:15:14.000Z",
+ "nb_passages" : 12552
+ },
+ {
+ "entity_id" : 1284,
+ "code_postal" : "56590",
+ "nom" : "2025 TEST AMICALE 56 GROIX ",
+ "ville" : "GROIX",
+ "nb_users" : 18,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-09T15:17:26.000Z",
+ "derniere_operation" : "2025-09-09T15:17:26.000Z",
+ "nb_passages" : 2236
+ },
+ {
+ "entity_id" : 1085,
+ "code_postal" : "56660",
+ "nom" : "2024 AMICALE 56 ST JEAN BREVELAY",
+ "ville" : "ST JEAN BREVELAY",
+ "nb_users" : 22,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-01-29T16:17:50.000Z",
+ "derniere_operation" : "2025-01-29T16:17:50.000Z",
+ "nb_passages" : 7903
+ },
+ {
+ "entity_id" : 918,
+ "code_postal" : "56770",
+ "nom" : "2023 AMICALE 56 PLOURAY",
+ "ville" : "PLOURAY",
+ "nb_users" : 20,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-09-14T08:04:40.000Z",
+ "derniere_operation" : "2023-09-14T08:04:40.000Z",
+ "nb_passages" : 0
+ },
+ {
+ "entity_id" : 129,
+ "code_postal" : "56800",
+ "nom" : "2024 AMICALE 56 CAMPENEAC",
+ "ville" : "CAMPENEAC",
+ "nb_users" : 32,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-22T17:15:55.000Z",
+ "derniere_operation" : "2024-10-22T17:15:55.000Z",
+ "nb_passages" : 8769
+ },
+ {
+ "entity_id" : 9,
+ "code_postal" : "56800",
+ "nom" : "2024 AMICALE 56 PLOERMEL",
+ "ville" : "PLOERMEL",
+ "nb_users" : 59,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-23T18:37:32.000Z",
+ "derniere_operation" : "2025-09-23T18:37:32.000Z",
+ "nb_passages" : 21443
+ },
+ {
+ "entity_id" : 2,
+ "code_postal" : "56930",
+ "nom" : "2024 AMICALE 56 PLUMELIAU",
+ "ville" : "PLUMELIAU",
+ "nb_users" : 36,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-05-08T16:18:21.000Z",
+ "derniere_operation" : "2025-05-08T16:18:21.000Z",
+ "nb_passages" : 23846
+ },
+ {
+ "entity_id" : 1128,
+ "code_postal" : "56930",
+ "nom" : "PLUMELIAU ADRESSE ",
+ "ville" : "PLUMÉLIAU-BIEUZY",
+ "nb_users" : 42,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-10T02:00:17.000Z",
+ "derniere_operation" : "2024-10-10T02:00:17.000Z",
+ "nb_passages" : 4510
+ },
+ {
+ "entity_id" : 631,
+ "code_postal" : "56930",
+ "nom" : "Présentation Mayenne",
+ "ville" : "UDSP 53",
+ "nb_users" : 34,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-03-07T14:28:38.000Z",
+ "derniere_operation" : "2025-03-07T14:28:38.000Z",
+ "nb_passages" : 9540
+ },
+ {
+ "entity_id" : 770,
+ "code_postal" : "57260",
+ "nom" : "2024 AMICALE 57 DIEUZE",
+ "ville" : "DIEUZE",
+ "nb_users" : 27,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-13T20:20:27.000Z",
+ "derniere_operation" : "2024-11-13T20:20:27.000Z",
+ "nb_passages" : 14469
+ },
+ {
+ "entity_id" : 1196,
+ "code_postal" : "57500",
+ "nom" : "2025 TAST AMICALE 57 SAINT-AVOLD",
+ "ville" : "SAINT-AVOLD",
+ "nb_users" : 7,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-02-12T13:50:09.000Z",
+ "derniere_operation" : "2025-02-12T13:50:09.000Z",
+ "nb_passages" : 426
+ },
+ {
+ "entity_id" : 791,
+ "code_postal" : "57935",
+ "nom" : "2024 AMICALE 57 LUTTANGE",
+ "ville" : "LUTTANGE",
+ "nb_users" : 9,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-12-11T18:32:12.000Z",
+ "derniere_operation" : "2023-12-11T18:32:12.000Z",
+ "nb_passages" : 611
+ },
+ {
+ "entity_id" : 1098,
+ "code_postal" : "58140",
+ "nom" : "2024 TEST AMICALE 58 LORMES",
+ "ville" : "LORMES",
+ "nb_users" : 22,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-26T08:57:27.000Z",
+ "derniere_operation" : "2024-09-26T08:57:27.000Z",
+ "nb_passages" : 1331
+ },
+ {
+ "entity_id" : 1199,
+ "code_postal" : "59190",
+ "nom" : "2025 TEST AMICALE 59 HAZEBROUCK",
+ "ville" : "HAZEBROUCK",
+ "nb_users" : 51,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-10-06T07:20:52.000Z",
+ "derniere_operation" : "2025-10-06T07:20:52.000Z",
+ "nb_passages" : 24822
+ },
+ {
+ "entity_id" : 1190,
+ "code_postal" : "60150",
+ "nom" : "2025 AMICALE 60 THOUROTTE",
+ "ville" : "THOUROTTE",
+ "nb_users" : 79,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-01-29T21:38:01.000Z",
+ "derniere_operation" : "2025-01-29T21:38:01.000Z",
+ "nb_passages" : 10252
+ },
+ {
+ "entity_id" : 816,
+ "code_postal" : "60230",
+ "nom" : "2025 AMICALE 60 CHAMBLY ",
+ "ville" : "CHAMBLY",
+ "nb_users" : 109,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-10-03T15:30:36.000Z",
+ "derniere_operation" : "2025-10-03T15:30:36.000Z",
+ "nb_passages" : 31486
+ },
+ {
+ "entity_id" : 912,
+ "code_postal" : "60260",
+ "nom" : "PAS 2023 AMICALE 60 LAMORLAYE",
+ "ville" : "LAMORLAYE",
+ "nb_users" : 17,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-09-04T14:12:28.000Z",
+ "derniere_operation" : "2023-09-04T14:12:28.000Z",
+ "nb_passages" : 2
+ },
+ {
+ "entity_id" : 1135,
+ "code_postal" : "60530",
+ "nom" : "2024 TEST AMICALE 60 NEUILLY EN THELLE",
+ "ville" : "NEUILLY EN THELLE",
+ "nb_users" : 9,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-15T08:15:53.000Z",
+ "derniere_operation" : "2024-10-15T08:15:53.000Z",
+ "nb_passages" : 1847
+ },
+ {
+ "entity_id" : 1242,
+ "code_postal" : "61110",
+ "nom" : "2025 AMICALE 61 BRETONCELLES",
+ "ville" : "BRETONCELLES",
+ "nb_users" : 31,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-04-25T06:32:09.000Z",
+ "derniere_operation" : "2025-04-25T06:32:09.000Z",
+ "nb_passages" : 2949
+ },
+ {
+ "entity_id" : 1239,
+ "code_postal" : "61400",
+ "nom" : "2025 AMICALE 61 MORTAGNE AU PERCHE",
+ "ville" : "MORTAGNE AU PERCHE",
+ "nb_users" : 36,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-04-23T03:47:43.000Z",
+ "derniere_operation" : "2025-04-23T03:47:43.000Z",
+ "nb_passages" : 5803
+ },
+ {
+ "entity_id" : 1075,
+ "code_postal" : "61410",
+ "nom" : "2024 AMICALE 61 COUTERNE",
+ "ville" : "RIVES-D'ANDAINE",
+ "nb_users" : 48,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-10-06T16:19:36.000Z",
+ "derniere_operation" : "2025-10-06T16:19:36.000Z",
+ "nb_passages" : 13047
+ },
+ {
+ "entity_id" : 1240,
+ "code_postal" : "61500",
+ "nom" : "2025 TEST AMICALE 61 SEES ",
+ "ville" : "SEES",
+ "nb_users" : 27,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-04-25T05:43:48.000Z",
+ "derniere_operation" : "2025-04-25T05:43:48.000Z",
+ "nb_passages" : 824
+ },
+ {
+ "entity_id" : 1241,
+ "code_postal" : "61570",
+ "nom" : "2025 TEST AMICALE 61 MORTREE",
+ "ville" : "MORTREE",
+ "nb_users" : 11,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-04-25T06:04:03.000Z",
+ "derniere_operation" : "2025-04-25T06:04:03.000Z",
+ "nb_passages" : 333
+ },
+ {
+ "entity_id" : 895,
+ "code_postal" : "61600",
+ "nom" : "2024 AMICALE 61 LA FERTÉ MACé",
+ "ville" : "LA FERTÉ MACé",
+ "nb_users" : 12,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-19T11:50:00.000Z",
+ "derniere_operation" : "2024-10-19T11:50:00.000Z",
+ "nb_passages" : 2780
+ },
+ {
+ "entity_id" : 1206,
+ "code_postal" : "62150",
+ "nom" : "2025 TEST AMICALE 62 BRUAY-HOUDAIN",
+ "ville" : "HOUDAIN",
+ "nb_users" : 89,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-02-26T16:11:05.000Z",
+ "derniere_operation" : "2025-02-26T16:11:05.000Z",
+ "nb_passages" : 17277
+ },
+ {
+ "entity_id" : 1194,
+ "code_postal" : "63000",
+ "nom" : "2025 TEST AMICALE 63 CLERMONT-FERRAND CTACODI",
+ "ville" : "CLERMONT-FERRAND",
+ "nb_users" : 47,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-02-10T22:12:08.000Z",
+ "derniere_operation" : "2025-02-10T22:12:08.000Z",
+ "nb_passages" : 1690
+ },
+ {
+ "entity_id" : 1028,
+ "code_postal" : "63160",
+ "nom" : "2024 AMICALE 63 BILLOM",
+ "ville" : "BILLOM",
+ "nb_users" : 45,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-07-06T17:06:06.000Z",
+ "derniere_operation" : "2024-07-06T17:06:06.000Z",
+ "nb_passages" : 3181
+ },
+ {
+ "entity_id" : 970,
+ "code_postal" : "63190",
+ "nom" : "2023 AMICALE 63 RAVEL",
+ "ville" : "RAVEL",
+ "nb_users" : 12,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-11-05T18:35:43.000Z",
+ "derniere_operation" : "2023-11-05T18:35:43.000Z",
+ "nb_passages" : 728
+ },
+ {
+ "entity_id" : 975,
+ "code_postal" : "63270",
+ "nom" : "2023 TEST AMICALE 63 VIC LE COMTE",
+ "ville" : "VIC LE COMTE",
+ "nb_users" : 6,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-11-08T12:46:22.000Z",
+ "derniere_operation" : "2023-11-08T12:46:22.000Z",
+ "nb_passages" : 436
+ },
+ {
+ "entity_id" : 493,
+ "code_postal" : "63350",
+ "nom" : "2024 AMICALE 63 JOZE \/ CULHAT",
+ "ville" : "JOZE",
+ "nb_users" : 13,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-04-10T13:55:46.000Z",
+ "derniere_operation" : "2025-04-10T13:55:46.000Z",
+ "nb_passages" : 3507
+ },
+ {
+ "entity_id" : 940,
+ "code_postal" : "63360",
+ "nom" : "2024 AMICALE 63 SAINT BEAUZIRE",
+ "ville" : "SAINT BEAUZIRE",
+ "nb_users" : 42,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-10T21:01:03.000Z",
+ "derniere_operation" : "2024-10-10T21:01:03.000Z",
+ "nb_passages" : 2501
+ },
+ {
+ "entity_id" : 1022,
+ "code_postal" : "63600",
+ "nom" : "2024 TEST AMICALE 63 AMBERT",
+ "ville" : "AMBERT",
+ "nb_users" : 6,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-03-13T16:51:24.000Z",
+ "derniere_operation" : "2024-03-13T16:51:24.000Z",
+ "nb_passages" : 17
+ },
+ {
+ "entity_id" : 1107,
+ "code_postal" : "63700",
+ "nom" : "2024 AMICALE 63 MONTAIGUT EN COMBRAILLE",
+ "ville" : "MONTAIGUT EN COMBRAILLE",
+ "nb_users" : 30,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-29T12:27:36.000Z",
+ "derniere_operation" : "2024-09-29T12:27:36.000Z",
+ "nb_passages" : 2679
+ },
+ {
+ "entity_id" : 806,
+ "code_postal" : "63720",
+ "nom" : "2024 AMICALE 63 ENNEZAT",
+ "ville" : "ENNEZAT",
+ "nb_users" : 7,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-02-25T11:01:04.000Z",
+ "derniere_operation" : "2025-02-25T11:01:04.000Z",
+ "nb_passages" : 12252
+ },
+ {
+ "entity_id" : 983,
+ "code_postal" : "63720",
+ "nom" : "2024 AMICALE 63 SAINT IGNAT",
+ "ville" : "SAINT IGNAT",
+ "nb_users" : 11,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-22T11:04:09.000Z",
+ "derniere_operation" : "2024-09-22T11:04:09.000Z",
+ "nb_passages" : 1736
+ },
+ {
+ "entity_id" : 1143,
+ "code_postal" : "65390",
+ "nom" : "2024 AMICALE 65 ANDREST",
+ "ville" : "ANDREST",
+ "nb_users" : 25,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-21T20:57:58.000Z",
+ "derniere_operation" : "2024-10-21T20:57:58.000Z",
+ "nb_passages" : 2398
+ },
+ {
+ "entity_id" : 1001,
+ "code_postal" : "66000",
+ "nom" : "2024 TEST AMICALE 66 PERPIGNAN SUD",
+ "ville" : "PERPIGNAN",
+ "nb_users" : 165,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-01-16T16:54:13.000Z",
+ "derniere_operation" : "2024-01-16T16:54:13.000Z",
+ "nb_passages" : 29095
+ },
+ {
+ "entity_id" : 1193,
+ "code_postal" : "67230",
+ "nom" : "2025 AMICALE 67 BENFELD",
+ "ville" : "BENFELD",
+ "nb_users" : 17,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-02-07T16:41:50.000Z",
+ "derniere_operation" : "2025-02-07T16:41:50.000Z",
+ "nb_passages" : 2212
+ },
+ {
+ "entity_id" : 340,
+ "code_postal" : "68150",
+ "nom" : "2024 AMICALE 68 RIBEAUVILLE",
+ "ville" : "RIBEAUVILLE",
+ "nb_users" : 41,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-05T18:47:39.000Z",
+ "derniere_operation" : "2024-11-05T18:47:39.000Z",
+ "nb_passages" : 12682
+ },
+ {
+ "entity_id" : 663,
+ "code_postal" : "68160",
+ "nom" : "2023 AMICALE 68 STE MARIE AUX MINES ",
+ "ville" : "SAINTE MARIE AUX MINES",
+ "nb_users" : 9,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-26T13:21:54.000Z",
+ "derniere_operation" : "2024-09-26T13:21:54.000Z",
+ "nb_passages" : 557
+ },
+ {
+ "entity_id" : 397,
+ "code_postal" : "68240",
+ "nom" : "2023 AMICALE 68 KAYSERSBERG VIGNOBLE PAS 2024",
+ "ville" : "KAYSERSBERG VIGNOBLE",
+ "nb_users" : 48,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-10-18T14:39:40.000Z",
+ "derniere_operation" : "2023-10-18T14:39:40.000Z",
+ "nb_passages" : 7019
+ },
+ {
+ "entity_id" : 464,
+ "code_postal" : "68680",
+ "nom" : "2024 AMICALE 68 KEMBS",
+ "ville" : "KEMBS",
+ "nb_users" : 26,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-08T08:27:16.000Z",
+ "derniere_operation" : "2024-10-08T08:27:16.000Z",
+ "nb_passages" : 14956
+ },
+ {
+ "entity_id" : 962,
+ "code_postal" : "68770",
+ "nom" : "2024 AMICALE 68 AMMERSCHWIHR",
+ "ville" : "AMMERSCHWIHR",
+ "nb_users" : 21,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-14T19:46:09.000Z",
+ "derniere_operation" : "2024-11-14T19:46:09.000Z",
+ "nb_passages" : 1451
+ },
+ {
+ "entity_id" : 422,
+ "code_postal" : "69280",
+ "nom" : "2024 AMICALE 69 MARCY-CHARBONNIèRES",
+ "ville" : "MARCY-L'ÉTOILE",
+ "nb_users" : 23,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-29T08:04:39.000Z",
+ "derniere_operation" : "2025-09-29T08:04:39.000Z",
+ "nb_passages" : 15626
+ },
+ {
+ "entity_id" : 896,
+ "code_postal" : "69360",
+ "nom" : "2024 AMICALE 69 ST CASERNE DE L'OZON ",
+ "ville" : "CASERNE DE L'OZON ",
+ "nb_users" : 15,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-14T16:59:30.000Z",
+ "derniere_operation" : "2024-10-14T16:59:30.000Z",
+ "nb_passages" : 3863
+ },
+ {
+ "entity_id" : 1280,
+ "code_postal" : "71000",
+ "nom" : "2025 AMICALE 71 MACON",
+ "ville" : "SANCE",
+ "nb_users" : 49,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-03T13:21:51.000Z",
+ "derniere_operation" : "2025-09-03T13:21:51.000Z",
+ "nb_passages" : 9706
+ },
+ {
+ "entity_id" : 1104,
+ "code_postal" : "71150",
+ "nom" : "2024 AMICALE 71 CHAGNY",
+ "ville" : "CHAGNY",
+ "nb_users" : 35,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-27T20:16:24.000Z",
+ "derniere_operation" : "2024-09-27T20:16:24.000Z",
+ "nb_passages" : 6226
+ },
+ {
+ "entity_id" : 1122,
+ "code_postal" : "71160",
+ "nom" : "2025 AMICALE 71 DIGOIN",
+ "ville" : "DIGOIN",
+ "nb_users" : 65,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-06T14:13:46.000Z",
+ "derniere_operation" : "2024-10-06T14:13:46.000Z",
+ "nb_passages" : 3287
+ },
+ {
+ "entity_id" : 1115,
+ "code_postal" : "71170",
+ "nom" : "2024 AMICALE 71 CHAUFFAILLES",
+ "ville" : "CHAUFFAILLES",
+ "nb_users" : 30,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-01T14:08:56.000Z",
+ "derniere_operation" : "2025-09-01T14:08:56.000Z",
+ "nb_passages" : 6074
+ },
+ {
+ "entity_id" : 794,
+ "code_postal" : "71250",
+ "nom" : "2024 TEST AMICALE 71 SALORNAY SUR GUYE",
+ "ville" : "SALORNAY SUR GUYE",
+ "nb_users" : 15,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-05T11:09:59.000Z",
+ "derniere_operation" : "2024-10-05T11:09:59.000Z",
+ "nb_passages" : 3195
+ },
+ {
+ "entity_id" : 1124,
+ "code_postal" : "71460",
+ "nom" : "2024 AMICALE 71 JONCY",
+ "ville" : "JONCY",
+ "nb_users" : 10,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-07T20:00:31.000Z",
+ "derniere_operation" : "2024-10-07T20:00:31.000Z",
+ "nb_passages" : 1144
+ },
+ {
+ "entity_id" : 1270,
+ "code_postal" : "71480",
+ "nom" : "2025 AMICALE 71 CONDAL-DOMMARTIN-VARENNES ",
+ "ville" : "DOMMARTIN LES CUISEAUX",
+ "nb_users" : 24,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-07-17T07:56:59.000Z",
+ "derniere_operation" : "2025-07-17T07:56:59.000Z",
+ "nb_passages" : 1513
+ },
+ {
+ "entity_id" : 1114,
+ "code_postal" : "71500",
+ "nom" : "2025 AMICALE 71 LOUHANS",
+ "ville" : "BRANGES",
+ "nb_users" : 70,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-14T08:57:26.000Z",
+ "derniere_operation" : "2025-09-14T08:57:26.000Z",
+ "nb_passages" : 325
+ },
+ {
+ "entity_id" : 1291,
+ "code_postal" : "71500",
+ "nom" : "2025 AMICALE 71 LOUHANS ok",
+ "ville" : "BRANGES",
+ "nb_users" : 66,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-17T15:35:35.000Z",
+ "derniere_operation" : "2025-09-17T15:35:35.000Z",
+ "nb_passages" : 5905
+ },
+ {
+ "entity_id" : 1091,
+ "code_postal" : "71850",
+ "nom" : "2024 CONGRES NATIONAL MACON",
+ "ville" : "MACON",
+ "nb_users" : 129,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-24T11:43:38.000Z",
+ "derniere_operation" : "2024-09-24T11:43:38.000Z",
+ "nb_passages" : 5913
+ },
+ {
+ "entity_id" : 1052,
+ "code_postal" : "72220",
+ "nom" : "2024 AMICALE 72 ST MARS D'OUTILLÉ ",
+ "ville" : "ST MARS D'OUTILLÉ ",
+ "nb_users" : 10,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-23T17:20:25.000Z",
+ "derniere_operation" : "2025-09-23T17:20:25.000Z",
+ "nb_passages" : 1944
+ },
+ {
+ "entity_id" : 544,
+ "code_postal" : "72220",
+ "nom" : "2024 AMICALE 72 TELOCHÉ",
+ "ville" : "TELOCHÉ",
+ "nb_users" : 27,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-06-15T16:26:52.000Z",
+ "derniere_operation" : "2025-06-15T16:26:52.000Z",
+ "nb_passages" : 4563
+ },
+ {
+ "entity_id" : 1301,
+ "code_postal" : "72230",
+ "nom" : "AMICALE CONGRES LE MANS ",
+ "ville" : "MULSANNE",
+ "nb_users" : 5,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-10-07T17:02:57.000Z",
+ "derniere_operation" : "2025-10-07T17:02:57.000Z",
+ "nb_passages" : 56
+ },
+ {
+ "entity_id" : 409,
+ "code_postal" : "72250",
+ "nom" : "2024 AMICALE 72 PARIGNE L'EVEQUE ET BRETT",
+ "ville" : "PARIGNE L'EVEQUE",
+ "nb_users" : 31,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-17T07:48:52.000Z",
+ "derniere_operation" : "2025-09-17T07:48:52.000Z",
+ "nb_passages" : 21780
+ },
+ {
+ "entity_id" : 1132,
+ "code_postal" : "72250",
+ "nom" : "2025 AMICALE 72 CHALLES",
+ "ville" : "CHALLES",
+ "nb_users" : 7,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-14T15:35:03.000Z",
+ "derniere_operation" : "2024-10-14T15:35:03.000Z",
+ "nb_passages" : 105
+ },
+ {
+ "entity_id" : 1262,
+ "code_postal" : "73480",
+ "nom" : "2025 AMICALE 73 VAL-CENIS",
+ "ville" : "VAL-CENIS",
+ "nb_users" : 24,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-06-18T11:10:11.000Z",
+ "derniere_operation" : "2025-06-18T11:10:11.000Z",
+ "nb_passages" : 1047
+ },
+ {
+ "entity_id" : 908,
+ "code_postal" : "74140",
+ "nom" : "2023 AMICALE 74 DOUVAINE",
+ "ville" : "DOUVAINE",
+ "nb_users" : 19,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-08-28T13:02:12.000Z",
+ "derniere_operation" : "2023-08-28T13:02:12.000Z",
+ "nb_passages" : 8
+ },
+ {
+ "entity_id" : 976,
+ "code_postal" : "74200",
+ "nom" : "2024 AMICALE 74 REYVROZ ",
+ "ville" : "REYVROZ ",
+ "nb_users" : 14,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-24T07:48:24.000Z",
+ "derniere_operation" : "2024-11-24T07:48:24.000Z",
+ "nb_passages" : 977
+ },
+ {
+ "entity_id" : 982,
+ "code_postal" : "74360",
+ "nom" : "2023 AMICALE 74360 ABONDANCE",
+ "ville" : "ABONDANCE",
+ "nb_users" : 29,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-12-03T18:37:17.000Z",
+ "derniere_operation" : "2024-12-03T18:37:17.000Z",
+ "nb_passages" : 2994
+ },
+ {
+ "entity_id" : 793,
+ "code_postal" : "74420",
+ "nom" : "2024 AMICALE 74 BOëGE",
+ "ville" : "BOëGE",
+ "nb_users" : 23,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-31T13:21:49.000Z",
+ "derniere_operation" : "2024-10-31T13:21:49.000Z",
+ "nb_passages" : 10826
+ },
+ {
+ "entity_id" : 217,
+ "code_postal" : "74500",
+ "nom" : "2023 AMICALE 74 EVIAN LES BAINS",
+ "ville" : "EVIAN LES BAINS",
+ "nb_users" : 101,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-30T09:33:49.000Z",
+ "derniere_operation" : "2025-09-30T09:33:49.000Z",
+ "nb_passages" : 28624
+ },
+ {
+ "entity_id" : 1207,
+ "code_postal" : "74500",
+ "nom" : "2025 TEST amicale 74 Saint Paul Haut Gavot ",
+ "ville" : "SAINT PAUL HAUT GAVOT ",
+ "nb_users" : 5,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-02-26T16:27:21.000Z",
+ "derniere_operation" : "2025-02-26T16:27:21.000Z",
+ "nb_passages" : 220
+ },
+ {
+ "entity_id" : 322,
+ "code_postal" : "76210",
+ "nom" : "2024 TEST AMICALE 76 BOLBEC ",
+ "ville" : "BOLBEC",
+ "nb_users" : 63,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-08-19T15:58:22.000Z",
+ "derniere_operation" : "2025-08-19T15:58:22.000Z",
+ "nb_passages" : 5226
+ },
+ {
+ "entity_id" : 319,
+ "code_postal" : "76290",
+ "nom" : "pas 2024 AMICALE 76 MONTIVILLIERS PAS 2024 ",
+ "ville" : "MONTIVILLIERS",
+ "nb_users" : 49,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-09-27T11:30:34.000Z",
+ "derniere_operation" : "2023-09-27T11:30:34.000Z",
+ "nb_passages" : 37698
+ },
+ {
+ "entity_id" : 841,
+ "code_postal" : "76360",
+ "nom" : "2024 AMICALE 76 BARENTIN ",
+ "ville" : "BARENTIN",
+ "nb_users" : 65,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-25T14:44:35.000Z",
+ "derniere_operation" : "2024-10-25T14:44:35.000Z",
+ "nb_passages" : 14104
+ },
+ {
+ "entity_id" : 616,
+ "code_postal" : "76530",
+ "nom" : "2024 AMICALE 76 GRAND COURONNE ",
+ "ville" : "GRAND COURONNE ",
+ "nb_users" : 40,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-29T06:07:46.000Z",
+ "derniere_operation" : "2024-09-29T06:07:46.000Z",
+ "nb_passages" : 10021
+ },
+ {
+ "entity_id" : 1228,
+ "code_postal" : "76560",
+ "nom" : "2025 TEST AMICALE 76 DOUDEVILLE",
+ "ville" : "DOUDEVILLE",
+ "nb_users" : 5,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-03-26T21:34:58.000Z",
+ "derniere_operation" : "2025-03-26T21:34:58.000Z",
+ "nb_passages" : 1269
+ },
+ {
+ "entity_id" : 927,
+ "code_postal" : "76570",
+ "nom" : "2024 AMICALE 76 PAVILLY",
+ "ville" : "PAVILLY",
+ "nb_users" : 45,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-31T15:28:17.000Z",
+ "derniere_operation" : "2024-10-31T15:28:17.000Z",
+ "nb_passages" : 7022
+ },
+ {
+ "entity_id" : 1018,
+ "code_postal" : "76590",
+ "nom" : "2024 AMICALE 76 LONGUEVILLE SUR SCIE",
+ "ville" : "LONGUEVILLE SUR SCIE",
+ "nb_users" : 19,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-03-05T14:22:01.000Z",
+ "derniere_operation" : "2024-03-05T14:22:01.000Z",
+ "nb_passages" : 2553
+ },
+ {
+ "entity_id" : 115,
+ "code_postal" : "76630",
+ "nom" : "2023 AMICALE 76 BAILLY EN RIVIèRE",
+ "ville" : "BAILLY EN RIVIèRE",
+ "nb_users" : 14,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-03-06T14:23:19.000Z",
+ "derniere_operation" : "2025-03-06T14:23:19.000Z",
+ "nb_passages" : 2253
+ },
+ {
+ "entity_id" : 972,
+ "code_postal" : "76670",
+ "nom" : "2024 AMICALE 76 YERVILLE",
+ "ville" : "YERVILLE",
+ "nb_users" : 35,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-07-15T04:12:51.000Z",
+ "derniere_operation" : "2025-07-15T04:12:51.000Z",
+ "nb_passages" : 9792
+ },
+ {
+ "entity_id" : 901,
+ "code_postal" : "76840",
+ "nom" : "2024 AMICALE 76 SAINT MARTIN DE BOSCHERVILLE",
+ "ville" : "SAINT MARTIN DE BOSCHERVILLE",
+ "nb_users" : 68,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-03T09:35:45.000Z",
+ "derniere_operation" : "2024-11-03T09:35:45.000Z",
+ "nb_passages" : 3318
+ },
+ {
+ "entity_id" : 1024,
+ "code_postal" : "77330",
+ "nom" : "2024 AMICALE 77 OZOIR LA FERRIERE ",
+ "ville" : "OZOIR LA FERRIERE ",
+ "nb_users" : 28,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-03-22T14:54:53.000Z",
+ "derniere_operation" : "2024-03-22T14:54:53.000Z",
+ "nb_passages" : 7307
+ },
+ {
+ "entity_id" : 1116,
+ "code_postal" : "78110",
+ "nom" : "2024 TEST AMICALE 78 LE VESINET",
+ "ville" : "LE VESINET",
+ "nb_users" : 5,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-03T16:16:56.000Z",
+ "derniere_operation" : "2024-10-03T16:16:56.000Z",
+ "nb_passages" : 1590
+ },
+ {
+ "entity_id" : 795,
+ "code_postal" : "78140",
+ "nom" : "2024 AMICALE 78 VéLIZY-VILLACOUBLAY",
+ "ville" : "VéLIZY-VILLACOUBLAY",
+ "nb_users" : 75,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-29T10:29:32.000Z",
+ "derniere_operation" : "2024-09-29T10:29:32.000Z",
+ "nb_passages" : 16829
+ },
+ {
+ "entity_id" : 1292,
+ "code_postal" : "78180",
+ "nom" : "2025 TEST AMICALE 78 MONTIGNY LE BRETONNEUX",
+ "ville" : "MONTIGNY LE BRETONNEUX",
+ "nb_users" : 7,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-17T16:25:11.000Z",
+ "derniere_operation" : "2025-09-17T16:25:11.000Z",
+ "nb_passages" : 9230
+ },
+ {
+ "entity_id" : 569,
+ "code_postal" : "78200",
+ "nom" : "2024 AMICALE 78 MAGNANVILLE",
+ "ville" : "MAGNANVILLE",
+ "nb_users" : 63,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-08-22T06:44:32.000Z",
+ "derniere_operation" : "2024-08-22T06:44:32.000Z",
+ "nb_passages" : 104826
+ },
+ {
+ "entity_id" : 7,
+ "code_postal" : "78300",
+ "nom" : "2024 AMICALE 78 POISSY",
+ "ville" : "POISSY",
+ "nb_users" : 49,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-10-05T16:57:22.000Z",
+ "derniere_operation" : "2025-10-05T16:57:22.000Z",
+ "nb_passages" : 140861
+ },
+ {
+ "entity_id" : 1245,
+ "code_postal" : "78390",
+ "nom" : "AMICALE TEST 78 ",
+ "ville" : "BOIS D'ARCY",
+ "nb_users" : 10,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-05-14T08:35:51.000Z",
+ "derniere_operation" : "2025-05-14T08:35:51.000Z",
+ "nb_passages" : 948
+ },
+ {
+ "entity_id" : 408,
+ "code_postal" : "78570",
+ "nom" : "2024 AMICALE 78 CHANTELOUP LES VIGNES",
+ "ville" : "CHANTELOUP LES VIGNES",
+ "nb_users" : 72,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-18T12:21:40.000Z",
+ "derniere_operation" : "2024-10-18T12:21:40.000Z",
+ "nb_passages" : 32915
+ },
+ {
+ "entity_id" : 1290,
+ "code_postal" : "79120",
+ "nom" : "2025 AMICALE 79 LEZAY ",
+ "ville" : "LEZAY",
+ "nb_users" : 24,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-17T07:42:16.000Z",
+ "derniere_operation" : "2025-09-17T07:42:16.000Z",
+ "nb_passages" : 3552
+ },
+ {
+ "entity_id" : 1090,
+ "code_postal" : "79160",
+ "nom" : "2024 AMICALE 79 COULONGES SUR L'AUTIZE",
+ "ville" : "COULONGES SUR L'AUTIZE",
+ "nb_users" : 35,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-19T20:38:59.000Z",
+ "derniere_operation" : "2024-09-19T20:38:59.000Z",
+ "nb_passages" : 3363
+ },
+ {
+ "entity_id" : 197,
+ "code_postal" : "79170",
+ "nom" : "2024 AMICALE 79 BRIOUX SUR BOUTONNE",
+ "ville" : "BRIOUX SUR BOUTONNE",
+ "nb_users" : 26,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-10-04T12:02:52.000Z",
+ "derniere_operation" : "2025-10-04T12:02:52.000Z",
+ "nb_passages" : 7832
+ },
+ {
+ "entity_id" : 876,
+ "code_postal" : "79250",
+ "nom" : "2024 AMICALE 79 NUEIL LES AUBIERS",
+ "ville" : "NUEIL LES AUBIERS",
+ "nb_users" : 22,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-10-05T16:13:57.000Z",
+ "derniere_operation" : "2025-10-05T16:13:57.000Z",
+ "nb_passages" : 7466
+ },
+ {
+ "entity_id" : 1276,
+ "code_postal" : "79300",
+ "nom" : "2025 AMICALE 79 BRESSUIRE",
+ "ville" : "BRESSUIRE",
+ "nb_users" : 45,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-08-30T17:56:22.000Z",
+ "derniere_operation" : "2025-08-30T17:56:22.000Z",
+ "nb_passages" : 14629
+ },
+ {
+ "entity_id" : 944,
+ "code_postal" : "79310",
+ "nom" : "2023 TEST AMICALE MAZIERES ST PARDOUX",
+ "ville" : "Mazières en Gâtine",
+ "nb_users" : 8,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-10-07T12:13:52.000Z",
+ "derniere_operation" : "2023-10-07T12:13:52.000Z",
+ "nb_passages" : 482
+ },
+ {
+ "entity_id" : 1045,
+ "code_postal" : "79320",
+ "nom" : "2024 AMICALE 79 MONCOUTANT SUR SEVRE",
+ "ville" : "MONCOUTANT SUR SEVRE",
+ "nb_users" : 35,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-05-29T16:25:38.000Z",
+ "derniere_operation" : "2024-05-29T16:25:38.000Z",
+ "nb_passages" : 4064
+ },
+ {
+ "entity_id" : 878,
+ "code_postal" : "79330",
+ "nom" : "2024 AMICALE 79 SAINT VARENT",
+ "ville" : "SAINT VARENT",
+ "nb_users" : 44,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-10-07T17:16:20.000Z",
+ "derniere_operation" : "2025-10-07T17:16:20.000Z",
+ "nb_passages" : 21248
+ },
+ {
+ "entity_id" : 1034,
+ "code_postal" : "79410",
+ "nom" : "2024 AMICALE 79 CHERVEUX",
+ "ville" : "CHERVEUX",
+ "nb_users" : 7,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-10-01T16:44:50.000Z",
+ "derniere_operation" : "2025-10-01T16:44:50.000Z",
+ "nb_passages" : 2302
+ },
+ {
+ "entity_id" : 823,
+ "code_postal" : "79600",
+ "nom" : "2023 AMICALE 79 LE THOUET",
+ "ville" : "AIRVAULT",
+ "nb_users" : 30,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-23T12:04:47.000Z",
+ "derniere_operation" : "2024-09-23T12:04:47.000Z",
+ "nb_passages" : 4135
+ },
+ {
+ "entity_id" : 877,
+ "code_postal" : "79600",
+ "nom" : "2025 AMICALE 79 ASSAIS LES JUMEAUX ",
+ "ville" : "ASSAIS LES JUMEAUX ",
+ "nb_users" : 13,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-03-04T18:11:12.000Z",
+ "derniere_operation" : "2024-03-04T18:11:12.000Z",
+ "nb_passages" : 877
+ },
+ {
+ "entity_id" : 1026,
+ "code_postal" : "79800",
+ "nom" : "2024 AMICALE 79 LA MOTHE SAINT-HERAY",
+ "ville" : "LA MOTHE SAINT-HERAY",
+ "nb_users" : 25,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-12T05:14:00.000Z",
+ "derniere_operation" : "2025-09-12T05:14:00.000Z",
+ "nb_passages" : 19194
+ },
+ {
+ "entity_id" : 966,
+ "code_postal" : "82120",
+ "nom" : "2023 AMICALE 82 LAVIT",
+ "ville" : "LAVIT",
+ "nb_users" : 8,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-10-24T20:26:46.000Z",
+ "derniere_operation" : "2023-10-24T20:26:46.000Z",
+ "nb_passages" : 666
+ },
+ {
+ "entity_id" : 312,
+ "code_postal" : "83170",
+ "nom" : "2024 AMICALE 83 TOURVES",
+ "ville" : "TOURVES",
+ "nb_users" : 20,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-09-02T18:07:23.000Z",
+ "derniere_operation" : "2024-09-02T18:07:23.000Z",
+ "nb_passages" : 782
+ },
+ {
+ "entity_id" : 1266,
+ "code_postal" : "83190",
+ "nom" : "2025 TEST AMICALE 83 OLLIOULES",
+ "ville" : "OLLIOULES",
+ "nb_users" : 8,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-08-16T13:07:33.000Z",
+ "derniere_operation" : "2025-08-16T13:07:33.000Z",
+ "nb_passages" : 4902
+ },
+ {
+ "entity_id" : 1179,
+ "code_postal" : "83240",
+ "nom" : "2025 TEST AMICALE 83 CAVALAIRE SUR MER",
+ "ville" : "CAVALAIRE SUR MER ",
+ "nb_users" : 46,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-01-13T18:00:30.000Z",
+ "derniere_operation" : "2025-01-13T18:00:30.000Z",
+ "nb_passages" : 8239
+ },
+ {
+ "entity_id" : 712,
+ "code_postal" : "83560",
+ "nom" : "2023 AMICALE 83 GINASSERVIS",
+ "ville" : "GINASSERVIS",
+ "nb_users" : 31,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-11-11T16:00:36.000Z",
+ "derniere_operation" : "2023-11-11T16:00:36.000Z",
+ "nb_passages" : 2739
+ },
+ {
+ "entity_id" : 320,
+ "code_postal" : "83560",
+ "nom" : "2024 AMICALE 83 VINON SUR VERDON",
+ "ville" : "VINON SUR VERDON",
+ "nb_users" : 30,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-15T12:59:03.000Z",
+ "derniere_operation" : "2024-10-15T12:59:03.000Z",
+ "nb_passages" : 19295
+ },
+ {
+ "entity_id" : 709,
+ "code_postal" : "83690",
+ "nom" : "2023 AMICALE 83 SALERNES",
+ "ville" : "SALERNES",
+ "nb_users" : 16,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-08-21T19:07:11.000Z",
+ "derniere_operation" : "2023-08-21T19:07:11.000Z",
+ "nb_passages" : 3753
+ },
+ {
+ "entity_id" : 570,
+ "code_postal" : "83790",
+ "nom" : "2024 AMICALE 83 PIGNANS",
+ "ville" : "PIGNANS",
+ "nb_users" : 26,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-07-25T05:32:00.000Z",
+ "derniere_operation" : "2025-07-25T05:32:00.000Z",
+ "nb_passages" : 56214
+ },
+ {
+ "entity_id" : 1197,
+ "code_postal" : "84000",
+ "nom" : "2025 AMICALE 84 AVIGNON",
+ "ville" : "AVIGNON",
+ "nb_users" : 96,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-02-12T20:56:44.000Z",
+ "derniere_operation" : "2025-02-12T20:56:44.000Z",
+ "nb_passages" : 721
+ },
+ {
+ "entity_id" : 922,
+ "code_postal" : "84220",
+ "nom" : "2024 AMICALE 84 GORDES",
+ "ville" : "GORDES",
+ "nb_users" : 18,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-12-13T08:49:12.000Z",
+ "derniere_operation" : "2024-12-13T08:49:12.000Z",
+ "nb_passages" : 8925
+ },
+ {
+ "entity_id" : 755,
+ "code_postal" : "84420",
+ "nom" : "2025 AMICALE 84 VALLEE DU RHONE ",
+ "ville" : "VALLEE DU RHONE",
+ "nb_users" : 64,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-10T16:21:06.000Z",
+ "derniere_operation" : "2025-09-10T16:21:06.000Z",
+ "nb_passages" : 22610
+ },
+ {
+ "entity_id" : 963,
+ "code_postal" : "85100",
+ "nom" : "2023 AMICALE 85 LES SABLES D'OLONNE PAS 2024 ",
+ "ville" : "LES SABLES D'OLONNE",
+ "nb_users" : 6,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-10-21T15:45:32.000Z",
+ "derniere_operation" : "2023-10-21T15:45:32.000Z",
+ "nb_passages" : 628
+ },
+ {
+ "entity_id" : 1048,
+ "code_postal" : "85170",
+ "nom" : "LES LUCS SUR BOULOGNE",
+ "ville" : "LES LUCS SUR BOULOGNE",
+ "nb_users" : 7,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-06-07T20:15:25.000Z",
+ "derniere_operation" : "2024-06-07T20:15:25.000Z",
+ "nb_passages" : 27
+ },
+ {
+ "entity_id" : 334,
+ "code_postal" : "85200",
+ "nom" : "2024 AMICALE 85 FONTENAY LE COMTE",
+ "ville" : "FONTENAY LE COMTE",
+ "nb_users" : 58,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-13T20:57:29.000Z",
+ "derniere_operation" : "2024-11-13T20:57:29.000Z",
+ "nb_passages" : 103089
+ },
+ {
+ "entity_id" : 1032,
+ "code_postal" : "85260",
+ "nom" : "2024 AMICALE 85 LES BROUZILS",
+ "ville" : "LES BROUZILS ",
+ "nb_users" : 17,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-08-25T18:17:24.000Z",
+ "derniere_operation" : "2025-08-25T18:17:24.000Z",
+ "nb_passages" : 3316
+ },
+ {
+ "entity_id" : 801,
+ "code_postal" : "85310",
+ "nom" : "2024 AMICALE 85 LA CHAIZE LE VICOMTE",
+ "ville" : "LA CHAIZE LE VICOMTE",
+ "nb_users" : 30,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-13T13:23:31.000Z",
+ "derniere_operation" : "2025-09-13T13:23:31.000Z",
+ "nb_passages" : 6366
+ },
+ {
+ "entity_id" : 1218,
+ "code_postal" : "85330",
+ "nom" : "2025 TEST AMICALE 85 NOIRMOUTIER",
+ "ville" : "NOIRMOUTIER",
+ "nb_users" : 25,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-10-06T16:54:01.000Z",
+ "derniere_operation" : "2025-10-06T16:54:01.000Z",
+ "nb_passages" : 24474
+ },
+ {
+ "entity_id" : 636,
+ "code_postal" : "85360",
+ "nom" : "2024 AMICALE 85 LA TRANCHE SUR ",
+ "ville" : "LA TRANCHE SUR MER",
+ "nb_users" : 29,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-17T17:52:34.000Z",
+ "derniere_operation" : "2024-10-17T17:52:34.000Z",
+ "nb_passages" : 4827
+ },
+ {
+ "entity_id" : 1063,
+ "code_postal" : "85420",
+ "nom" : "2024 AMICALE 85 MAILLEZAIS",
+ "ville" : "MAILLEZAIS",
+ "nb_users" : 27,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-07-16T07:40:20.000Z",
+ "derniere_operation" : "2024-07-16T07:40:20.000Z",
+ "nb_passages" : 2734
+ },
+ {
+ "entity_id" : 784,
+ "code_postal" : "85440",
+ "nom" : "2024 AMICALE 85 TALMONT SAINT HILAIRE",
+ "ville" : "TALMONT ST HILAIRE",
+ "nb_users" : 34,
+ "nb_operations" : 2,
+ "premiere_operation" : "2024-09-05T14:28:34.000Z",
+ "derniere_operation" : "2025-09-30T19:44:29.000Z",
+ "nb_passages" : 26046
+ },
+ {
+ "entity_id" : 1064,
+ "code_postal" : "85490",
+ "nom" : "2024 AMICALE 85 BENET ",
+ "ville" : "BENET46.39959",
+ "nb_users" : 25,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-10T17:36:49.000Z",
+ "derniere_operation" : "2025-09-10T17:36:49.000Z",
+ "nb_passages" : 4333
+ },
+ {
+ "entity_id" : 1200,
+ "code_postal" : "85520",
+ "nom" : "2025 AMICALE 85 JARD SUR MER",
+ "ville" : "JARD SUR MER",
+ "nb_users" : 27,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-05-09T16:01:58.000Z",
+ "derniere_operation" : "2025-05-09T16:01:58.000Z",
+ "nb_passages" : 11834
+ },
+ {
+ "entity_id" : 1233,
+ "code_postal" : "85600",
+ "nom" : "2025 AMICALE 85 MONTAIGU",
+ "ville" : "MONTAIGU",
+ "nb_users" : 49,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-30T09:18:28.000Z",
+ "derniere_operation" : "2025-09-30T09:18:28.000Z",
+ "nb_passages" : 50514
+ },
+ {
+ "entity_id" : 1256,
+ "code_postal" : "85620",
+ "nom" : "2025 TEST AMICALE 85 ROCHESERVIERE",
+ "ville" : "ROCHESERVIERE",
+ "nb_users" : 28,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-06-05T20:43:11.000Z",
+ "derniere_operation" : "2025-06-05T20:43:11.000Z",
+ "nb_passages" : 1911
+ },
+ {
+ "entity_id" : 589,
+ "code_postal" : "85640",
+ "nom" : "2024 AMICALE 85 MOUCHAMPS",
+ "ville" : "MOUCHAMPS",
+ "nb_users" : 30,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-08T16:25:05.000Z",
+ "derniere_operation" : "2024-10-08T16:25:05.000Z",
+ "nb_passages" : 12015
+ },
+ {
+ "entity_id" : 1214,
+ "code_postal" : "86115",
+ "nom" : "TEST SECTEUR VIENNE 86",
+ "ville" : "JAUNAY-MARIGNY",
+ "nb_users" : 8,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-03-16T11:15:26.000Z",
+ "derniere_operation" : "2025-03-16T11:15:26.000Z",
+ "nb_passages" : 851
+ },
+ {
+ "entity_id" : 947,
+ "code_postal" : "86170",
+ "nom" : "2024 AMICALE 86 NEUVILLE",
+ "ville" : "NEUVILLE",
+ "nb_users" : 43,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-02T09:49:18.000Z",
+ "derniere_operation" : "2024-10-02T09:49:18.000Z",
+ "nb_passages" : 9567
+ },
+ {
+ "entity_id" : 826,
+ "code_postal" : "86190",
+ "nom" : "2024 AMICALE 86 LATILLE",
+ "ville" : "LATILLLE",
+ "nb_users" : 24,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-03-17T13:29:34.000Z",
+ "derniere_operation" : "2025-03-17T13:29:34.000Z",
+ "nb_passages" : 8484
+ },
+ {
+ "entity_id" : 1213,
+ "code_postal" : "86190",
+ "nom" : "2025 AMICALE 86 VOUILLE ",
+ "ville" : "VOUILLE",
+ "nb_users" : 25,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-03-12T07:25:04.000Z",
+ "derniere_operation" : "2025-03-12T07:25:04.000Z",
+ "nb_passages" : 3891
+ },
+ {
+ "entity_id" : 661,
+ "code_postal" : "86200",
+ "nom" : "2024 AMICALE 86 LOUDUN",
+ "ville" : "LOUDUN",
+ "nb_users" : 45,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-26T05:58:08.000Z",
+ "derniere_operation" : "2025-09-26T05:58:08.000Z",
+ "nb_passages" : 24004
+ },
+ {
+ "entity_id" : 1148,
+ "code_postal" : "86260",
+ "nom" : "2024 AMICALE 86 LA ROCHE POSAY",
+ "ville" : "LA ROCHE POSAY",
+ "nb_users" : 10,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-25T07:13:33.000Z",
+ "derniere_operation" : "2024-10-25T07:13:33.000Z",
+ "nb_passages" : 1193
+ },
+ {
+ "entity_id" : 836,
+ "code_postal" : "86300",
+ "nom" : "2024 AMICALE 86 CHAUVIGNY",
+ "ville" : "CHAUVIGNY",
+ "nb_users" : 47,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-25T15:34:16.000Z",
+ "derniere_operation" : "2025-09-25T15:34:16.000Z",
+ "nb_passages" : 24793
+ },
+ {
+ "entity_id" : 324,
+ "code_postal" : "88120",
+ "nom" : "2024 AMICALE 88 LE SYNDICAT ",
+ "ville" : "LE SYNDICAT",
+ "nb_users" : 46,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-25T10:16:01.000Z",
+ "derniere_operation" : "2024-10-25T10:16:01.000Z",
+ "nb_passages" : 33704
+ },
+ {
+ "entity_id" : 352,
+ "code_postal" : "88360",
+ "nom" : "2024 AMICALE 88 RUPT SUR MOSELLE",
+ "ville" : "RUPT SUR MOSELLE",
+ "nb_users" : 16,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-25T16:45:40.000Z",
+ "derniere_operation" : "2024-10-25T16:45:40.000Z",
+ "nb_passages" : 1877
+ },
+ {
+ "entity_id" : 1023,
+ "code_postal" : "88500",
+ "nom" : "2024 AMICALE 88 MIRECOURT",
+ "ville" : "MIRECOURT",
+ "nb_users" : 18,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-24T15:24:41.000Z",
+ "derniere_operation" : "2025-09-24T15:24:41.000Z",
+ "nb_passages" : 6024
+ },
+ {
+ "entity_id" : 1145,
+ "code_postal" : "88500",
+ "nom" : "2024 TEST AMICALE 88 AMBACOURT ",
+ "ville" : "AMBACOURT",
+ "nb_users" : 10,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-22T12:54:24.000Z",
+ "derniere_operation" : "2024-10-22T12:54:24.000Z",
+ "nb_passages" : 1115
+ },
+ {
+ "entity_id" : 846,
+ "code_postal" : "88700",
+ "nom" : "2024 AMICALE 88 RAMBERVILLERS",
+ "ville" : "RAMBERVILLERS",
+ "nb_users" : 34,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-14T13:13:34.000Z",
+ "derniere_operation" : "2024-10-14T13:13:34.000Z",
+ "nb_passages" : 4520
+ },
+ {
+ "entity_id" : 1166,
+ "code_postal" : "88800",
+ "nom" : "2024 TEST AMICALE 88 VITTEL",
+ "ville" : "VITTEL",
+ "nb_users" : 14,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-19T16:31:46.000Z",
+ "derniere_operation" : "2024-11-19T16:31:46.000Z",
+ "nb_passages" : 2397
+ },
+ {
+ "entity_id" : 960,
+ "code_postal" : "89340",
+ "nom" : "2024 AMICALE 89 CHAMPIGNY",
+ "ville" : "CHAMPIGNY",
+ "nb_users" : 13,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-10-27T15:54:51.000Z",
+ "derniere_operation" : "2023-10-27T15:54:51.000Z",
+ "nb_passages" : 0
+ },
+ {
+ "entity_id" : 1126,
+ "code_postal" : "89800",
+ "nom" : "2024 TEST AMICALE 89 CHABLIS",
+ "ville" : "CHABLIS",
+ "nb_users" : 15,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-08T14:23:55.000Z",
+ "derniere_operation" : "2024-10-08T14:23:55.000Z",
+ "nb_passages" : 2858
+ },
+ {
+ "entity_id" : 602,
+ "code_postal" : "90500",
+ "nom" : "2024 AMICALE 90 BEAUCOURT",
+ "ville" : "BEAUCOURT",
+ "nb_users" : 22,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-31T12:39:56.000Z",
+ "derniere_operation" : "2024-10-31T12:39:56.000Z",
+ "nb_passages" : 11059
+ },
+ {
+ "entity_id" : 913,
+ "code_postal" : "91100",
+ "nom" : "2024 AMICALE 91 CORBEIL ESSONNES",
+ "ville" : "CORBEIL ESSONNES",
+ "nb_users" : 124,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-07-02T17:42:25.000Z",
+ "derniere_operation" : "2025-07-02T17:42:25.000Z",
+ "nb_passages" : 56706
+ },
+ {
+ "entity_id" : 772,
+ "code_postal" : "91120",
+ "nom" : "2025 AMICALE 91 PALAISEAU",
+ "ville" : "PALAISEAU",
+ "nb_users" : 79,
+ "nb_operations" : 2,
+ "premiere_operation" : "2024-10-08T08:58:09.000Z",
+ "derniere_operation" : "2025-10-06T17:53:23.000Z",
+ "nb_passages" : 276262
+ },
+ {
+ "entity_id" : 555,
+ "code_postal" : "91150",
+ "nom" : "2024 AMICALE 91 ETAMPES",
+ "ville" : "ETAMPES",
+ "nb_users" : 43,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-09-25T19:04:14.000Z",
+ "derniere_operation" : "2025-09-25T19:04:14.000Z",
+ "nb_passages" : 38202
+ },
+ {
+ "entity_id" : 1157,
+ "code_postal" : "91360",
+ "nom" : "2024 TEST AMICALE 91 EPINAY SUR ORGE OK",
+ "ville" : "EPINAY SUR ORGE",
+ "nb_users" : 14,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-06T19:45:27.000Z",
+ "derniere_operation" : "2024-11-06T19:45:27.000Z",
+ "nb_passages" : 3481
+ },
+ {
+ "entity_id" : 1118,
+ "code_postal" : "91460",
+ "nom" : "2024 TEST AMICALE 91 MARCOUSSIS",
+ "ville" : "MARCOUSSIS",
+ "nb_users" : 24,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-03T20:01:52.000Z",
+ "derniere_operation" : "2024-10-03T20:01:52.000Z",
+ "nb_passages" : 0
+ },
+ {
+ "entity_id" : 690,
+ "code_postal" : "91590",
+ "nom" : "2023 AMICALE 91 CERNY\/LA FERTE ALAIS PAS 2024",
+ "ville" : "CERNY",
+ "nb_users" : 88,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-10-05T11:44:30.000Z",
+ "derniere_operation" : "2023-10-05T11:44:30.000Z",
+ "nb_passages" : 7970
+ },
+ {
+ "entity_id" : 931,
+ "code_postal" : "91600",
+ "nom" : "2024 AMICALE 91 SAVIGNY\/ORGE ",
+ "ville" : "SAVIGNY\/ORGE",
+ "nb_users" : 47,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-05-11T14:27:35.000Z",
+ "derniere_operation" : "2025-05-11T14:27:35.000Z",
+ "nb_passages" : 62636
+ },
+ {
+ "entity_id" : 694,
+ "code_postal" : "91630",
+ "nom" : "2023 AMICALE 91 MAROLLES EN HUREPOIX",
+ "ville" : "MAROLLES EN HUREPOIX",
+ "nb_users" : 8,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-11-05T21:46:59.000Z",
+ "derniere_operation" : "2024-11-05T21:46:59.000Z",
+ "nb_passages" : 12277
+ },
+ {
+ "entity_id" : 1070,
+ "code_postal" : "91650",
+ "nom" : "2024 AMICALE 91 BREUILLET ESSONNE",
+ "ville" : "BREUILLET ESSONNE",
+ "nb_users" : 22,
+ "nb_operations" : 1,
+ "premiere_operation" : "2025-10-01T16:08:01.000Z",
+ "derniere_operation" : "2025-10-01T16:08:01.000Z",
+ "nb_passages" : 45577
+ },
+ {
+ "entity_id" : 1141,
+ "code_postal" : "91660",
+ "nom" : "2024 AMICALE 91 MEREVILLE",
+ "ville" : "LE MÉRÉVILLOIS",
+ "nb_users" : 11,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-20T08:56:55.000Z",
+ "derniere_operation" : "2024-10-20T08:56:55.000Z",
+ "nb_passages" : 2240
+ },
+ {
+ "entity_id" : 865,
+ "code_postal" : "95450",
+ "nom" : "2024 AMICALE 95 VIGNY",
+ "ville" : "VIGNY",
+ "nb_users" : 21,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-10-28T16:45:08.000Z",
+ "derniere_operation" : "2024-10-28T16:45:08.000Z",
+ "nb_passages" : 8916
+ },
+ {
+ "entity_id" : 695,
+ "code_postal" : "95500",
+ "nom" : "2022 AMICALE 95 GONESSE",
+ "ville" : "GONESSE",
+ "nb_users" : 66,
+ "nb_operations" : 1,
+ "premiere_operation" : "2023-10-21T07:28:38.000Z",
+ "derniere_operation" : "2023-10-21T07:28:38.000Z",
+ "nb_passages" : 2350
+ },
+ {
+ "entity_id" : 393,
+ "code_postal" : "95660",
+ "nom" : "2024 AMICALE 95 CHAMPAGNE SUR OISE",
+ "ville" : "CHAMPAGNE SUR OISE",
+ "nb_users" : 41,
+ "nb_operations" : 1,
+ "premiere_operation" : "2024-12-06T14:31:12.000Z",
+ "derniere_operation" : "2024-12-06T14:31:12.000Z",
+ "nb_passages" : 5034
+ }
+]}
diff --git a/api/scripts/orga/TODO-ISOLATION-OPERATIONS.md b/api/scripts/orga/TODO-ISOLATION-OPERATIONS.md
new file mode 100644
index 00000000..f609e5d9
--- /dev/null
+++ b/api/scripts/orga/TODO-ISOLATION-OPERATIONS.md
@@ -0,0 +1,473 @@
+# TODO - Isolation complète des opérations
+
+## 🎯 Objectif
+
+Mettre en place une **isolation complète par opération** où chaque opération est totalement autonome et peut être supprimée indépendamment sans impacter les autres opérations ou la table centrale `users`.
+
+## 📊 Architecture cible
+
+```
+operations (id: 850)
+ ├── ope_users (id: 2500, fk_operation: 850, fk_user: 100)
+ │ ├── ope_users_sectors (fk_user: 2500 ← ope_users.id, fk_sector: 5400)
+ │ └── ope_pass (fk_user: 2500 ← ope_users.id, fk_sector: 5400)
+ └── ope_sectors (id: 5400, fk_operation: 850)
+
+users (id: 100) ← table centrale (conservée même si opération supprimée)
+```
+
+---
+
+## ✅ Tâche 1 : Modification du schéma SQL
+
+### 📁 Fichier : `scripts/orga/fix_fk_constraints.sql`
+
+### Actions
+
+- [ ] **1.1** Tester le script SQL sur **dva_geo** (DEV)
+ ```bash
+ incus exec dva-geo -- mysql rca_geo < /var/www/geosector/api/scripts/orga/fix_fk_constraints.sql
+ ```
+
+- [ ] **1.2** Vérifier les contraintes après exécution :
+ ```sql
+ SELECT TABLE_NAME, COLUMN_NAME, REFERENCED_TABLE_NAME, REFERENCED_COLUMN_NAME
+ FROM information_schema.KEY_COLUMN_USAGE
+ WHERE TABLE_SCHEMA = 'rca_geo'
+ AND TABLE_NAME IN ('ope_users_sectors', 'ope_pass')
+ AND COLUMN_NAME = 'fk_user';
+ ```
+ Résultat attendu :
+ - `ope_users_sectors.fk_user → ope_users.id`
+ - `ope_pass.fk_user → ope_users.id`
+
+- [ ] **1.3** Appliquer sur **rca_geo** (RECETTE) après validation sur dva_geo
+
+- [ ] **1.4** Appliquer sur **pra_geo** (PRODUCTION) après validation sur rca_geo
+
+### ⚠️ Important
+
+- Les données existantes doivent être **nettoyées avant** d'appliquer le script
+- Ou bien : recréer toutes les données avec la nouvelle migration
+- Les FK `ON DELETE CASCADE` supprimeront automatiquement `ope_users_sectors` et `ope_pass` quand `ope_users` est supprimé
+
+---
+
+## ✅ Tâche 2 : Correction du script de migration2
+
+### 📁 Fichiers concernés
+
+1. `scripts/migration2/php/lib/SectorMigrator.php`
+2. `scripts/migration2/php/lib/PassageMigrator.php`
+
+### Actions
+
+#### 2.1 SectorMigrator.php - Migration de ope_users_sectors
+
+- [ ] **Ligne 253** : Changer de `users.id` vers `ope_users.id`
+
+```php
+// ❌ AVANT
+':fk_user' => $us['fk_user'], // ID de users (table centrale)
+
+// ✅ APRÈS
+':fk_user' => $userMapping[$us['fk_user']], // ID de ope_users (mapping)
+```
+
+#### 2.2 PassageMigrator.php - Migration de ope_pass
+
+- [ ] **Ligne 64-67** : Vérifier le mapping existe
+- [ ] **Ligne 77** : Passer `ope_users.id` au lieu de `users.id`
+
+```php
+// ❌ AVANT (ligne 77)
+$newPassId = $this->insertPassage($passage, $newOperationId, $newOpeSectorId, $passage['fk_user']);
+
+// ✅ APRÈS
+$newOpeUserId = $userMapping[$passage['fk_user']];
+$newPassId = $this->insertPassage($passage, $newOperationId, $newOpeSectorId, $newOpeUserId);
+```
+
+- [ ] **Ligne 164** : Utiliser le paramètre `$userId` qui sera maintenant `ope_users.id`
+
+```php
+// ❌ AVANT
+':fk_user' => $userId, // ID de users (table centrale)
+
+// ✅ APRÈS (le paramètre $userId contiendra déjà ope_users.id)
+':fk_user' => $userId, // ID de ope_users
+```
+
+- [ ] **Ligne 71** : Corriger `verifyUserSectorAssociation` pour vérifier avec `ope_users.id`
+
+```php
+// ❌ AVANT
+if (!$this->verifyUserSectorAssociation($newOperationId, $passage['fk_user'], $newOpeSectorId)) {
+
+// ✅ APRÈS
+if (!$this->verifyUserSectorAssociation($newOperationId, $newOpeUserId, $newOpeSectorId)) {
+```
+
+#### 2.3 Tester la migration complète
+
+- [ ] **Sur dva_geo** : Vider les données d'une entité et relancer la migration
+ ```bash
+ php php/migrate_from_backup.php --mode=entity --entity-id=5
+ ```
+
+- [ ] **Vérifier** dans la base que :
+ - `ope_users_sectors.fk_user` contient des IDs de `ope_users.id`
+ - `ope_pass.fk_user` contient des IDs de `ope_users.id`
+ - Les valeurs correspondent bien au mapping
+
+- [ ] **Vérifier** qu'on peut supprimer une opération et que tout part avec (CASCADE)
+ ```sql
+ DELETE FROM operations WHERE id = 850;
+ -- Doit supprimer automatiquement :
+ -- - ope_users (ON DELETE CASCADE depuis operations)
+ -- - ope_users_sectors (ON DELETE CASCADE depuis ope_users)
+ -- - ope_pass (ON DELETE CASCADE depuis ope_users)
+ -- - ope_sectors (ON DELETE CASCADE depuis operations)
+ ```
+
+---
+
+## ✅ Tâche 3 : Vérifications API
+
+### Impact sur les endpoints API
+
+#### 3.1 Vérifier les requêtes utilisant `ope_pass.fk_user`
+
+- [ ] **Rechercher** tous les endpoints qui lisent `ope_pass.fk_user`
+ ```bash
+ grep -r "ope_pass.*fk_user" src/Controllers/
+ grep -r "fk_user.*ope_pass" src/Controllers/
+ ```
+
+- [ ] **Vérifier** que ces endpoints :
+ - Font-ils des JOIN avec `users` via `ope_pass.fk_user` ?
+ - Si OUI : Ajouter un JOIN via `ope_users` :
+ ```sql
+ -- ❌ AVANT
+ SELECT op.*, u.encrypted_name
+ FROM ope_pass op
+ JOIN users u ON op.fk_user = u.id
+
+ -- ✅ APRÈS
+ SELECT op.*, u.encrypted_name
+ FROM ope_pass op
+ JOIN ope_users ou ON op.fk_user = ou.id
+ JOIN users u ON ou.fk_user = u.id
+ ```
+
+#### 3.2 Vérifier les requêtes utilisant `ope_users_sectors.fk_user`
+
+- [ ] **Rechercher** tous les endpoints qui lisent `ope_users_sectors.fk_user`
+ ```bash
+ grep -r "ope_users_sectors.*fk_user" src/Controllers/
+ ```
+
+- [ ] **Vérifier** la même chose : si JOIN avec `users`, ajouter passage par `ope_users`
+
+#### 3.3 Endpoints probablement concernés
+
+À vérifier :
+- [ ] `OperationController` - Liste des utilisateurs d'une opération
+- [ ] `PassageController` - Liste/détails des passages
+- [ ] `SectorController` - Liste des secteurs avec utilisateurs affectés
+- [ ] Tout endpoint retournant des statistiques par utilisateur
+
+---
+
+## ✅ Tâche 4 : Corrections API - Response JSON Login
+
+### Impact sur la réponse JSON du login
+
+#### 4.1 Groupe `users_sectors` - Ajouter `ope_user_id`
+
+**Problème identifié** : Flutter reçoit `users_sectors` avec `id` (users.id) mais les `passages` ont `fk_user` (ope_users.id). Le mapping est impossible.
+
+**Solution** : Modifier la requête dans `LoginController.php` (lignes 426 et 1181) pour retourner les deux IDs :
+
+```sql
+-- ✅ APRÈS
+SELECT DISTINCT
+ u.id as user_id, -- users.id (table centrale, pour gestion membres)
+ ou.id as ope_user_id, -- ope_users.id (pour lier avec passages/sectors)
+ ou.first_name,
+ u.encrypted_name,
+ u.sect_name,
+ us.fk_sector
+FROM users u
+JOIN ope_users ou ON u.id = ou.fk_user
+JOIN ope_users_sectors us ON ou.id = us.fk_user AND ou.fk_operation = us.fk_operation
+WHERE us.fk_sector IN ($sectorIdsString)
+AND us.fk_operation = ?
+AND us.chk_active = 1
+AND u.chk_active = 1
+AND u.id != ?
+```
+
+**Résultat JSON attendu** :
+```json
+{
+ "user_id": 123, // users.id (pour gestion des membres dans l'interface)
+ "ope_user_id": 50, // ope_users.id (pour lier avec passages.fk_user et sectors)
+ "first_name": "Jane",
+ "name": "Jane Smith",
+ "sect_name": "Smith",
+ "fk_sector": 456
+}
+```
+
+**Usage Flutter** :
+```dart
+// Trouver les passages d'un utilisateur
+passages.where((p) => p.fkUser == usersSectors[i].opeUserId) // ✅ OK
+```
+
+- [ ] **Modifier** `LoginController.php` ligne 426 (méthode `login()`)
+- [ ] **Modifier** `LoginController.php` ligne 1181 (méthode `checkSession()`)
+- [ ] **Tester** la réponse JSON du login en mode admin
+
+---
+
+## ✅ Tâche 5 : Vérifications Flutter - Gestion des IDs
+
+### Impact sur l'application mobile
+
+#### 5.1 Modèles de données
+
+- [ ] **Vérifier** le modèle `UserSector` (ou équivalent)
+ - Ajouter le champ `opeUserId` (int) pour stocker `ope_users.id`
+ - Conserver `userId` (int) pour stocker `users.id`
+
+- [ ] **Vérifier** le modèle `Passage` (ou équivalent)
+ - Le champ `fkUser` pointe maintenant vers `ope_users.id`
+
+#### 5.2 Gestion des secteurs (Mode Admin)
+
+- [ ] **Création de secteur**
+ - L'API crée dans `ope_sectors`
+ - Attribution des users : utiliser `ope_user_id` (pas `user_id`)
+ - Endpoint : `POST /api/sectors`
+ - Body : `{ ..., users: [50, 51, 52] }` ← IDs de `ope_users`
+
+- [ ] **Modification de secteur**
+ - Attribution des users : utiliser `ope_user_id`
+ - Endpoint : `PUT /api/sectors/:id`
+ - Body : `{ ..., users: [50, 51, 52] }` ← IDs de `ope_users`
+
+- [ ] **Suppression de secteur**
+ - L'API supprime dans `ope_pass`, `ope_users_sectors` et `ope_sectors`
+ - CASCADE gère automatiquement les dépendances
+ - Endpoint : `DELETE /api/sectors/:id`
+
+#### 5.3 Gestion des membres (Mode Admin)
+
+- [ ] **Création de membre**
+ - L'API crée dans `users` (table centrale)
+ - L'API crée aussi dans `ope_users` pour l'opération active
+ - **Réponse attendue** :
+ ```json
+ {
+ "status": "success",
+ "user": {
+ "id": 123, // users.id
+ "ope_user_id": 50, // ope_users.id (nouveau)
+ "first_name": "John",
+ "name": "John Doe",
+ ...
+ }
+ }
+ ```
+ - Endpoint : `POST /api/users`
+ - Flutter stocke les 2 IDs : `userId` et `opeUserId`
+
+- [ ] **Modification de membre**
+ - L'API met à jour `users` (table centrale)
+ - L'API met à jour aussi `ope_users` pour l'opération active
+ - Endpoint : `PUT /api/users/:id`
+
+- [ ] **Suppression de membre**
+ - L'API supprime de `ope_users` (opération active)
+ - L'API supprime de `users` (table centrale)
+ - CASCADE supprime automatiquement `ope_users_sectors` et `ope_pass`
+ - Endpoint : `DELETE /api/users/:id?transfer_to=XX`
+
+#### 5.4 Gestion des passages (Mode Admin & User)
+
+- [ ] **Création de passage**
+ - Attribution automatique du `ope_sectors.id` le plus proche
+ - Attribution du `ope_users.id` (utilisateur connecté ou sélectionné)
+ - Endpoint : `POST /api/passages`
+ - Body : `{ ..., fk_user: 50, fk_sector: 456 }` ← IDs de `ope_users` et `ope_sectors`
+
+- [ ] **Modification de passage**
+ - Attribution du `ope_users.id` si changement d'utilisateur
+ - Endpoint : `PUT /api/passages/:id`
+ - Body : `{ ..., fk_user: 50 }` ← ID de `ope_users`
+
+- [ ] **Suppression de passage**
+ - L'API supprime dans `ope_pass`
+ - Endpoint : `DELETE /api/passages/:id`
+
+#### 5.5 Interface Flutter - Mapping des IDs
+
+**Scénarios à gérer** :
+
+1. **Affichage des secteurs avec utilisateurs affectés** :
+ ```dart
+ // Utiliser usersSectors[i].opeUserId pour lier avec passages
+ final userPassages = passages.where((p) =>
+ p.fkUser == usersSectors[i].opeUserId &&
+ p.fkSector == sector.id
+ ).toList();
+ ```
+
+2. **Attribution d'un passage à un utilisateur** :
+ ```dart
+ // Envoyer ope_user_id dans la requête API
+ await apiService.createPassage({
+ ...passageData,
+ 'fk_user': userSector.opeUserId, // ope_users.id
+ 'fk_sector': sector.id
+ });
+ ```
+
+3. **Affichage du nom d'un utilisateur depuis un passage** :
+ ```dart
+ // Chercher dans usersSectors avec ope_user_id
+ final userSector = usersSectors.firstWhere(
+ (us) => us.opeUserId == passage.fkUser,
+ orElse: () => null
+ );
+ final userName = userSector?.name ?? 'Inconnu';
+ ```
+
+4. **Gestion des membres** :
+ ```dart
+ // Conserver les 2 IDs lors de la création
+ final newMember = await apiService.createUser(userData);
+ membres.add(Member(
+ userId: newMember['id'], // users.id
+ opeUserId: newMember['ope_user_id'], // ope_users.id
+ ...
+ ));
+ ```
+
+#### 5.6 Tests d'affichage
+
+- [ ] Tester l'affichage des passages avec noms d'utilisateurs
+- [ ] Tester l'affichage des secteurs avec utilisateurs affectés
+- [ ] Tester la création d'un membre (vérifier que les 2 IDs sont reçus)
+- [ ] Tester la suppression d'un membre (vérifier le transfert de passages)
+- [ ] Tester la création d'un secteur avec attribution d'utilisateurs
+- [ ] Tester la création d'un passage avec attribution d'utilisateur
+- [ ] Tester la suppression d'une opération (doit tout nettoyer)
+
+---
+
+## 📋 Ordre d'exécution recommandé
+
+1. ✅ **Corriger le code de migration2** (PHP)
+2. ✅ **Tester sur dva_geo** avec schéma modifié
+3. ✅ **Vérifier l'API** sur dva_geo
+4. ✅ **Vérifier Flutter** avec dva_geo
+5. 🚀 **Déployer le schéma SQL** sur rca_geo
+6. 🚀 **Déployer le code** sur rca_geo
+7. ✅ **Tester en recette**
+8. 🚀 **Déployer en production** (pra_geo)
+
+---
+
+## 🔍 Requêtes SQL utiles pour vérification
+
+### Vérifier les contraintes FK actuelles
+
+```sql
+SELECT
+ TABLE_NAME,
+ COLUMN_NAME,
+ CONSTRAINT_NAME,
+ REFERENCED_TABLE_NAME,
+ REFERENCED_COLUMN_NAME
+FROM information_schema.KEY_COLUMN_USAGE
+WHERE TABLE_SCHEMA = DATABASE()
+AND (TABLE_NAME = 'ope_pass' OR TABLE_NAME = 'ope_users_sectors')
+AND COLUMN_NAME = 'fk_user';
+```
+
+### Vérifier l'intégrité des données après migration
+
+```sql
+-- Vérifier que tous les fk_user de ope_pass existent dans ope_users
+SELECT COUNT(*) as orphans
+FROM ope_pass op
+LEFT JOIN ope_users ou ON op.fk_user = ou.id
+WHERE ou.id IS NULL;
+-- Résultat attendu : 0
+
+-- Vérifier que tous les fk_user de ope_users_sectors existent dans ope_users
+SELECT COUNT(*) as orphans
+FROM ope_users_sectors ous
+LEFT JOIN ope_users ou ON ous.fk_user = ou.id
+WHERE ou.id IS NULL;
+-- Résultat attendu : 0
+```
+
+### Tester la suppression en cascade
+
+```sql
+-- Compter avant suppression
+SELECT
+ (SELECT COUNT(*) FROM ope_users WHERE fk_operation = 850) as ope_users_count,
+ (SELECT COUNT(*) FROM ope_users_sectors WHERE fk_operation = 850) as ope_users_sectors_count,
+ (SELECT COUNT(*) FROM ope_pass WHERE fk_operation = 850) as ope_pass_count,
+ (SELECT COUNT(*) FROM ope_sectors WHERE fk_operation = 850) as ope_sectors_count;
+
+-- Supprimer l'opération
+DELETE FROM operations WHERE id = 850;
+
+-- Vérifier que tout a été supprimé (doit retourner 0 partout)
+SELECT
+ (SELECT COUNT(*) FROM ope_users WHERE fk_operation = 850) as ope_users_count,
+ (SELECT COUNT(*) FROM ope_users_sectors WHERE fk_operation = 850) as ope_users_sectors_count,
+ (SELECT COUNT(*) FROM ope_pass WHERE fk_operation = 850) as ope_pass_count,
+ (SELECT COUNT(*) FROM ope_sectors WHERE fk_operation = 850) as ope_sectors_count;
+```
+
+---
+
+## 📝 Notes importantes
+
+### Avantages de cette architecture
+
+✅ **Isolation complète** : Supprimer une opération supprime tout (ope_users, secteurs, passages)
+✅ **Performance** : Pas de jointures complexes avec la table centrale `users`
+✅ **Historique** : Les données d'une opération sont figées dans le temps
+✅ **Simplicité** : Requêtes plus simples, moins de risques d'incohérences
+
+### Implications
+
+⚠️ **Duplication** : Un utilisateur travaillant sur 3 opérations aura 3 entrées dans `ope_users`
+⚠️ **Taille** : La table `ope_users` sera plus volumineuse
+⚠️ **Jointures** : Pour remonter aux infos de la table `users`, il faut passer par `ope_users.fk_user`
+
+### Rétrocompatibilité
+
+❌ Ce changement **CASSE** la compatibilité avec les données existantes
+✅ Nécessite une **re-migration complète** de toutes les entités après modification du schéma
+✅ Ou bien : script de transformation des données existantes (plus complexe)
+
+---
+
+## 🎯 Statut
+
+- [ ] Schéma SQL modifié sur dva_geo
+- [ ] Code migration2 corrigé
+- [ ] API vérifiée et corrigée
+- [ ] Flutter vérifié et corrigé
+- [ ] Tests complets sur dva_geo
+- [ ] Déploiement rca_geo
+- [ ] Déploiement pra_geo
diff --git a/api/scripts/orga/fix_fk_constraints.sql b/api/scripts/orga/fix_fk_constraints.sql
new file mode 100644
index 00000000..3c4ada97
--- /dev/null
+++ b/api/scripts/orga/fix_fk_constraints.sql
@@ -0,0 +1,65 @@
+-- ================================================================================
+-- Script de migration : Correction des contraintes FK pour isolation par opération
+-- ================================================================================
+--
+-- Ce script modifie les contraintes de clés étrangères pour que :
+-- - ope_users_sectors.fk_user → pointe vers ope_users.id (au lieu de users.id)
+-- - ope_pass.fk_user → pointe vers ope_users.id (au lieu de users.id)
+--
+-- Cela permet une isolation complète des opérations : supprimer une opération
+-- supprime automatiquement tous ses ope_users, ope_sectors, ope_users_sectors et ope_pass.
+--
+-- ORDRE D'EXÉCUTION :
+-- 1. dva_geo (DEV) - test
+-- 2. rca_geo (RECETTE)
+-- 3. pra_geo (PRODUCTION)
+--
+-- ================================================================================
+
+USE dva_geo; -- Adapter selon l'environnement (dva_geo, rca_geo, pra_geo)
+
+-- ================================================================================
+-- 1. Modification de ope_users_sectors.fk_user
+-- ================================================================================
+
+-- Supprimer l'ancienne contrainte FK
+ALTER TABLE ope_users_sectors
+DROP FOREIGN KEY ope_users_sectors_ibfk_2;
+
+-- Recréer la contrainte FK vers ope_users.id
+ALTER TABLE ope_users_sectors
+ADD CONSTRAINT ope_users_sectors_ibfk_2
+FOREIGN KEY (fk_user) REFERENCES ope_users (id) ON DELETE CASCADE ON UPDATE CASCADE;
+
+-- ================================================================================
+-- 2. Modification de ope_pass.fk_user
+-- ================================================================================
+
+-- Supprimer l'ancienne contrainte FK
+ALTER TABLE ope_pass
+DROP FOREIGN KEY ope_pass_ibfk_3;
+
+-- Recréer la contrainte FK vers ope_users.id
+ALTER TABLE ope_pass
+ADD CONSTRAINT ope_pass_ibfk_3
+FOREIGN KEY (fk_user) REFERENCES ope_users (id) ON DELETE CASCADE ON UPDATE CASCADE;
+
+-- ================================================================================
+-- Vérification finale
+-- ================================================================================
+
+SELECT
+ TABLE_NAME,
+ COLUMN_NAME,
+ CONSTRAINT_NAME,
+ REFERENCED_TABLE_NAME,
+ REFERENCED_COLUMN_NAME
+FROM information_schema.KEY_COLUMN_USAGE
+WHERE TABLE_SCHEMA = DATABASE()
+AND TABLE_NAME IN ('ope_users_sectors', 'ope_pass')
+AND COLUMN_NAME = 'fk_user'
+ORDER BY TABLE_NAME;
+
+-- Résultat attendu :
+-- ope_pass | fk_user | ope_pass_ibfk_3 | ope_users | id
+-- ope_users_sectors | fk_user | ope_users_sectors_ibfk_2 | ope_users | id
diff --git a/api/scripts/orga/fix_fk_constraints_safe.sql b/api/scripts/orga/fix_fk_constraints_safe.sql
new file mode 100644
index 00000000..ab7a8c9c
--- /dev/null
+++ b/api/scripts/orga/fix_fk_constraints_safe.sql
@@ -0,0 +1,121 @@
+-- ================================================================================
+-- Script de migration SÉCURISÉ : Correction des contraintes FK pour isolation par opération
+-- ================================================================================
+--
+-- Ce script modifie les contraintes de clés étrangères pour que :
+-- - ope_users_sectors.fk_user → pointe vers ope_users.id (au lieu de users.id)
+-- - ope_pass.fk_user → pointe vers ope_users.id (au lieu de users.id)
+--
+-- Version SÉCURISÉE : Vérifie l'existence des contraintes avant de les supprimer
+--
+-- ================================================================================
+
+USE dva_geo;
+
+-- ================================================================================
+-- Afficher les contraintes FK actuelles
+-- ================================================================================
+
+SELECT
+ TABLE_NAME,
+ COLUMN_NAME,
+ CONSTRAINT_NAME,
+ REFERENCED_TABLE_NAME,
+ REFERENCED_COLUMN_NAME
+FROM information_schema.KEY_COLUMN_USAGE
+WHERE TABLE_SCHEMA = 'dva_geo'
+AND TABLE_NAME IN ('ope_users_sectors', 'ope_pass')
+AND COLUMN_NAME = 'fk_user'
+ORDER BY TABLE_NAME;
+
+-- ================================================================================
+-- 1. Modification de ope_users_sectors.fk_user
+-- ================================================================================
+
+-- Supprimer l'ancienne contrainte FK si elle existe
+SET @constraint_exists = (
+ SELECT COUNT(*)
+ FROM information_schema.KEY_COLUMN_USAGE
+ WHERE TABLE_SCHEMA = 'dva_geo'
+ AND TABLE_NAME = 'ope_users_sectors'
+ AND COLUMN_NAME = 'fk_user'
+ AND CONSTRAINT_NAME LIKE '%ibfk%'
+);
+
+SET @sql = IF(@constraint_exists > 0,
+ CONCAT('ALTER TABLE ope_users_sectors DROP FOREIGN KEY ',
+ (SELECT CONSTRAINT_NAME
+ FROM information_schema.KEY_COLUMN_USAGE
+ WHERE TABLE_SCHEMA = 'dva_geo'
+ AND TABLE_NAME = 'ope_users_sectors'
+ AND COLUMN_NAME = 'fk_user'
+ AND CONSTRAINT_NAME LIKE '%ibfk%'
+ LIMIT 1)),
+ 'SELECT "Aucune contrainte FK à supprimer sur ope_users_sectors" AS message'
+);
+
+PREPARE stmt FROM @sql;
+EXECUTE stmt;
+DEALLOCATE PREPARE stmt;
+
+-- Recréer la contrainte FK vers ope_users.id
+ALTER TABLE ope_users_sectors
+ADD CONSTRAINT ope_users_sectors_ibfk_2
+FOREIGN KEY (fk_user) REFERENCES ope_users (id) ON DELETE CASCADE ON UPDATE CASCADE;
+
+-- ================================================================================
+-- 2. Modification de ope_pass.fk_user
+-- ================================================================================
+
+-- Supprimer l'ancienne contrainte FK si elle existe
+SET @constraint_exists = (
+ SELECT COUNT(*)
+ FROM information_schema.KEY_COLUMN_USAGE
+ WHERE TABLE_SCHEMA = 'dva_geo'
+ AND TABLE_NAME = 'ope_pass'
+ AND COLUMN_NAME = 'fk_user'
+ AND CONSTRAINT_NAME LIKE '%ibfk%'
+);
+
+SET @sql = IF(@constraint_exists > 0,
+ CONCAT('ALTER TABLE ope_pass DROP FOREIGN KEY ',
+ (SELECT CONSTRAINT_NAME
+ FROM information_schema.KEY_COLUMN_USAGE
+ WHERE TABLE_SCHEMA = 'dva_geo'
+ AND TABLE_NAME = 'ope_pass'
+ AND COLUMN_NAME = 'fk_user'
+ AND CONSTRAINT_NAME LIKE '%ibfk%'
+ LIMIT 1)),
+ 'SELECT "Aucune contrainte FK à supprimer sur ope_pass" AS message'
+);
+
+PREPARE stmt FROM @sql;
+EXECUTE stmt;
+DEALLOCATE PREPARE stmt;
+
+-- Recréer la contrainte FK vers ope_users.id
+ALTER TABLE ope_pass
+ADD CONSTRAINT ope_pass_ibfk_3
+FOREIGN KEY (fk_user) REFERENCES ope_users (id) ON DELETE CASCADE ON UPDATE CASCADE;
+
+-- ================================================================================
+-- Vérification finale
+-- ================================================================================
+
+SELECT
+ TABLE_NAME,
+ COLUMN_NAME,
+ CONSTRAINT_NAME,
+ REFERENCED_TABLE_NAME,
+ REFERENCED_COLUMN_NAME
+FROM information_schema.KEY_COLUMN_USAGE
+WHERE TABLE_SCHEMA = 'dva_geo'
+AND TABLE_NAME IN ('ope_users_sectors', 'ope_pass')
+AND COLUMN_NAME = 'fk_user'
+ORDER BY TABLE_NAME;
+
+-- Résultat attendu :
+-- ope_pass | fk_user | ope_pass_ibfk_3 | ope_users | id
+-- ope_users_sectors | fk_user | ope_users_sectors_ibfk_2 | ope_users | id
+
+SELECT '✓ Contraintes FK modifiées avec succès !' AS status;
diff --git a/api/scripts/orga/truncate_all_tables.sql b/api/scripts/orga/truncate_all_tables.sql
new file mode 100644
index 00000000..63a485ac
--- /dev/null
+++ b/api/scripts/orga/truncate_all_tables.sql
@@ -0,0 +1,93 @@
+-- ================================================================================
+-- Script de nettoyage complet des tables - DVA_GEO
+-- ================================================================================
+--
+-- Ce script vide toutes les tables pour repartir à zéro.
+-- ATTENTION : Toutes les données seront perdues !
+--
+-- Usage : À exécuter sur dva_geo UNIQUEMENT (environnement de développement)
+--
+-- ================================================================================
+
+USE dva_geo;
+
+-- Désactiver temporairement les vérifications de clés étrangères
+SET FOREIGN_KEY_CHECKS = 0;
+
+-- ================================================================================
+-- 1. Tables dépendantes (dans l'ordre des dépendances)
+-- ================================================================================
+
+TRUNCATE TABLE ope_pass_histo;
+TRUNCATE TABLE ope_pass;
+TRUNCATE TABLE ope_users_sectors;
+TRUNCATE TABLE sectors_adresses;
+TRUNCATE TABLE ope_sectors;
+TRUNCATE TABLE ope_users;
+TRUNCATE TABLE medias;
+TRUNCATE TABLE operations;
+
+-- ================================================================================
+-- 2. Tables liées aux utilisateurs
+-- ================================================================================
+
+TRUNCATE TABLE user_devices;
+
+-- ================================================================================
+-- 3. Tables de chat
+-- ================================================================================
+
+TRUNCATE TABLE chat_messages;
+TRUNCATE TABLE chat_participants;
+TRUNCATE TABLE chat_read_receipts;
+TRUNCATE TABLE chat_rooms;
+
+-- ================================================================================
+-- 4. Tables principales
+-- ================================================================================
+
+TRUNCATE TABLE users;
+TRUNCATE TABLE entites;
+
+-- Réactiver les vérifications de clés étrangères
+SET FOREIGN_KEY_CHECKS = 1;
+
+-- ================================================================================
+-- Vérification : Compter les lignes restantes
+-- ================================================================================
+
+SELECT
+ 'ope_pass_histo' AS table_name, COUNT(*) AS rows_count FROM ope_pass_histo
+UNION ALL
+SELECT 'ope_pass', COUNT(*) FROM ope_pass
+UNION ALL
+SELECT 'ope_users_sectors', COUNT(*) FROM ope_users_sectors
+UNION ALL
+SELECT 'sectors_adresses', COUNT(*) FROM sectors_adresses
+UNION ALL
+SELECT 'ope_sectors', COUNT(*) FROM ope_sectors
+UNION ALL
+SELECT 'ope_users', COUNT(*) FROM ope_users
+UNION ALL
+SELECT 'medias', COUNT(*) FROM medias
+UNION ALL
+SELECT 'operations', COUNT(*) FROM operations
+UNION ALL
+SELECT 'user_devices', COUNT(*) FROM user_devices
+UNION ALL
+SELECT 'chat_messages', COUNT(*) FROM chat_messages
+UNION ALL
+SELECT 'chat_participants', COUNT(*) FROM chat_participants
+UNION ALL
+SELECT 'chat_read_receipts', COUNT(*) FROM chat_read_receipts
+UNION ALL
+SELECT 'chat_rooms', COUNT(*) FROM chat_rooms
+UNION ALL
+SELECT 'users', COUNT(*) FROM users
+UNION ALL
+SELECT 'entites', COUNT(*) FROM entites
+ORDER BY table_name;
+
+-- Résultat attendu : 0 partout
+
+SELECT '✓ Toutes les tables ont été vidées avec succès !' AS status;
diff --git a/api/scripts/orga/verify_isolation.sql b/api/scripts/orga/verify_isolation.sql
new file mode 100644
index 00000000..16a7b6a8
--- /dev/null
+++ b/api/scripts/orga/verify_isolation.sql
@@ -0,0 +1,150 @@
+-- ================================================================================
+-- Script de vérification : Isolation complète des opérations
+-- ================================================================================
+--
+-- Ce script vérifie que l'isolation par opération fonctionne correctement
+--
+-- ================================================================================
+
+USE dva_geo;
+
+-- ================================================================================
+-- 1. Vérifier les contraintes FK
+-- ================================================================================
+
+SELECT '=== VÉRIFICATION DES CONTRAINTES FK ===' AS '';
+
+SELECT
+ TABLE_NAME,
+ COLUMN_NAME,
+ CONSTRAINT_NAME,
+ REFERENCED_TABLE_NAME,
+ REFERENCED_COLUMN_NAME
+FROM information_schema.KEY_COLUMN_USAGE
+WHERE TABLE_SCHEMA = 'dva_geo'
+AND TABLE_NAME IN ('ope_users_sectors', 'ope_pass')
+AND COLUMN_NAME = 'fk_user'
+ORDER BY TABLE_NAME;
+
+-- Résultat attendu :
+-- ope_pass | fk_user | ope_pass_ibfk_3 | ope_users | id
+-- ope_users_sectors | fk_user | ope_users_sectors_ibfk_2 | ope_users | id
+
+-- ================================================================================
+-- 2. Vérifier l'intégrité des données (pas d'orphelins)
+-- ================================================================================
+
+SELECT '=== VÉRIFICATION INTÉGRITÉ DES DONNÉES ===' AS '';
+
+-- Vérifier que tous les fk_user de ope_pass existent dans ope_users
+SELECT
+ 'ope_pass → ope_users' AS verification,
+ COUNT(*) as orphelins
+FROM ope_pass op
+LEFT JOIN ope_users ou ON op.fk_user = ou.id
+WHERE ou.id IS NULL;
+-- Résultat attendu : 0
+
+-- Vérifier que tous les fk_user de ope_users_sectors existent dans ope_users
+SELECT
+ 'ope_users_sectors → ope_users' AS verification,
+ COUNT(*) as orphelins
+FROM ope_users_sectors ous
+LEFT JOIN ope_users ou ON ous.fk_user = ou.id
+WHERE ou.id IS NULL;
+-- Résultat attendu : 0
+
+-- ================================================================================
+-- 3. Statistiques de migration
+-- ================================================================================
+
+SELECT '=== STATISTIQUES DE MIGRATION ===' AS '';
+
+-- Nombre d'entités
+SELECT 'Entités' AS table_name, COUNT(*) AS count FROM entites
+UNION ALL
+-- Nombre d'opérations
+SELECT 'Opérations' AS table_name, COUNT(*) AS count FROM operations
+UNION ALL
+-- Nombre d'utilisateurs dans la table centrale
+SELECT 'Users (centrale)' AS table_name, COUNT(*) AS count FROM users
+UNION ALL
+-- Nombre d'utilisateurs dans les opérations
+SELECT 'ope_users' AS table_name, COUNT(*) AS count FROM ope_users
+UNION ALL
+-- Nombre de secteurs
+SELECT 'ope_sectors' AS table_name, COUNT(*) AS count FROM ope_sectors
+UNION ALL
+-- Nombre d'associations user-secteur
+SELECT 'ope_users_sectors' AS table_name, COUNT(*) AS count FROM ope_users_sectors
+UNION ALL
+-- Nombre de passages
+SELECT 'ope_pass' AS table_name, COUNT(*) AS count FROM ope_pass
+UNION ALL
+-- Nombre d'historiques de passage
+SELECT 'ope_pass_histo' AS table_name, COUNT(*) AS count FROM ope_pass_histo;
+
+-- ================================================================================
+-- 4. Détail par opération
+-- ================================================================================
+
+SELECT '=== DÉTAIL PAR OPÉRATION ===' AS '';
+
+SELECT
+ o.id AS operation_id,
+ o.libelle AS operation_name,
+ (SELECT COUNT(*) FROM ope_users WHERE fk_operation = o.id) AS nb_users,
+ (SELECT COUNT(*) FROM ope_sectors WHERE fk_operation = o.id) AS nb_sectors,
+ (SELECT COUNT(*) FROM ope_users_sectors WHERE fk_operation = o.id) AS nb_user_sector_links,
+ (SELECT COUNT(*) FROM ope_pass WHERE fk_operation = o.id) AS nb_passages
+FROM operations o
+ORDER BY o.id;
+
+-- ================================================================================
+-- 5. Vérifier la relation users → ope_users
+-- ================================================================================
+
+SELECT '=== RELATION users → ope_users ===' AS '';
+
+SELECT
+ u.id AS user_id,
+ u.first_name,
+ u.sect_name,
+ COUNT(DISTINCT ou.fk_operation) AS nb_operations,
+ GROUP_CONCAT(DISTINCT ou.fk_operation ORDER BY ou.fk_operation) AS operations_ids
+FROM users u
+LEFT JOIN ope_users ou ON u.id = ou.fk_user
+GROUP BY u.id, u.first_name, u.sect_name
+ORDER BY u.id;
+
+-- ================================================================================
+-- 6. TEST DE SUPPRESSION (commenté pour sécurité)
+-- ================================================================================
+
+SELECT '=== INSTRUCTIONS POUR TEST DE SUPPRESSION ===' AS '';
+SELECT 'Pour tester la suppression en CASCADE, décommentez la section ci-dessous' AS instruction;
+
+-- Compter avant suppression (remplacer [ID_OPERATION] par un ID réel)
+/*
+SET @operation_id = [ID_OPERATION];
+
+SELECT
+ CONCAT('Opération ID: ', @operation_id) AS info,
+ (SELECT COUNT(*) FROM ope_users WHERE fk_operation = @operation_id) as ope_users_count,
+ (SELECT COUNT(*) FROM ope_users_sectors WHERE fk_operation = @operation_id) as ope_users_sectors_count,
+ (SELECT COUNT(*) FROM ope_pass WHERE fk_operation = @operation_id) as ope_pass_count,
+ (SELECT COUNT(*) FROM ope_sectors WHERE fk_operation = @operation_id) as ope_sectors_count;
+
+-- Supprimer l'opération
+DELETE FROM operations WHERE id = @operation_id;
+
+-- Vérifier que tout a été supprimé (doit retourner 0 partout)
+SELECT
+ CONCAT('Après suppression de l''opération ID: ', @operation_id) AS info,
+ (SELECT COUNT(*) FROM ope_users WHERE fk_operation = @operation_id) as ope_users_count,
+ (SELECT COUNT(*) FROM ope_users_sectors WHERE fk_operation = @operation_id) as ope_users_sectors_count,
+ (SELECT COUNT(*) FROM ope_pass WHERE fk_operation = @operation_id) as ope_pass_count,
+ (SELECT COUNT(*) FROM ope_sectors WHERE fk_operation = @operation_id) as ope_sectors_count;
+*/
+
+SELECT '✓ Vérifications terminées avec succès !' AS status;
diff --git a/api/scripts/patch_migration_scripts.sh b/api/scripts/patch_migration_scripts.sh
new file mode 100644
index 00000000..bb28b548
--- /dev/null
+++ b/api/scripts/patch_migration_scripts.sh
@@ -0,0 +1,182 @@
+#!/bin/bash
+#
+# Script de patch pour adapter migrate_from_backup.php et migrate_batch.sh
+# pour fonctionner avec --env=rca|pra et source=geosector
+#
+
+set -euo pipefail
+
+SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
+PHP_SCRIPT="$SCRIPT_DIR/php/migrate_from_backup.php"
+BATCH_SCRIPT="$SCRIPT_DIR/migrate_batch.sh"
+
+echo "=== Patching migration scripts ==="
+echo ""
+
+# Backup des fichiers originaux
+echo "Creating backups..."
+cp "$PHP_SCRIPT" "$PHP_SCRIPT.backup"
+cp "$BATCH_SCRIPT" "$BATCH_SCRIPT.backup"
+echo "✓ Backups created"
+echo ""
+
+# ============================================================
+# PATCH 1: migrate_from_backup.php - Configuration multi-env
+# ============================================================
+
+echo "Patching migrate_from_backup.php..."
+
+# Étape 1: Remplacer les constantes DB par configuration multi-env
+sed -i '31,50s/.*/ \/\/ REPLACED BY PATCH - see below/' "$PHP_SCRIPT"
+
+# Insérer la nouvelle configuration après la ligne 38
+sed -i '38a\
+ private $env;\
+\
+ \/\/ Configuration multi-environnement\
+ private const ENVIRONMENTS = [\
+ '\''rca'\'' => [\
+ '\''host'\'' => '\''13.23.33.3'\'', \/\/ maria3 sur IN3\
+ '\''port'\'' => 3306,\
+ '\''user'\'' => '\''rca_geo_user'\'',\
+ '\''pass'\'' => '\''UPf3C0cQ805LypyM71iW'\'',\
+ '\''target_db'\'' => '\''rca_geo'\'',\
+ '\''source_db'\'' => '\''geosector'\'' \/\/ Base synchronisée par PM7\
+ ],\
+ '\''pra'\'' => [\
+ '\''host'\'' => '\''13.23.33.4'\'', \/\/ maria4 sur IN4\
+ '\''port'\'' => 3306,\
+ '\''user'\'' => '\''pra_geo_user'\'',\
+ '\''pass'\'' => '\''d2jAAGGWi8fxFrWgXjOA'\'',\
+ '\''target_db'\'' => '\''pra_geo'\'',\
+ '\''source_db'\'' => '\''geosector'\'' \/\/ Base synchronisée par PM7\
+ ]\
+ ];' "$PHP_SCRIPT"
+
+# Étape 2: Modifier le constructeur pour accepter $env
+sed -i 's/public function __construct($sourceDbName, $targetDbName, $mode/public function __construct($env, $mode/' "$PHP_SCRIPT"
+
+# Étape 3: Adapter le corps du constructeur
+sed -i '/public function __construct/,/^ }$/{
+ s/\$this->sourceDbName = \$sourceDbName;/\$this->env = \$env;\n if (!isset(self::ENVIRONMENTS[\$env])) {\n throw new Exception("Invalid environment: \$env. Use '\''rca'\'' or '\''pra'\''");\n }\n \$config = self::ENVIRONMENTS[\$env];\n \$this->sourceDbName = \$config['\''source_db'\''];\n \$this->targetDbName = \$config['\''target_db'\''];/
+ s/\$this->targetDbName = \$targetDbName;//
+ s/Source: {\$sourceDbName}/Environment: \$env/
+ s/Cible: {\$targetDbName}/Source: {\$this->sourceDbName} → Target: {\$this->targetDbName}/
+}' "$PHP_SCRIPT"
+
+# Étape 4: Modifier connect() pour utiliser la config de l'env
+sed -i '/public function connect()/,/^ }$/{
+ s/self::DB_HOST/self::ENVIRONMENTS[\$this->env]['\''host'\'']/g
+ s/self::DB_PORT/self::ENVIRONMENTS[\$this->env]['\''port'\'']/g
+ s/self::DB_USER_ROOT/self::ENVIRONMENTS[\$this->env]['\''user'\'']/g
+ s/self::DB_PASS_ROOT/self::ENVIRONMENTS[\$this->env]['\''pass'\'']/g
+ s/self::DB_USER/self::ENVIRONMENTS[\$this->env]['\''user'\'']/g
+ s/self::DB_PASS/self::ENVIRONMENTS[\$this->env]['\''pass'\'']/g
+}' "$PHP_SCRIPT"
+
+# Étape 5: Modifier parseArguments() - supprimer source-db et target-db, ajouter env
+sed -i '/function parseArguments/,/^}$/{
+ s/'\''source-db'\'' => null,/'\''env'\'' => '\''rca'\'',/
+ s/'\''target-db'\'' => '\''pra_geo'\'',//
+}' "$PHP_SCRIPT"
+
+# Étape 6: Modifier showHelp()
+sed -i '/function showHelp/,/^}$/{
+ s/--source-db=NAME.*\[REQUIS\]/--env=ENV Environment: '\''rca'\'' (recette) ou '\''pra'\'' (production) [défaut: rca]/
+ s/--target-db=NAME.*/ (supprimé - déduit automatiquement de --env)/
+ s/--source-db=geosector_20251007/--env=rca/g
+ s/--target-db=pra_geo//g
+ s/--target-db=rca_geo//g
+}' "$PHP_SCRIPT"
+
+# Étape 7: Modifier la validation des arguments
+sed -i '/Validation des arguments/,/exit(1);/{
+ s/if (!$args\['\''source-db'\''\])/if (!isset(self::ENVIRONMENTS[\$args['\''env'\'']]))/
+ s/--source-db est requis/--env doit être '\''rca'\'' ou '\''pra'\''/
+}' "$PHP_SCRIPT"
+
+# Étape 8: Modifier l'instanciation de BackupMigration
+sed -i '/new BackupMigration/,/);/{
+ s/\$args\['\''source-db'\''\],/\$args['\''env'\''],/
+ s/\$args\['\''target-db'\''\],//
+}' "$PHP_SCRIPT"
+
+echo "✓ migrate_from_backup.php patched"
+echo ""
+
+# ============================================================
+# PATCH 2: migrate_batch.sh - Adapter pour env rca/pra
+# ============================================================
+
+echo "Patching migrate_batch.sh..."
+
+# Étape 1: Détecter l'environnement automatiquement ou via paramètre
+sed -i '/# Configuration/a\
+\
+# Détection automatique de l'\''environnement\
+if [ -f "/etc/hostname" ]; then\
+ CONTAINER_NAME=$(cat /etc/hostname)\
+ case $CONTAINER_NAME in\
+ rca-geo)\
+ ENV="rca"\
+ ;;\
+ pra-geo)\
+ ENV="pra"\
+ ;;\
+ *)\
+ ENV="rca" # Défaut\
+ ;;\
+ esac\
+else\
+ ENV="rca" # Défaut\
+fi' "$BATCH_SCRIPT"
+
+# Étape 2: Remplacer SOURCE_DB et TARGET_DB
+sed -i 's/SOURCE_DB="geosector_20251013_13"/# SOURCE_DB removed - always "geosector" (deduced from --env)/' "$BATCH_SCRIPT"
+sed -i 's/TARGET_DB="pra_geo"/# TARGET_DB removed - deduced from --env/' "$BATCH_SCRIPT"
+
+# Étape 3: Ajouter option --env dans le parsing
+sed -i '/--interactive|-i)/i\
+ --env)\
+ ENV="$2"\
+ shift 2\
+ ;;' "$BATCH_SCRIPT"
+
+# Étape 4: Modifier les appels à migrate_from_backup.php - ligne 200
+sed -i '200,210s/--source-db="\$SOURCE_DB"/--env="$ENV"/' "$BATCH_SCRIPT"
+sed -i '200,210s/--target-db="\$TARGET_DB"//' "$BATCH_SCRIPT"
+
+# Étape 5: Modifier les appels dans la boucle - ligne 374
+sed -i '374,380s/--source-db="\$SOURCE_DB"/--env="$ENV"/' "$BATCH_SCRIPT"
+sed -i '374,380s/--target-db="\$TARGET_DB"//' "$BATCH_SCRIPT"
+
+# Étape 6: Mettre à jour les messages de log
+sed -i 's/📁 Source: \$SOURCE_DB/🌍 Environment: $ENV/' "$BATCH_SCRIPT"
+sed -i 's/📁 Cible: \$TARGET_DB/📁 Source: geosector → Target: (déduit de $ENV)/' "$BATCH_SCRIPT"
+
+echo "✓ migrate_batch.sh patched"
+echo ""
+
+# ============================================================
+# Résumé
+# ============================================================
+
+echo "=== Patch completed ==="
+echo ""
+echo "Backups saved:"
+echo " - $PHP_SCRIPT.backup"
+echo " - $BATCH_SCRIPT.backup"
+echo ""
+echo "New usage:"
+echo " # Sur rca-geo (détection auto)"
+echo " ./migrate_batch.sh"
+echo ""
+echo " # Sur pra-geo avec --env explicite"
+echo " ./migrate_batch.sh --env=pra"
+echo ""
+echo " # Migration d'une entité spécifique"
+echo " php php/migrate_from_backup.php --env=rca --mode=entity --entity-id=45"
+echo ""
+echo "To restore backups:"
+echo " cp $PHP_SCRIPT.backup $PHP_SCRIPT"
+echo " cp $BATCH_SCRIPT.backup $BATCH_SCRIPT"
diff --git a/api/scripts/php/create_missing_stripe_locations.php b/api/scripts/php/create_missing_stripe_locations.php
new file mode 100755
index 00000000..52f76ecf
--- /dev/null
+++ b/api/scripts/php/create_missing_stripe_locations.php
@@ -0,0 +1,240 @@
+#!/usr/bin/env php
+getEnvironment();
+$dbConfig = $config->getDatabaseConfig();
+
+echo "\n";
+echo "=============================================================================\n";
+echo " Création des Stripe Terminal Locations manquantes\n";
+echo "=============================================================================\n";
+echo "Environnement : " . strtoupper($env) . "\n";
+echo "Base de données : " . $dbConfig['name'] . "\n";
+echo "\n";
+
+try {
+ // Initialiser la base de données avec la configuration
+ Database::init($dbConfig);
+ $db = Database::getInstance();
+
+ // StripeService est un singleton
+ $stripeService = StripeService::getInstance();
+
+ // 1. Identifier les comptes sans Location
+ echo "📋 Recherche des comptes Stripe sans Location...\n\n";
+
+ $stmt = $db->query("
+ SELECT
+ sa.id,
+ sa.fk_entite,
+ sa.stripe_account_id,
+ sa.stripe_location_id,
+ e.encrypted_name,
+ e.adresse1,
+ e.adresse2,
+ e.code_postal,
+ e.ville
+ FROM stripe_accounts sa
+ INNER JOIN entites e ON sa.fk_entite = e.id
+ WHERE sa.stripe_account_id IS NOT NULL
+ AND (sa.stripe_location_id IS NULL OR sa.stripe_location_id = '')
+ AND e.chk_active = 1
+ ");
+
+ $accountsWithoutLocation = $stmt->fetchAll(PDO::FETCH_ASSOC);
+ $total = count($accountsWithoutLocation);
+
+ if ($total === 0) {
+ echo "✅ Aucun compte sans Location trouvé. Tous les comptes sont à jour !\n\n";
+ exit(0);
+ }
+
+ echo "ℹ️ Trouvé $total compte(s) sans Location :\n\n";
+
+ foreach ($accountsWithoutLocation as $account) {
+ $name = !empty($account['encrypted_name'])
+ ? ApiService::decryptData($account['encrypted_name'])
+ : 'Amicale #' . $account['fk_entite'];
+
+ echo " - Entité #{$account['fk_entite']} : $name\n";
+ echo " Stripe Account : {$account['stripe_account_id']}\n";
+ echo " Adresse : {$account['adresse1']}, {$account['code_postal']} {$account['ville']}\n\n";
+ }
+
+ // Demander confirmation
+ echo "⚠️ Voulez-vous créer les Locations manquantes ? (yes/no) : ";
+ $handle = fopen("php://stdin", "r");
+ $line = trim(fgets($handle));
+ fclose($handle);
+
+ if ($line !== 'yes') {
+ echo "❌ Opération annulée.\n\n";
+ exit(0);
+ }
+
+ echo "\n🚀 Création des Locations...\n\n";
+
+ // Initialiser Stripe avec la bonne clé selon le mode
+ $stripeConfig = $config->getStripeConfig();
+ $stripeMode = $stripeConfig['mode'] ?? 'test';
+ $stripeSecretKey = ($stripeMode === 'live')
+ ? $stripeConfig['secret_key_live']
+ : $stripeConfig['secret_key_test'];
+
+ \Stripe\Stripe::setApiKey($stripeSecretKey);
+ echo "ℹ️ Mode Stripe : " . strtoupper($stripeMode) . "\n\n";
+
+ $success = 0;
+ $errors = 0;
+
+ // 2. Créer les Locations manquantes
+ foreach ($accountsWithoutLocation as $account) {
+ $entiteId = $account['fk_entite'];
+ $stripeAccountId = $account['stripe_account_id'];
+
+ $name = !empty($account['encrypted_name'])
+ ? ApiService::decryptData($account['encrypted_name'])
+ : 'Amicale #' . $entiteId;
+
+ echo "🔧 Entité #{$entiteId} : $name\n";
+
+ try {
+ // Construire l'adresse
+ $adresse1 = !empty($account['adresse1']) ? $account['adresse1'] : 'Adresse non renseignée';
+ $ville = !empty($account['ville']) ? $account['ville'] : 'Ville';
+ $codePostal = !empty($account['code_postal']) ? $account['code_postal'] : '00000';
+
+ // Construire l'adresse pour Stripe (ne pas envoyer line2 si vide)
+ $addressData = [
+ 'line1' => $adresse1,
+ 'city' => $ville,
+ 'postal_code' => $codePostal,
+ 'country' => 'FR',
+ ];
+
+ // Ajouter line2 seulement s'il n'est pas vide
+ if (!empty($account['adresse2'])) {
+ $addressData['line2'] = $account['adresse2'];
+ }
+
+ // Créer la Location via Stripe API
+ $location = \Stripe\Terminal\Location::create([
+ 'display_name' => $name,
+ 'address' => $addressData,
+ 'metadata' => [
+ 'entite_id' => $entiteId,
+ 'type' => 'tap_to_pay',
+ 'created_by' => 'migration_script'
+ ]
+ ], [
+ 'stripe_account' => $stripeAccountId
+ ]);
+
+ $locationId = $location->id;
+
+ // Mettre à jour la base de données
+ $updateStmt = $db->prepare("
+ UPDATE stripe_accounts
+ SET stripe_location_id = :location_id,
+ updated_at = NOW()
+ WHERE id = :id
+ ");
+
+ $updateStmt->execute([
+ 'location_id' => $locationId,
+ 'id' => $account['id']
+ ]);
+
+ echo " ✅ Location créée : $locationId\n\n";
+ $success++;
+
+ } catch (\Stripe\Exception\ApiErrorException $e) {
+ echo " ❌ Erreur Stripe : " . $e->getMessage() . "\n\n";
+ $errors++;
+ } catch (Exception $e) {
+ echo " ❌ Erreur : " . $e->getMessage() . "\n\n";
+ $errors++;
+ }
+ }
+
+ // 3. Résumé
+ echo "\n";
+ echo "=============================================================================\n";
+ echo " Résumé de l'opération\n";
+ echo "=============================================================================\n";
+ echo "✅ Locations créées avec succès : $success\n";
+ echo "❌ Erreurs : $errors\n";
+ echo "📊 Total traité : $total\n";
+ echo "\n";
+
+ // 4. Vérification finale
+ echo "🔍 Vérification finale...\n";
+ $stmt = $db->query("
+ SELECT COUNT(*) as remaining
+ FROM stripe_accounts sa
+ WHERE sa.stripe_account_id IS NOT NULL
+ AND (sa.stripe_location_id IS NULL OR sa.stripe_location_id = '')
+ ");
+ $remaining = $stmt->fetch(PDO::FETCH_ASSOC);
+
+ echo " ℹ️ Comptes restants sans Location : " . $remaining['remaining'] . "\n\n";
+
+ if ($remaining['remaining'] == 0) {
+ echo "🎉 Tous les comptes Stripe ont maintenant une Location !\n\n";
+ }
+
+} catch (Exception $e) {
+ echo "\n";
+ echo "=============================================================================\n";
+ echo " ❌ ERREUR\n";
+ echo "=============================================================================\n";
+ echo "Message : " . $e->getMessage() . "\n";
+ echo "Fichier : " . $e->getFile() . ":" . $e->getLine() . "\n";
+ echo "\n";
+ exit(1);
+}
diff --git a/api/scripts/php/migrate_from_backup.php b/api/scripts/php/migrate_from_backup.php
new file mode 100755
index 00000000..c4c91477
--- /dev/null
+++ b/api/scripts/php/migrate_from_backup.php
@@ -0,0 +1,2047 @@
+#!/usr/bin/env php
+sourceDbName = $sourceDbName;
+ $this->targetDbName = $targetDbName;
+ $this->mode = $mode;
+ $this->entityId = $entityId;
+ $this->logFile = $logFile ?? '/var/back/migration_' . date('Ymd_His') . '.log';
+ $this->deleteBefore = $deleteBefore;
+
+ $this->log("=== Migration depuis backup PM7 ===");
+ $this->log("Source: {$sourceDbName}");
+ $this->log("Cible: {$targetDbName}");
+ $this->log("Mode: {$mode}");
+ if ($entityId) {
+ $this->log("Entité ID: {$entityId}");
+ }
+ if ($deleteBefore) {
+ $this->log("⚠️ Suppression des données existantes activée");
+ }
+ }
+
+ /**
+ * Connexion aux bases de données
+ */
+ public function connect() {
+ try {
+ // Connexion à la base source (backup restauré) - avec user ROOT pour accès multi-bases
+ $dsn = sprintf('mysql:host=%s;port=%d;dbname=%s;charset=utf8mb4',
+ self::DB_HOST, self::DB_PORT, $this->sourceDbName);
+ $this->sourceDb = new PDO($dsn, self::DB_USER_ROOT, self::DB_PASS_ROOT, [
+ PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION,
+ PDO::ATTR_DEFAULT_FETCH_MODE => PDO::FETCH_ASSOC,
+ PDO::ATTR_TIMEOUT => 600,
+ PDO::MYSQL_ATTR_INIT_COMMAND => "SET NAMES utf8mb4"
+ ]);
+ $this->log("✓ Connexion à la base source: {$this->sourceDbName} sur " . self::DB_HOST);
+
+ // Connexion à la base cible (pra_geo) - avec user dédié
+ $dsn = sprintf('mysql:host=%s;port=%d;dbname=%s;charset=utf8mb4',
+ self::DB_HOST, self::DB_PORT, $this->targetDbName);
+ $this->targetDb = new PDO($dsn, self::DB_USER, self::DB_PASS, [
+ PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION,
+ PDO::ATTR_DEFAULT_FETCH_MODE => PDO::FETCH_ASSOC,
+ PDO::ATTR_TIMEOUT => 600,
+ PDO::MYSQL_ATTR_INIT_COMMAND => "SET NAMES utf8mb4"
+ ]);
+ $this->log("✓ Connexion à la base cible: {$this->targetDbName} sur " . self::DB_HOST);
+
+ // Vérifier les versions MariaDB
+ $sourceVersion = $this->sourceDb->query("SELECT VERSION()")->fetchColumn();
+ $targetVersion = $this->targetDb->query("SELECT VERSION()")->fetchColumn();
+ $this->log(" Source version: $sourceVersion");
+ $this->log(" Cible version: $targetVersion");
+
+ return true;
+ } catch (PDOException $e) {
+ $this->log("✗ Erreur de connexion: " . $e->getMessage(), 'ERROR');
+ return false;
+ }
+ }
+
+ /**
+ * Suppression des données d'une entité dans la TARGET
+ * L'ordre de suppression respecte les contraintes FK (inverse de la migration)
+ */
+ private function deleteEntityData($entityId) {
+ try {
+ $this->targetDb->beginTransaction();
+
+ // Ordre de suppression inverse pour respecter les FK
+ $deletionOrder = [
+ 'medias',
+ 'ope_pass_histo',
+ 'ope_pass',
+ 'ope_users_sectors',
+ 'ope_users',
+ 'sectors_adresses',
+ 'ope_sectors',
+ 'operations',
+ 'users',
+ // NE PAS supprimer entites car peut avoir d'autres données liées
+ ];
+
+ foreach ($deletionOrder as $table) {
+ $deleted = 0;
+
+ switch ($table) {
+ case 'users':
+ $stmt = $this->targetDb->prepare("DELETE FROM $table WHERE fk_entite = ?");
+ $stmt->execute([$entityId]);
+ $deleted = $stmt->rowCount();
+ break;
+
+ case 'operations':
+ $stmt = $this->targetDb->prepare("DELETE FROM $table WHERE fk_entite = ?");
+ $stmt->execute([$entityId]);
+ $deleted = $stmt->rowCount();
+ break;
+
+ case 'sectors_adresses':
+ // Via ope_sectors -> operations
+ $stmt = $this->targetDb->prepare("
+ DELETE sa FROM $table sa
+ INNER JOIN ope_sectors s ON sa.fk_sector = s.id
+ INNER JOIN operations o ON s.fk_operation = o.id
+ WHERE o.fk_entite = ?
+ ");
+ $stmt->execute([$entityId]);
+ $deleted = $stmt->rowCount();
+ break;
+
+ case 'medias':
+ // Medias peut avoir fk_entite OU fk_operation
+ $stmt = $this->targetDb->prepare("
+ DELETE FROM $table
+ WHERE fk_entite = ?
+ OR fk_operation IN (SELECT id FROM operations WHERE fk_entite = ?)
+ ");
+ $stmt->execute([$entityId, $entityId]);
+ $deleted = $stmt->rowCount();
+ break;
+
+ case 'ope_pass_histo':
+ // Via ope_pass -> operations
+ $stmt = $this->targetDb->prepare("
+ DELETE h FROM $table h
+ INNER JOIN ope_pass p ON h.fk_pass = p.id
+ INNER JOIN operations o ON p.fk_operation = o.id
+ WHERE o.fk_entite = ?
+ ");
+ $stmt->execute([$entityId]);
+ $deleted = $stmt->rowCount();
+ break;
+
+ default:
+ // Tables avec fk_operation directe (ope_pass, ope_users, ope_users_sectors, ope_sectors)
+ $stmt = $this->targetDb->prepare("
+ DELETE t FROM $table t
+ INNER JOIN operations o ON t.fk_operation = o.id
+ WHERE o.fk_entite = ?
+ ");
+ $stmt->execute([$entityId]);
+ $deleted = $stmt->rowCount();
+ break;
+ }
+
+ if ($deleted > 0) {
+ $this->log(" ✓ $table: $deleted ligne(s) supprimée(s)");
+ }
+ }
+
+ $this->targetDb->commit();
+ $this->log("✓ Suppression terminée\n");
+
+ } catch (PDOException $e) {
+ if ($this->targetDb->inTransaction()) {
+ $this->targetDb->rollBack();
+ }
+ $this->log("✗ Erreur lors de la suppression: " . $e->getMessage(), 'ERROR');
+ throw $e;
+ }
+ }
+
+ /**
+ * Exécution de la migration
+ */
+ public function migrate() {
+ $startTime = microtime(true);
+ $totalRecords = 0;
+ $totalErrors = 0;
+
+ try {
+ if ($this->mode === 'global') {
+ // Migration globale de toutes les amicales
+ $this->log("\n=== Migration GLOBALE de toutes les amicales ===\n");
+
+ foreach ($this->migrationSteps as $step) {
+ $this->log("--- Migration: $step ---");
+ $result = $this->migrateTable($step);
+ $totalRecords += $result['success'];
+ $totalErrors += $result['errors'];
+ $this->log(" ✓ Réussis: {$result['success']}, Erreurs: {$result['errors']}");
+ }
+
+ } elseif ($this->mode === 'entity' && $this->entityId) {
+ // Migration d'une amicale spécifique
+ $this->log("\n=== Migration de l'amicale ID: {$this->entityId} ===\n");
+
+ // Vérifier que l'entité existe dans la source
+ $stmt = $this->sourceDb->prepare("SELECT rowid, libelle FROM users_entites WHERE rowid = ?");
+ $stmt->execute([$this->entityId]);
+ $entity = $stmt->fetch();
+
+ if (!$entity) {
+ $this->log("✗ Entité ID {$this->entityId} introuvable dans la source", 'ERROR');
+ return false;
+ }
+
+ $this->log("Entité trouvée: {$entity['libelle']}");
+
+ // Supprimer les données existantes si demandé
+ if ($this->deleteBefore) {
+ $this->log("\n⚠️ Suppression des données existantes de l'entité...");
+ $this->deleteEntityData($this->entityId);
+ }
+
+ // Migrer les données de l'entité
+ $entityTables = ['entites', 'users', 'operations', 'ope_sectors',
+ 'sectors_adresses', 'ope_users', 'ope_users_sectors',
+ 'ope_pass', 'ope_pass_histo', 'medias'];
+
+ // Compteurs pour le résumé
+ $stats = ['entites' => 0, 'users' => 0, 'operations' => 0, 'ope_sectors' => 0,
+ 'sectors_adresses' => 0, 'ope_users' => 0, 'ope_users_sectors' => 0,
+ 'ope_pass' => 0, 'ope_pass_histo' => 0, 'medias' => 0];
+
+ foreach ($entityTables as $table) {
+ $this->log("--- Migration: $table (entité {$this->entityId}) ---");
+ $result = $this->migrateTable($table, $this->entityId);
+ $totalRecords += $result['success'];
+ $totalErrors += $result['errors'];
+ $stats[$table] = $result['success'];
+ $this->log(" ✓ Réussis: {$result['success']}, Erreurs: {$result['errors']}");
+ }
+
+ // Ligne spéciale pour parsing par le script bash
+ // Note: sectorFromGPS sera défini dans la fonction migrateOpePass, donc on ne l'a pas ici
+ $this->log("#STATS# OPE:{$stats['operations']} USER:{$stats['users']} SECTOR:{$stats['ope_sectors']} PASS:{$stats['ope_pass']} #END#");
+ } else {
+ $this->log("✗ Mode invalide ou entity_id manquant", 'ERROR');
+ return false;
+ }
+
+ $duration = round(microtime(true) - $startTime, 2);
+
+ $this->log("\n=== Migration terminée ===");
+ $this->log("Durée totale: {$duration}s");
+ $this->log("Total enregistrements migrés: $totalRecords");
+ $this->log("Total erreurs: $totalErrors");
+
+ return true;
+
+ } catch (Exception $e) {
+ $this->log("✗ Erreur critique: " . $e->getMessage(), 'ERROR');
+ return false;
+ }
+ }
+
+ /**
+ * Migration d'une table spécifique
+ */
+ private function migrateTable($tableName, $entityId = null) {
+ try {
+ switch ($tableName) {
+ // Tables de référence (x_*)
+ case 'x_devises':
+ case 'x_entites_types':
+ case 'x_types_passages':
+ case 'x_types_reglements':
+ case 'x_users_roles':
+ case 'x_users_titres':
+ return $this->migrateReferenceTable($tableName, $entityId);
+
+ // Tables géographiques (avec FK vers x_devises, x_regions, x_pays)
+ case 'x_pays':
+ case 'x_regions':
+ case 'x_departements':
+ case 'x_villes':
+ return $this->migrateGeoTable($tableName, $entityId);
+
+ // Tables principales
+ case 'entites':
+ return $this->migrateEntites($entityId);
+
+ case 'users':
+ return $this->migrateUsers($entityId);
+
+ case 'operations':
+ return $this->migrateOperations($entityId);
+
+ case 'ope_sectors':
+ return $this->migrateOpeSectors($entityId);
+
+ case 'sectors_adresses':
+ return $this->migrateSectorsAdresses($entityId);
+
+ case 'ope_users':
+ return $this->migrateOpeUsers($entityId);
+
+ case 'ope_users_sectors':
+ return $this->migrateOpeUsersSectors($entityId);
+
+ case 'ope_pass':
+ return $this->migrateOpePass($entityId);
+
+ case 'ope_pass_histo':
+ return $this->migrateOpePassHisto($entityId);
+
+ case 'medias':
+ return $this->migrateMedias($entityId);
+
+ default:
+ $this->log(" ⚠ Table non gérée: $tableName", 'WARNING');
+ return ['success' => 0, 'errors' => 0];
+ }
+ } catch (Exception $e) {
+ $this->log(" ✗ Erreur lors de la migration de $tableName: " . $e->getMessage(), 'ERROR');
+ return ['success' => 0, 'errors' => 1];
+ }
+ }
+
+ /**
+ * Migration des tables de référence simples (sans FK)
+ */
+ private function migrateReferenceTable($tableName, $entityId = null) {
+ $success = 0;
+ $errors = 0;
+
+ try {
+ // Récupérer toutes les données de la source
+ $stmt = $this->sourceDb->query("SELECT * FROM $tableName");
+ $rows = $stmt->fetchAll();
+
+ if (empty($rows)) {
+ $this->log(" → Aucune donnée à migrer pour $tableName");
+ return ['success' => 0, 'errors' => 0];
+ }
+
+ // Préparer la requête d'insertion
+ $columns = array_keys($rows[0]);
+ $placeholders = array_map(fn($col) => ":$col", $columns);
+
+ $sql = sprintf(
+ "INSERT INTO %s (%s) VALUES (%s) ON DUPLICATE KEY UPDATE %s",
+ $tableName,
+ implode(', ', $columns),
+ implode(', ', $placeholders),
+ implode(', ', array_map(fn($col) => "$col = VALUES($col)", $columns))
+ );
+
+ $insertStmt = $this->targetDb->prepare($sql);
+
+ // Insérer chaque ligne
+ foreach ($rows as $row) {
+ try {
+ $insertStmt->execute($row);
+ $success++;
+ } catch (PDOException $e) {
+ $this->log(" ✗ Erreur insertion $tableName: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+ }
+
+ } catch (Exception $e) {
+ $this->log(" ✗ Erreur migration $tableName: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+
+ return ['success' => $success, 'errors' => $errors];
+ }
+
+ /**
+ * Compte le nombre de lignes dans une table TARGET pour une entité donnée
+ */
+ private function countTargetRows(string $tableName, $entityId = null): int {
+ try {
+ if ($entityId) {
+ // Pour entites : juste l'ID
+ if ($tableName === 'entites') {
+ $sql = "SELECT COUNT(*) as count FROM $tableName WHERE id = :entity_id";
+ $stmt = $this->targetDb->prepare($sql);
+ $stmt->execute(['entity_id' => $entityId]);
+ }
+ // Pour users : fk_entite
+ else if ($tableName === 'users') {
+ $sql = "SELECT COUNT(*) as count FROM $tableName WHERE fk_entite = :entity_id";
+ $stmt = $this->targetDb->prepare($sql);
+ $stmt->execute(['entity_id' => $entityId]);
+ }
+ // Pour operations : fk_entite directe
+ else if ($tableName === 'operations') {
+ $sql = "SELECT COUNT(*) as count FROM $tableName WHERE fk_entite = :entity_id";
+ $stmt = $this->targetDb->prepare($sql);
+ $stmt->execute(['entity_id' => $entityId]);
+ }
+ // Pour sectors_adresses : via ope_sectors
+ else if ($tableName === 'sectors_adresses') {
+ $sql = "SELECT COUNT(*) as count FROM $tableName sa
+ INNER JOIN ope_sectors s ON sa.fk_sector = s.id
+ INNER JOIN operations o ON s.fk_operation = o.id
+ WHERE o.fk_entite = :entity_id";
+ $stmt = $this->targetDb->prepare($sql);
+ $stmt->execute(['entity_id' => $entityId]);
+ }
+ // Pour ope_pass_histo : via ope_pass
+ else if ($tableName === 'ope_pass_histo') {
+ $sql = "SELECT COUNT(*) as count FROM $tableName h
+ INNER JOIN ope_pass p ON h.fk_pass = p.id
+ INNER JOIN operations o ON p.fk_operation = o.id
+ WHERE o.fk_entite = :entity_id";
+ $stmt = $this->targetDb->prepare($sql);
+ $stmt->execute(['entity_id' => $entityId]);
+ }
+ // Pour medias : peut avoir fk_entite OU fk_operation
+ else if ($tableName === 'medias') {
+ $sql = "SELECT COUNT(*) as count FROM $tableName
+ WHERE fk_entite = :entity_id
+ OR fk_operation IN (SELECT id FROM operations WHERE fk_entite = :entity_id2)";
+ $stmt = $this->targetDb->prepare($sql);
+ $stmt->execute(['entity_id' => $entityId, 'entity_id2' => $entityId]);
+ }
+ // Pour les tables avec fk_operation directe
+ else if (in_array($tableName, ['ope_sectors', 'ope_users', 'ope_users_sectors', 'ope_pass'])) {
+ $sql = "SELECT COUNT(*) as count FROM $tableName t
+ INNER JOIN operations o ON t.fk_operation = o.id
+ WHERE o.fk_entite = :entity_id";
+ $stmt = $this->targetDb->prepare($sql);
+ $stmt->execute(['entity_id' => $entityId]);
+ } else {
+ return 0;
+ }
+ } else {
+ $sql = "SELECT COUNT(*) as count FROM $tableName";
+ $stmt = $this->targetDb->prepare($sql);
+ $stmt->execute();
+ }
+
+ $result = $stmt->fetch(PDO::FETCH_ASSOC);
+ return (int)$result['count'];
+ } catch (PDOException $e) {
+ $this->log(" ⚠ Erreur comptage TARGET $tableName: " . $e->getMessage(), 'WARNING');
+ return 0;
+ }
+ }
+
+ /**
+ * Migration des tables géographiques (avec FK)
+ */
+ private function migrateGeoTable($tableName, $entityId = null) {
+ // Utiliser la même logique que les tables de référence
+ // car elles respectent déjà l'ordre des FK
+ return $this->migrateReferenceTable($tableName, $entityId);
+ }
+
+ /**
+ * Migration de la table entites (avec chiffrement)
+ */
+ private function migrateEntites($entityId = null) {
+ $success = 0;
+ $errors = 0;
+
+ try {
+ // Construire la requête de sélection
+ // Mapping: users_entites (source) → entites (cible)
+ $sql = "SELECT rowid, libelle, adresse1, adresse2, cp, ville,
+ fk_region, fk_type, tel1, tel2, email,
+ gps_lat, gps_lng, iban, bic,
+ demo, chk_mdp_manuel, chk_copie_mail_recu, chk_accept_sms,
+ date_modif, fk_user_modif, active
+ FROM users_entites";
+
+ if ($entityId) {
+ $sql .= " WHERE rowid = :entity_id";
+ }
+
+ $stmt = $this->sourceDb->prepare($sql);
+ if ($entityId) {
+ $stmt->execute(['entity_id' => $entityId]);
+ } else {
+ $stmt->execute();
+ }
+
+ $rows = $stmt->fetchAll();
+ $this->log(" → " . count($rows) . " entité(s) trouvée(s) dans SOURCE");
+
+ if (empty($rows)) {
+ $this->log(" → Aucune entité à migrer");
+ return ['success' => 0, 'errors' => 0];
+ }
+
+ // Préparer l'insertion
+ $insertSql = "INSERT INTO entites (
+ id, encrypted_name, adresse1, adresse2, code_postal, ville,
+ fk_region, fk_type, encrypted_phone, encrypted_mobile, encrypted_email,
+ gps_lat, gps_lng, chk_stripe, encrypted_stripe_id, encrypted_iban, encrypted_bic,
+ chk_demo, chk_mdp_manuel, chk_username_manuel, chk_user_delete_pass,
+ chk_copie_mail_recu, chk_accept_sms, chk_lot_actif,
+ created_at, fk_user_creat, updated_at, fk_user_modif, chk_active
+ ) VALUES (
+ :id, :encrypted_name, :adresse1, :adresse2, :code_postal, :ville,
+ :fk_region, :fk_type, :encrypted_phone, :encrypted_mobile, :encrypted_email,
+ :gps_lat, :gps_lng, :chk_stripe, :encrypted_stripe_id, :encrypted_iban, :encrypted_bic,
+ :chk_demo, :chk_mdp_manuel, :chk_username_manuel, :chk_user_delete_pass,
+ :chk_copie_mail_recu, :chk_accept_sms, :chk_lot_actif,
+ :created_at, :fk_user_creat, :updated_at, :fk_user_modif, :chk_active
+ ) ON DUPLICATE KEY UPDATE
+ encrypted_name = VALUES(encrypted_name),
+ adresse1 = VALUES(adresse1),
+ adresse2 = VALUES(adresse2),
+ code_postal = VALUES(code_postal),
+ ville = VALUES(ville),
+ fk_region = VALUES(fk_region),
+ fk_type = VALUES(fk_type),
+ encrypted_phone = VALUES(encrypted_phone),
+ encrypted_mobile = VALUES(encrypted_mobile),
+ encrypted_email = VALUES(encrypted_email),
+ updated_at = VALUES(updated_at),
+ fk_user_modif = VALUES(fk_user_modif)";
+
+ $insertStmt = $this->targetDb->prepare($insertSql);
+
+ // Insérer chaque entité
+ foreach ($rows as $row) {
+ try {
+ // Chiffrer les données sensibles
+ $encryptedName = ApiService::encryptSearchableData($row['libelle']);
+ $encryptedPhone = !empty($row['tel1']) ? ApiService::encryptData($row['tel1']) : '';
+ $encryptedMobile = !empty($row['tel2']) ? ApiService::encryptData($row['tel2']) : '';
+ $encryptedEmail = !empty($row['email']) ? ApiService::encryptSearchableData($row['email']) : '';
+ $encryptedIban = !empty($row['iban']) ? ApiService::encryptData($row['iban']) : '';
+ $encryptedBic = !empty($row['bic']) ? ApiService::encryptData($row['bic']) : '';
+
+ $insertStmt->execute([
+ 'id' => $row['rowid'],
+ 'encrypted_name' => $encryptedName,
+ 'adresse1' => $row['adresse1'] ?? '',
+ 'adresse2' => $row['adresse2'] ?? '',
+ 'code_postal' => $row['cp'] ?? '', // cp → code_postal
+ 'ville' => $row['ville'] ?? '',
+ 'fk_region' => $row['fk_region'],
+ 'fk_type' => $row['fk_type'] ?? 1,
+ 'encrypted_phone' => $encryptedPhone, // tel1
+ 'encrypted_mobile' => $encryptedMobile, // tel2
+ 'encrypted_email' => $encryptedEmail,
+ 'gps_lat' => $row['gps_lat'] ?? '',
+ 'gps_lng' => $row['gps_lng'] ?? '',
+ 'chk_stripe' => 0, // Pas dans source
+ 'encrypted_stripe_id' => '', // Pas dans source
+ 'encrypted_iban' => $encryptedIban,
+ 'encrypted_bic' => $encryptedBic,
+ 'chk_demo' => $row['demo'] ?? 0, // demo → chk_demo
+ 'chk_mdp_manuel' => $row['chk_mdp_manuel'] ?? 0,
+ 'chk_username_manuel' => 0, // Pas dans source
+ 'chk_user_delete_pass' => 0, // Pas dans source
+ 'chk_copie_mail_recu' => $row['chk_copie_mail_recu'] ?? 0,
+ 'chk_accept_sms' => $row['chk_accept_sms'] ?? 0,
+ 'chk_lot_actif' => 0, // Pas dans source
+ 'created_at' => null, // Pas date_creat dans source
+ 'fk_user_creat' => null, // Pas dans source
+ 'updated_at' => $row['date_modif'],
+ 'fk_user_modif' => $row['fk_user_modif'] ?? 0, // Peut être NULL
+ 'chk_active' => $row['active'] ?? 1
+ ]);
+
+ $success++;
+ } catch (PDOException $e) {
+ $this->log(" ✗ Erreur insertion entité {$row['rowid']}: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+ }
+
+ } catch (Exception $e) {
+ $this->log(" ✗ Erreur migration entites: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+
+ // Compter les lignes en TARGET après migration
+ $targetCount = $this->countTargetRows('entites', $entityId);
+ $sourceCount = count($rows ?? []);
+ $diff = $targetCount - $sourceCount;
+ $diffStr = $diff >= 0 ? "+$diff" : "$diff";
+
+ $this->log(" ✓ Entités migrées avec succès: $success, Erreurs: $errors");
+ $this->log(" 📊 SOURCE: $sourceCount → TARGET: $targetCount (différence: $diffStr)");
+ return ['success' => $success, 'errors' => $errors];
+ }
+
+ /**
+ * Migration de la table users (avec chiffrement et détection mobile)
+ */
+ private function migrateUsers($entityId = null) {
+ $success = 0;
+ $errors = 0;
+
+ try {
+ // Construire la requête de sélection
+ $sql = "SELECT u.rowid, u.fk_entite, u.fk_role, u.fk_titre,
+ u.libelle, u.prenom, u.nom_tournee, u.username, u.userpass,
+ u.telephone, u.mobile, u.email, u.alert_email, u.chk_suivi,
+ u.date_naissance, u.date_embauche,
+ u.date_creat, u.fk_user_creat, u.date_modif, u.fk_user_modif, u.active
+ FROM users u";
+
+ if ($entityId) {
+ $sql .= " WHERE u.fk_entite = :entity_id";
+ }
+
+ $stmt = $this->sourceDb->prepare($sql);
+ if ($entityId) {
+ $stmt->execute(['entity_id' => $entityId]);
+ } else {
+ $stmt->execute();
+ }
+
+ $rows = $stmt->fetchAll();
+ $this->log(" → " . count($rows) . " user(s) trouvé(s) dans SOURCE");
+
+ if (empty($rows)) {
+ $this->log(" → Aucun user à migrer");
+ return ['success' => 0, 'errors' => 0];
+ }
+
+ // Préparer l'insertion
+ $insertSql = "INSERT INTO users (
+ id, fk_entite, fk_role, fk_titre,
+ encrypted_name, first_name, sect_name, encrypted_user_name, user_pass_hash,
+ encrypted_phone, encrypted_mobile, encrypted_email, chk_alert_email, chk_suivi,
+ date_naissance, date_embauche,
+ created_at, fk_user_creat, updated_at, fk_user_modif, chk_active
+ ) VALUES (
+ :id, :fk_entite, :fk_role, :fk_titre,
+ :encrypted_name, :first_name, :sect_name, :encrypted_user_name, :user_pass_hash,
+ :encrypted_phone, :encrypted_mobile, :encrypted_email, :chk_alert_email, :chk_suivi,
+ :date_naissance, :date_embauche,
+ :created_at, :fk_user_creat, :updated_at, :fk_user_modif, :chk_active
+ ) ON DUPLICATE KEY UPDATE
+ fk_role = VALUES(fk_role),
+ encrypted_name = VALUES(encrypted_name),
+ first_name = VALUES(first_name),
+ encrypted_phone = VALUES(encrypted_phone),
+ encrypted_mobile = VALUES(encrypted_mobile),
+ encrypted_email = VALUES(encrypted_email),
+ updated_at = VALUES(updated_at),
+ fk_user_modif = VALUES(fk_user_modif)";
+
+ $insertStmt = $this->targetDb->prepare($insertSql);
+
+ // Insérer chaque user
+ foreach ($rows as $row) {
+ try {
+ // Chiffrer les données sensibles
+ $encryptedName = ApiService::encryptSearchableData($row['libelle']);
+ $encryptedUsername = !empty($row['username']) ? ApiService::encryptSearchableData($row['username']) : '';
+ $encryptedEmail = !empty($row['email']) ? ApiService::encryptSearchableData($row['email']) : '';
+
+ // Détection mobile vs fixe basée sur 06/07
+ $phone = $row['telephone'] ?? '';
+ $mobile = $row['mobile'] ?? '';
+
+ // Si phone commence par 06 ou 07, c'est un mobile
+ if (preg_match('/^0[67]/', $phone)) {
+ if (empty($mobile)) {
+ $mobile = $phone;
+ $phone = '';
+ }
+ }
+
+ $encryptedPhone = !empty($phone) ? ApiService::encryptData($phone) : '';
+ $encryptedMobile = !empty($mobile) ? ApiService::encryptData($mobile) : '';
+
+ $insertStmt->execute([
+ 'id' => $row['rowid'],
+ 'fk_entite' => $row['fk_entite'],
+ 'fk_role' => $row['fk_role'] ?? 1,
+ 'fk_titre' => $row['fk_titre'] ?? 1,
+ 'encrypted_name' => $encryptedName,
+ 'first_name' => $row['prenom'] ?? '',
+ 'sect_name' => $row['nom_tournee'] ?? '',
+ 'encrypted_user_name' => $encryptedUsername,
+ 'user_pass_hash' => $row['userpass'] ?? '',
+ 'encrypted_phone' => $encryptedPhone,
+ 'encrypted_mobile' => $encryptedMobile,
+ 'encrypted_email' => $encryptedEmail,
+ 'chk_alert_email' => $row['alert_email'] ?? 1,
+ 'chk_suivi' => $row['chk_suivi'] ?? 0,
+ 'date_naissance' => $row['date_naissance'],
+ 'date_embauche' => $row['date_embauche'],
+ 'created_at' => $row['date_creat'],
+ 'fk_user_creat' => $row['fk_user_creat'],
+ 'updated_at' => $row['date_modif'],
+ 'fk_user_modif' => $row['fk_user_modif'],
+ 'chk_active' => $row['active'] ?? 1
+ ]);
+
+ $success++;
+ } catch (PDOException $e) {
+ $this->log(" ✗ Erreur insertion user {$row['rowid']}: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+ }
+
+ } catch (Exception $e) {
+ $this->log(" ✗ Erreur migration users: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+
+ // Compter les lignes en TARGET après migration
+ $targetCount = $this->countTargetRows('users', $entityId);
+ $sourceCount = count($rows ?? []);
+ $diff = $targetCount - $sourceCount;
+ $diffStr = $diff >= 0 ? "+$diff" : "$diff";
+
+ $this->log(" ✓ Users migrés avec succès: $success, Erreurs: $errors");
+ $this->log(" 📊 SOURCE: $sourceCount → TARGET: $targetCount (différence: $diffStr)");
+ return ['success' => $success, 'errors' => $errors];
+ }
+
+ /**
+ * Migration de la table operations
+ */
+ private function migrateOperations($entityId = null) {
+ $success = 0;
+ $errors = 0;
+
+ try {
+ $sql = "SELECT rowid, fk_entite, libelle, date_deb, date_fin,
+ chk_distinct_sectors,
+ date_creat, fk_user_creat, date_modif, fk_user_modif, active
+ FROM operations";
+
+ if ($entityId) {
+ $sql .= " WHERE fk_entite = :entity_id";
+ }
+
+ // Ne migrer que les 3 dernières opérations (plus récentes)
+ $sql .= " ORDER BY rowid DESC LIMIT 3";
+
+ $stmt = $this->sourceDb->prepare($sql);
+ if ($entityId) {
+ $stmt->execute(['entity_id' => $entityId]);
+ } else {
+ $stmt->execute();
+ }
+
+ $rows = $stmt->fetchAll();
+ $this->log(" → " . count($rows) . " opération(s) trouvée(s) dans SOURCE");
+
+ if (empty($rows)) {
+ $this->log(" → Aucune opération à migrer");
+ return ['success' => 0, 'errors' => 0];
+ }
+
+ $insertSql = "INSERT INTO operations (
+ id, fk_entite, libelle, date_deb, date_fin, chk_distinct_sectors,
+ created_at, fk_user_creat, updated_at, fk_user_modif, chk_active
+ ) VALUES (
+ :id, :fk_entite, :libelle, :date_deb, :date_fin, :chk_distinct_sectors,
+ :created_at, :fk_user_creat, :updated_at, :fk_user_modif, :chk_active
+ ) ON DUPLICATE KEY UPDATE
+ libelle = VALUES(libelle),
+ date_deb = VALUES(date_deb),
+ date_fin = VALUES(date_fin),
+ updated_at = VALUES(updated_at)";
+
+ $insertStmt = $this->targetDb->prepare($insertSql);
+
+ foreach ($rows as $row) {
+ try {
+ $insertStmt->execute([
+ 'id' => $row['rowid'],
+ 'fk_entite' => $row['fk_entite'],
+ 'libelle' => $row['libelle'],
+ 'date_deb' => $row['date_deb'],
+ 'date_fin' => $row['date_fin'],
+ 'chk_distinct_sectors' => $row['chk_distinct_sectors'] ?? 0,
+ 'created_at' => $row['date_creat'],
+ 'fk_user_creat' => $row['fk_user_creat'] ?? 0, // 0 au lieu de NULL si vide
+ 'updated_at' => $row['date_modif'],
+ 'fk_user_modif' => $row['fk_user_modif'] ?? 0, // 0 au lieu de NULL si vide
+ 'chk_active' => $row['active'] ?? 1
+ ]);
+ $success++;
+ } catch (PDOException $e) {
+ $this->log(" ✗ Erreur insertion operation {$row['rowid']}: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+ }
+
+ } catch (Exception $e) {
+ $this->log(" ✗ Erreur migration operations: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+
+ // Compter les lignes en TARGET après migration
+ $targetCount = $this->countTargetRows('operations', $entityId);
+ $sourceCount = count($rows ?? []);
+ $diff = $targetCount - $sourceCount;
+ $diffStr = $diff >= 0 ? "+$diff" : "$diff";
+
+ $this->log(" ✓ Opérations migrées avec succès: $success, Erreurs: $errors");
+ $this->log(" 📊 SOURCE: $sourceCount → TARGET: $targetCount (différence: $diffStr)");
+ return ['success' => $success, 'errors' => $errors];
+ }
+
+ // Mapping des anciens secteurs vers les nouveaux
+ private $sectorMapping = [];
+
+ /**
+ * Récupère les IDs des 3 dernières opérations d'une entité
+ */
+ private function getLastOperationIds($entityId) {
+ $sql = "SELECT rowid FROM operations
+ WHERE fk_entite = ?
+ ORDER BY rowid DESC LIMIT 3";
+ $stmt = $this->sourceDb->prepare($sql);
+ $stmt->execute([$entityId]);
+ return array_column($stmt->fetchAll(), 'rowid');
+ }
+
+ /**
+ * Migration de la table ope_sectors
+ * IMPORTANT: La table source est "sectors" (ancien), la cible est "ope_sectors" (nouveau)
+ */
+ private function migrateOpeSectors($entityId = null) {
+ $success = 0;
+ $errors = 0;
+
+ try {
+ // Récupérer les secteurs utilisés depuis ope_users_sectors
+ // Car sectors seul ne contient pas le lien avec les opérations
+ // IMPORTANT: DISTINCT sur (fk_operation, s.rowid) car un secteur peut être réutilisé dans plusieurs opérations
+ $sql = "SELECT DISTINCT
+ ous.fk_operation,
+ s.rowid as old_sector_id,
+ s.libelle,
+ s.sector,
+ s.color,
+ s.date_creat,
+ s.fk_user_creat,
+ s.date_modif,
+ s.fk_user_modif,
+ s.active
+ FROM sectors s
+ INNER JOIN ope_users_sectors ous ON ous.fk_sector = s.rowid";
+
+ if ($entityId) {
+ // Récupérer les IDs des 3 dernières opérations
+ $operationIds = $this->getLastOperationIds($entityId);
+ if (empty($operationIds)) {
+ $this->log(" → Aucune opération trouvée pour l'entité");
+ return ['success' => 0, 'errors' => 0];
+ }
+ $operationIdsStr = implode(',', $operationIds);
+ $sql .= " INNER JOIN operations o ON ous.fk_operation = o.rowid
+ WHERE o.fk_entite = ? AND o.rowid IN ($operationIdsStr)";
+ }
+
+ $sql .= " ORDER BY ous.fk_operation, s.rowid";
+
+ $stmt = $this->sourceDb->prepare($sql);
+ if ($entityId) {
+ $stmt->execute([$entityId]);
+ } else {
+ $stmt->execute();
+ }
+
+ $rows = $stmt->fetchAll();
+ $this->log(" → " . count($rows) . " secteur(s) trouvé(s) dans SOURCE");
+
+ if (empty($rows)) {
+ $this->log(" → Aucun secteur à migrer");
+ return ['success' => 0, 'errors' => 0];
+ }
+
+ // Préparer l'insertion dans ope_sectors (NOUVEAU ID auto-increment)
+ $insertSql = "INSERT INTO ope_sectors (
+ fk_operation, fk_old_sector, libelle, sector, color,
+ created_at, fk_user_creat, updated_at, fk_user_modif, chk_active
+ ) VALUES (
+ :fk_operation, :fk_old_sector, :libelle, :sector, :color,
+ :created_at, :fk_user_creat, :updated_at, :fk_user_modif, :chk_active
+ ) ON DUPLICATE KEY UPDATE
+ libelle = VALUES(libelle),
+ sector = VALUES(sector),
+ color = VALUES(color),
+ updated_at = VALUES(updated_at)";
+
+ $insertStmt = $this->targetDb->prepare($insertSql);
+
+ foreach ($rows as $row) {
+ try {
+ $oldSectorId = $row['old_sector_id'];
+ $operationId = $row['fk_operation'];
+
+ // Vérifier si ce secteur existe déjà pour cette opération
+ $checkStmt = $this->targetDb->prepare("
+ SELECT id FROM ope_sectors
+ WHERE fk_operation = ? AND fk_old_sector = ?
+ ");
+ $checkStmt->execute([$operationId, $oldSectorId]);
+ $existing = $checkStmt->fetch();
+
+ if ($existing) {
+ // Déjà migré, utiliser l'ID existant
+ $newSectorId = $existing['id'];
+ $this->log(" → Secteur #{$oldSectorId} déjà migré (nouvel ID: {$newSectorId})");
+ } else {
+ // Insérer le nouveau secteur
+ $insertStmt->execute([
+ 'fk_operation' => $operationId,
+ 'fk_old_sector' => $oldSectorId,
+ 'libelle' => $row['libelle'],
+ 'sector' => $row['sector'],
+ 'color' => $row['color'] ?? '#4B77BE',
+ 'created_at' => $row['date_creat'],
+ 'fk_user_creat' => $row['fk_user_creat'] ?? 0, // 0 au lieu de NULL si vide
+ 'updated_at' => $row['date_modif'],
+ 'fk_user_modif' => $row['fk_user_modif'] ?? 0, // 0 au lieu de NULL si vide
+ 'chk_active' => $row['active'] ?? 1
+ ]);
+
+ // Récupérer le NOUVEAU ID auto-généré
+ $newSectorId = $this->targetDb->lastInsertId();
+ $this->log(" → Secteur #{$oldSectorId} migré (nouvel ID: {$newSectorId})");
+ }
+
+ // Stocker le mapping pour les tables suivantes
+ $mappingKey = $operationId . '_' . $oldSectorId;
+ $this->sectorMapping[$mappingKey] = $newSectorId;
+
+ $success++;
+ } catch (PDOException $e) {
+ $this->log(" ✗ Erreur insertion secteur {$row['old_sector_id']}: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+ }
+
+ $this->log(" → Mapping créé: " . count($this->sectorMapping) . " correspondances");
+
+ } catch (Exception $e) {
+ $this->log(" ✗ Erreur migration ope_sectors: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+
+ // Compter les lignes en TARGET après migration
+ $targetCount = $this->countTargetRows('ope_sectors', $entityId);
+ $sourceCount = count($rows ?? []);
+ $diff = $targetCount - $sourceCount;
+ $diffStr = $diff >= 0 ? "+$diff" : "$diff";
+
+ $this->log(" ✓ Secteurs migrés avec succès: $success, Erreurs: $errors");
+ $this->log(" 📊 SOURCE: $sourceCount → TARGET: $targetCount (différence: $diffStr)");
+ return ['success' => $success, 'errors' => $errors];
+ }
+
+ /**
+ * Migration de la table sectors_adresses
+ * IMPORTANT: Utilise le mapping créé par migrateOpeSectors()
+ */
+ private function migrateSectorsAdresses($entityId = null) {
+ $success = 0;
+ $errors = 0;
+ $skipped = 0;
+
+ try {
+ // Récupérer les adresses avec le lien vers sectors ET operations (via ope_users_sectors)
+ // NOTE: osm_id, osm_name, osm_date_creat n'existent PAS dans SOURCE, valeurs par défaut ajoutées
+ $sql = "SELECT sa.rowid, sa.fk_adresse, sa.fk_sector as old_sector_id,
+ sa.numero, sa.rue_bis, sa.rue, sa.cp, sa.ville,
+ sa.gps_lat, sa.gps_lng,
+ ous.fk_operation
+ FROM sectors_adresses sa
+ INNER JOIN ope_users_sectors ous ON ous.fk_sector = sa.fk_sector";
+
+ if ($entityId) {
+ $sql .= " INNER JOIN operations o ON ous.fk_operation = o.rowid
+ WHERE o.fk_entite = :entity_id";
+ }
+
+ $sql .= " GROUP BY sa.rowid, ous.fk_operation";
+
+ $stmt = $this->sourceDb->prepare($sql);
+ if ($entityId) {
+ $stmt->execute(['entity_id' => $entityId]);
+ } else {
+ $stmt->execute();
+ }
+
+ $rows = $stmt->fetchAll();
+ $this->log(" → " . count($rows) . " adresse(s) de secteur trouvée(s) dans SOURCE");
+
+ if (empty($rows)) {
+ $this->log(" → Aucune adresse de secteur à migrer");
+ return ['success' => 0, 'errors' => 0];
+ }
+
+ $insertSql = "INSERT INTO sectors_adresses (
+ id, fk_adresse, osm_id, fk_sector, osm_name,
+ numero, rue_bis, rue, cp, ville,
+ gps_lat, gps_lng, osm_date_creat,
+ created_at, updated_at
+ ) VALUES (
+ :id, :fk_adresse, :osm_id, :fk_sector, :osm_name,
+ :numero, :rue_bis, :rue, :cp, :ville,
+ :gps_lat, :gps_lng, :osm_date_creat,
+ :created_at, :updated_at
+ ) ON DUPLICATE KEY UPDATE
+ osm_name = VALUES(osm_name),
+ gps_lat = VALUES(gps_lat),
+ gps_lng = VALUES(gps_lng)";
+
+ $insertStmt = $this->targetDb->prepare($insertSql);
+
+ foreach ($rows as $row) {
+ try {
+ $oldSectorId = $row['old_sector_id'];
+ $operationId = $row['fk_operation'];
+
+ // Chercher le nouveau ID du secteur dans le mapping
+ $mappingKey = $operationId . '_' . $oldSectorId;
+
+ if (!isset($this->sectorMapping[$mappingKey])) {
+ $this->log(" ⚠ Adresse {$row['rowid']}: secteur {$oldSectorId} non trouvé dans mapping", 'WARNING');
+ $skipped++;
+ continue;
+ }
+
+ $newSectorId = $this->sectorMapping[$mappingKey];
+
+ $insertStmt->execute([
+ 'id' => $row['rowid'],
+ 'fk_adresse' => $row['fk_adresse'],
+ 'osm_id' => 0, // N'existe pas dans SOURCE
+ 'fk_sector' => $newSectorId, // NOUVEAU ID depuis mapping
+ 'osm_name' => '', // N'existe pas dans SOURCE
+ 'numero' => $row['numero'],
+ 'rue_bis' => $row['rue_bis'],
+ 'rue' => $row['rue'],
+ 'cp' => $row['cp'],
+ 'ville' => $row['ville'],
+ 'gps_lat' => $row['gps_lat'],
+ 'gps_lng' => $row['gps_lng'],
+ 'osm_date_creat' => null, // N'existe pas dans SOURCE
+ 'created_at' => null, // N'existe pas dans SOURCE
+ 'updated_at' => null // N'existe pas dans SOURCE
+ ]);
+ $success++;
+ } catch (PDOException $e) {
+ $this->log(" ✗ Erreur insertion sectors_adresses {$row['rowid']}: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+ }
+
+ if ($skipped > 0) {
+ $this->log(" → {$skipped} adresses ignorées (secteur non migré)");
+ }
+
+ } catch (Exception $e) {
+ $this->log(" ✗ Erreur migration sectors_adresses: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+
+ // Compter les lignes en TARGET après migration
+ $targetCount = $this->countTargetRows('sectors_adresses', $entityId);
+ $sourceCount = count($rows ?? []);
+ $diff = $targetCount - $sourceCount;
+ $diffStr = $diff >= 0 ? "+$diff" : "$diff";
+
+ $this->log(" ✓ Adresses de secteurs migrées avec succès: $success, Ignorées: $skipped, Erreurs: $errors");
+ $this->log(" 📊 SOURCE: $sourceCount → TARGET: $targetCount (différence: $diffStr)");
+ return ['success' => $success, 'errors' => $errors];
+ }
+
+ /**
+ * Migration de la table ope_users
+ */
+ private function migrateOpeUsers($entityId = null) {
+ $success = 0;
+ $errors = 0;
+
+ try {
+ // Joindre avec users pour récupérer fk_role, prenom, libelle, nom_tournee
+ // IMPORTANT: DISTINCT sur (fk_operation, fk_user) car il peut y avoir des doublons dans SOURCE
+ $sql = "SELECT DISTINCT ou.fk_operation, ou.fk_user,
+ u.fk_role, u.prenom, u.libelle, u.nom_tournee,
+ ou.date_creat, ou.fk_user_creat, ou.date_modif, ou.fk_user_modif, ou.active
+ FROM ope_users ou
+ INNER JOIN users u ON ou.fk_user = u.rowid";
+
+ if ($entityId) {
+ $operationIds = $this->getLastOperationIds($entityId);
+ if (empty($operationIds)) {
+ $this->log(" → Aucune opération trouvée pour l'entité");
+ return ['success' => 0, 'errors' => 0];
+ }
+ $operationIdsStr = implode(',', $operationIds);
+ $sql .= " WHERE ou.fk_operation IN ($operationIdsStr)";
+ }
+
+ $stmt = $this->sourceDb->prepare($sql);
+ $stmt->execute();
+
+ $rows = $stmt->fetchAll();
+ $this->log(" → " . count($rows) . " ope_user(s) trouvé(s) dans SOURCE");
+
+ if (empty($rows)) {
+ $this->log(" → Aucun ope_user à migrer");
+ return ['success' => 0, 'errors' => 0];
+ }
+
+ $insertSql = "INSERT INTO ope_users (
+ fk_operation, fk_user, fk_role,
+ first_name, encrypted_name, sect_name,
+ created_at, fk_user_creat, updated_at, fk_user_modif, chk_active
+ ) VALUES (
+ :fk_operation, :fk_user, :fk_role,
+ :first_name, :encrypted_name, :sect_name,
+ :created_at, :fk_user_creat, :updated_at, :fk_user_modif, :chk_active
+ ) ON DUPLICATE KEY UPDATE
+ updated_at = VALUES(updated_at)";
+
+ $insertStmt = $this->targetDb->prepare($insertSql);
+
+ foreach ($rows as $row) {
+ try {
+ // Vérifier que fk_user existe dans users de la TARGET
+ $checkUser = $this->targetDb->prepare("SELECT id FROM users WHERE id = ?");
+ $checkUser->execute([$row['fk_user']]);
+ if (!$checkUser->fetch()) {
+ // User n'existe pas, ignorer cet enregistrement
+ $this->log(" ⚠ OpeUser (op:{$row['fk_operation']}, user:{$row['fk_user']}): user non trouvé, ignoré", 'WARNING');
+ continue;
+ }
+
+ // Chiffrer les données sensibles depuis users
+ $encryptedName = !empty($row['libelle']) ? ApiService::encryptSearchableData($row['libelle']) : '';
+
+ $insertStmt->execute([
+ 'fk_operation' => $row['fk_operation'],
+ 'fk_user' => $row['fk_user'],
+ 'fk_role' => $row['fk_role'] ?? 1,
+ 'first_name' => $row['prenom'] ?? '',
+ 'encrypted_name' => $encryptedName,
+ 'sect_name' => $row['nom_tournee'] ?? '',
+ 'created_at' => $row['date_creat'],
+ 'fk_user_creat' => $row['fk_user_creat'],
+ 'updated_at' => $row['date_modif'],
+ 'fk_user_modif' => $row['fk_user_modif'],
+ 'chk_active' => $row['active'] ?? 1
+ ]);
+ $success++;
+ } catch (PDOException $e) {
+ $this->log(" ✗ Erreur insertion ope_user (op:{$row['fk_operation']}, user:{$row['fk_user']}): " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+ }
+
+ } catch (Exception $e) {
+ $this->log(" ✗ Erreur migration ope_users: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+
+ // Compter les lignes en TARGET après migration
+ $targetCount = $this->countTargetRows('ope_users', $entityId);
+ $sourceCount = count($rows ?? []);
+ $diff = $targetCount - $sourceCount;
+ $diffStr = $diff >= 0 ? "+$diff" : "$diff";
+
+ $this->log(" ✓ Ope_users migrés avec succès: $success, Erreurs: $errors");
+ $this->log(" 📊 SOURCE: $sourceCount → TARGET: $targetCount (différence: $diffStr)");
+ return ['success' => $success, 'errors' => $errors];
+ }
+
+ /**
+ * Migration de la table ope_users_sectors
+ * IMPORTANT: Utilise le mapping créé par migrateOpeSectors()
+ */
+ private function migrateOpeUsersSectors($entityId = null) {
+ $success = 0;
+ $errors = 0;
+ $skipped = 0;
+
+ try {
+ // NOTE: date_creat, fk_user_creat, date_modif, fk_user_modif n'existent PAS dans SOURCE
+ $sql = "SELECT ous.rowid, ous.fk_operation, ous.fk_user, ous.fk_sector as old_sector_id,
+ ous.active
+ FROM ope_users_sectors ous";
+
+ if ($entityId) {
+ $operationIds = $this->getLastOperationIds($entityId);
+ if (empty($operationIds)) {
+ $this->log(" → Aucune opération trouvée pour l'entité");
+ return ['success' => 0, 'errors' => 0];
+ }
+ $operationIdsStr = implode(',', $operationIds);
+ $sql .= " WHERE ous.fk_operation IN ($operationIdsStr)";
+ }
+
+ $stmt = $this->sourceDb->prepare($sql);
+ $stmt->execute();
+
+ $rows = $stmt->fetchAll();
+ $this->log(" → " . count($rows) . " ope_users_sector(s) trouvé(s) dans SOURCE");
+
+ if (empty($rows)) {
+ $this->log(" → Aucun ope_users_sector à migrer");
+ return ['success' => 0, 'errors' => 0];
+ }
+
+ $insertSql = "INSERT INTO ope_users_sectors (
+ id, fk_operation, fk_user, fk_sector,
+ created_at, fk_user_creat, updated_at, fk_user_modif, chk_active
+ ) VALUES (
+ :id, :fk_operation, :fk_user, :fk_sector,
+ :created_at, :fk_user_creat, :updated_at, :fk_user_modif, :chk_active
+ ) ON DUPLICATE KEY UPDATE
+ updated_at = VALUES(updated_at)";
+
+ $insertStmt = $this->targetDb->prepare($insertSql);
+
+ foreach ($rows as $row) {
+ try {
+ $oldSectorId = $row['old_sector_id'];
+ $operationId = $row['fk_operation'];
+
+ // Chercher le nouveau ID du secteur dans le mapping
+ $mappingKey = $operationId . '_' . $oldSectorId;
+
+ if (!isset($this->sectorMapping[$mappingKey])) {
+ $this->log(" ⚠ OpeUsersSector {$row['rowid']}: secteur {$oldSectorId} non trouvé dans mapping", 'WARNING');
+ $skipped++;
+ continue;
+ }
+
+ // Vérifier que fk_user existe dans users de la TARGET
+ $checkUser = $this->targetDb->prepare("SELECT id FROM users WHERE id = ?");
+ $checkUser->execute([$row['fk_user']]);
+ if (!$checkUser->fetch()) {
+ // User n'existe pas, ignorer cet enregistrement
+ $this->log(" ⚠ OpeUsersSector {$row['rowid']}: user {$row['fk_user']} non trouvé, ignoré", 'WARNING');
+ $skipped++;
+ continue;
+ }
+
+ $newSectorId = $this->sectorMapping[$mappingKey];
+
+ $insertStmt->execute([
+ 'id' => $row['rowid'],
+ 'fk_operation' => $operationId,
+ 'fk_user' => $row['fk_user'],
+ 'fk_sector' => $newSectorId, // NOUVEAU ID depuis mapping
+ 'created_at' => null, // N'existe pas dans SOURCE
+ 'fk_user_creat' => 0, // N'existe pas dans SOURCE - NOT NULL
+ 'updated_at' => null, // N'existe pas dans SOURCE
+ 'fk_user_modif' => 0, // N'existe pas dans SOURCE - NOT NULL
+ 'chk_active' => $row['active'] ?? 1
+ ]);
+ $success++;
+ } catch (PDOException $e) {
+ $this->log(" ✗ Erreur insertion ope_users_sector {$row['rowid']}: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+ }
+
+ if ($skipped > 0) {
+ $this->log(" → {$skipped} associations ignorées (secteur non migré)");
+ }
+
+ } catch (Exception $e) {
+ $this->log(" ✗ Erreur migration ope_users_sectors: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+
+ // Compter les lignes en TARGET après migration
+ $targetCount = $this->countTargetRows('ope_users_sectors', $entityId);
+ $sourceCount = count($rows ?? []);
+ $diff = $targetCount - $sourceCount;
+ $diffStr = $diff >= 0 ? "+$diff" : "$diff";
+
+ $this->log(" ✓ Ope_users_sectors migrés avec succès: $success, Ignorés: $skipped, Erreurs: $errors");
+ $this->log(" 📊 SOURCE: $sourceCount → TARGET: $targetCount (différence: $diffStr)");
+ return ['success' => $success, 'errors' => $errors];
+ }
+
+ /**
+ * Recherche un secteur par position GPS
+ * 1. Cherche si le point est contenu dans un secteur (ST_Contains)
+ * 2. Sinon, cherche le secteur le plus proche (ST_Distance)
+ * 3. Sinon, retourne le premier secteur de l'opération
+ */
+ private function findSectorByGPS($operationId, $lat, $lng) {
+ // Si GPS valide, TENTER une recherche géographique (mais ne pas bloquer si ça échoue)
+ if (!empty($lat) && !empty($lng) && $lat != 0 && $lng != 0) {
+ try {
+ // IMPORTANT : Le champ sector est stocké en BLOB, il faut le convertir avec ST_GeomFromWKB
+ // Construire le POINT avec CONCAT pour éviter le binding de paramètres
+
+ // 1. Chercher si le point est DANS un secteur
+ $sql = "SELECT id FROM ope_sectors
+ WHERE fk_operation = ?
+ AND ST_Contains(
+ ST_GeomFromWKB(sector),
+ ST_GeomFromText(CONCAT('POINT(', ?, ' ', ?, ')'), 4326)
+ )
+ LIMIT 1";
+
+ $stmt = $this->targetDb->prepare($sql);
+ $stmt->execute([$operationId, $lng, $lat]);
+ $result = $stmt->fetch();
+
+ if ($result) {
+ return $result['id'];
+ }
+
+ // 2. Sinon, chercher le secteur le PLUS PROCHE
+ $sql = "SELECT id,
+ ST_Distance(
+ ST_GeomFromWKB(sector),
+ ST_GeomFromText(CONCAT('POINT(', ?, ' ', ?, ')'), 4326)
+ ) as distance
+ FROM ope_sectors
+ WHERE fk_operation = ?
+ AND sector IS NOT NULL
+ ORDER BY distance ASC
+ LIMIT 1";
+
+ $stmt = $this->targetDb->prepare($sql);
+ $stmt->execute([$lng, $lat, $operationId]);
+ $result = $stmt->fetch();
+
+ if ($result) {
+ return $result['id'];
+ }
+ } catch (Exception $e) {
+ // La recherche GPS a échoué, on continue vers le fallback
+ // (Pas de log pour éviter de polluer avec des milliers de messages)
+ }
+ }
+
+ // 3. FALLBACK : Premier secteur de l'opération (TOUJOURS exécuté si GPS échoue)
+ try {
+ $sql = "SELECT id FROM ope_sectors
+ WHERE fk_operation = ?
+ LIMIT 1";
+
+ $stmt = $this->targetDb->prepare($sql);
+ $stmt->execute([$operationId]);
+ $result = $stmt->fetch();
+
+ return $result ? $result['id'] : null;
+ } catch (Exception $e) {
+ $this->log(" ⚠️ Erreur critique recherche secteur fallback: " . $e->getMessage(), 'ERROR');
+ return null;
+ }
+ }
+
+ /**
+ * Recherche un utilisateur valide pour un passage
+ * 1. Cherche un user affecté au secteur (via ope_users_sectors)
+ * 2. Sinon, cherche le premier user de l'opération (via ope_users)
+ */
+ private function findValidUser($operationId, $sectorId) {
+ try {
+ // 1. Chercher un user affecté au secteur
+ $sql = "SELECT fk_user FROM ope_users_sectors
+ WHERE fk_operation = :operation_id
+ AND fk_sector = :sector_id
+ LIMIT 1";
+
+ $stmt = $this->targetDb->prepare($sql);
+ $stmt->execute([
+ 'operation_id' => $operationId,
+ 'sector_id' => $sectorId
+ ]);
+ $result = $stmt->fetch();
+
+ if ($result) {
+ return $result['fk_user'];
+ }
+
+ // 2. FALLBACK : Premier user de l'opération
+ $sql = "SELECT fk_user FROM ope_users
+ WHERE fk_operation = :operation_id
+ LIMIT 1";
+
+ $stmt = $this->targetDb->prepare($sql);
+ $stmt->execute(['operation_id' => $operationId]);
+ $result = $stmt->fetch();
+
+ return $result ? $result['fk_user'] : null;
+
+ } catch (Exception $e) {
+ $this->log(" ⚠️ Erreur recherche user valide: " . $e->getMessage(), 'WARNING');
+ return null;
+ }
+ }
+
+ /**
+ * Migration de la table ope_pass (avec chiffrement)
+ */
+ private function migrateOpePass($entityId = null) {
+ $success = 0;
+ $errors = 0;
+
+ try {
+ $sql = "SELECT p.rowid, p.fk_operation, p.fk_sector, p.fk_user, p.fk_adresse,
+ p.date_eve, p.fk_type, p.numero, p.rue, p.rue_bis, p.ville, p.fk_habitat,
+ p.appt, p.niveau, p.gps_lat, p.gps_lng,
+ p.libelle, p.montant, p.fk_type_reglement, p.remarque, p.email,
+ p.recu, p.email_erreur,
+ p.chk_email_sent, p.phone, p.docremis, p.date_repasser,
+ p.nb_passages, p.chk_gps_maj, p.chk_map_create, p.chk_mobile,
+ p.chk_synchro, p.chk_api_adresse, p.chk_maj_adresse, p.anomalie,
+ p.date_creat, p.fk_user_creat, p.date_modif, p.fk_user_modif, p.active
+ FROM ope_pass p";
+
+ if ($entityId) {
+ $operationIds = $this->getLastOperationIds($entityId);
+ if (empty($operationIds)) {
+ $this->log(" → Aucune opération trouvée pour l'entité");
+ return ['success' => 0, 'errors' => 0];
+ }
+ $operationIdsStr = implode(',', $operationIds);
+ $sql .= " WHERE p.fk_operation IN ($operationIdsStr)";
+ }
+
+ $stmt = $this->sourceDb->prepare($sql);
+ $stmt->execute();
+
+ $rows = $stmt->fetchAll();
+ $this->log(" → " . count($rows) . " passage(s) trouvé(s) dans SOURCE");
+
+ if (empty($rows)) {
+ $this->log(" → Aucun passage à migrer");
+ return ['success' => 0, 'errors' => 0];
+ }
+
+ $insertSql = "INSERT INTO ope_pass (
+ id, fk_operation, fk_sector, fk_user, fk_adresse,
+ passed_at, fk_type, numero, rue, rue_bis, ville, fk_habitat,
+ appt, niveau, residence, gps_lat, gps_lng,
+ encrypted_name, montant, fk_type_reglement, remarque, encrypted_email,
+ nom_recu, date_recu, date_creat_recu, date_sent_recu, email_erreur,
+ chk_email_sent, encrypted_phone, docremis, date_repasser,
+ nb_passages, chk_gps_maj, chk_map_create, chk_mobile,
+ chk_synchro, chk_api_adresse, chk_maj_adresse, anomalie,
+ created_at, fk_user_creat, updated_at, fk_user_modif, chk_active
+ ) VALUES (
+ :id, :fk_operation, :fk_sector, :fk_user, :fk_adresse,
+ :passed_at, :fk_type, :numero, :rue, :rue_bis, :ville, :fk_habitat,
+ :appt, :niveau, :residence, :gps_lat, :gps_lng,
+ :encrypted_name, :montant, :fk_type_reglement, :remarque, :encrypted_email,
+ :nom_recu, :date_recu, :date_creat_recu, :date_sent_recu, :email_erreur,
+ :chk_email_sent, :encrypted_phone, :docremis, :date_repasser,
+ :nb_passages, :chk_gps_maj, :chk_map_create, :chk_mobile,
+ :chk_synchro, :chk_api_adresse, :chk_maj_adresse, :anomalie,
+ :created_at, :fk_user_creat, :updated_at, :fk_user_modif, :chk_active
+ ) ON DUPLICATE KEY UPDATE
+ encrypted_name = VALUES(encrypted_name),
+ montant = VALUES(montant),
+ encrypted_email = VALUES(encrypted_email),
+ encrypted_phone = VALUES(encrypted_phone),
+ updated_at = VALUES(updated_at)";
+
+ $insertStmt = $this->targetDb->prepare($insertSql);
+
+ // Traiter par lots de 1000
+ $batchSize = 1000;
+ $total = count($rows);
+ $batches = ceil($total / $batchSize);
+
+ // Compteurs détaillés
+ $sectorFromMapping = 0;
+ $sectorFromGPS = 0;
+ $sectorFallback = 0;
+ $userOriginal = 0;
+ $userReplaced = 0;
+
+ for ($i = 0; $i < $batches; $i++) {
+ $offset = $i * $batchSize;
+ $batch = array_slice($rows, $offset, $batchSize);
+
+ foreach ($batch as $row) {
+ try {
+ // ========== SECTEUR : Garantir un secteur valide ==========
+ $oldSectorId = $row['fk_sector'];
+ $operationId = $row['fk_operation'];
+ $mappingKey = $operationId . '_' . $oldSectorId;
+
+ $newSectorId = null;
+
+ // 1. Essayer avec le mapping
+ if (isset($this->sectorMapping[$mappingKey])) {
+ $newSectorId = $this->sectorMapping[$mappingKey];
+ $sectorFromMapping++;
+ }
+ // 2. Sinon, recherche géographique (avec fallback intégré)
+ else {
+ $newSectorId = $this->findSectorByGPS($operationId, $row['gps_lat'], $row['gps_lng']);
+ if ($newSectorId) {
+ // Si GPS valides, c'est une vraie recherche GPS
+ if (!empty($row['gps_lat']) && !empty($row['gps_lng'])) {
+ $sectorFromGPS++;
+ } else {
+ // Sinon, c'est le fallback (premier secteur)
+ $sectorFallback++;
+ }
+ }
+ }
+
+ // Si VRAIMENT aucun secteur (opération sans secteur ??)
+ if (!$newSectorId) {
+ $this->log(" ⚠️ Passage #{$row['rowid']} : opération {$operationId} sans secteur, IGNORÉ", 'WARNING');
+ $errors++;
+ continue;
+ }
+
+ // ========== UTILISATEUR : Garantir un user valide ==========
+ $fkUser = $row['fk_user'];
+
+ // Vérifier que fk_user existe dans users de la TARGET
+ if ($fkUser > 0) {
+ $checkUser = $this->targetDb->prepare("SELECT id FROM users WHERE id = ?");
+ $checkUser->execute([$fkUser]);
+ if ($checkUser->fetch()) {
+ // User valide
+ $userOriginal++;
+ } else {
+ // User n'existe pas, chercher un remplaçant
+ $replacementUser = $this->findValidUser($operationId, $newSectorId);
+ if ($replacementUser) {
+ $fkUser = $replacementUser;
+ $userReplaced++;
+ } else {
+ $this->log(" ⚠️ Passage #{$row['rowid']} : aucun user trouvé pour opération {$operationId}, IGNORÉ", 'WARNING');
+ $errors++;
+ continue;
+ }
+ }
+ } else {
+ // fk_user = 0 ou NULL, chercher un user valide
+ $replacementUser = $this->findValidUser($operationId, $newSectorId);
+ if ($replacementUser) {
+ $fkUser = $replacementUser;
+ $userReplaced++;
+ } else {
+ $this->log(" ⚠️ Passage #{$row['rowid']} : aucun user trouvé pour opération {$operationId}, IGNORÉ", 'WARNING');
+ $errors++;
+ continue;
+ }
+ }
+
+ // Chiffrer les données sensibles
+ $encryptedName = ApiService::encryptSearchableData($row['libelle']);
+ $encryptedEmail = !empty($row['email']) ? ApiService::encryptSearchableData($row['email']) : '';
+
+ // Détection mobile vs fixe
+ $phone = $row['phone'] ?? '';
+ $mobile = '';
+
+ if (preg_match('/^0[67]/', $phone)) {
+ $mobile = $phone;
+ $phone = '';
+ }
+
+ $encryptedPhone = !empty($mobile) ? ApiService::encryptData($mobile) : '';
+
+ // Vérification et correction du type de règlement (comme migrate_ope_pass.php)
+ $fkTypeReglement = $row['fk_type_reglement'] ?? 1;
+ if (!in_array($fkTypeReglement, [1, 2, 3])) {
+ $fkTypeReglement = 4; // Forcer à 4 ("-") si différent de 1, 2 ou 3
+ }
+
+ $insertStmt->execute([
+ 'id' => $row['rowid'],
+ 'fk_operation' => $operationId,
+ 'fk_sector' => $newSectorId, // NOUVEAU ID depuis mapping ou GPS
+ 'fk_user' => $fkUser, // User validé ou remplacé
+ 'fk_adresse' => $row['fk_adresse'],
+ 'passed_at' => $row['date_eve'],
+ 'fk_type' => $row['fk_type'],
+ 'numero' => $row['numero'],
+ 'rue' => $row['rue'],
+ 'rue_bis' => $row['rue_bis'],
+ 'ville' => $row['ville'],
+ 'fk_habitat' => $row['fk_habitat'] ?? 1,
+ 'appt' => $row['appt'],
+ 'niveau' => $row['niveau'],
+ 'residence' => '',
+ 'gps_lat' => $row['gps_lat'],
+ 'gps_lng' => $row['gps_lng'],
+ 'encrypted_name' => $encryptedName,
+ 'montant' => $row['montant'],
+ 'fk_type_reglement' => $fkTypeReglement,
+ 'remarque' => $row['remarque'],
+ 'encrypted_email' => $encryptedEmail,
+ 'nom_recu' => $row['recu'],
+ 'date_recu' => null,
+ 'date_creat_recu' => null,
+ 'date_sent_recu' => null,
+ 'email_erreur' => $row['email_erreur'],
+ 'chk_email_sent' => $row['chk_email_sent'] ?? 0,
+ 'encrypted_phone' => $encryptedPhone,
+ 'docremis' => $row['docremis'] ?? 0,
+ 'date_repasser' => $row['date_repasser'],
+ 'nb_passages' => $row['nb_passages'] ?? 1,
+ 'chk_gps_maj' => $row['chk_gps_maj'] ?? 0,
+ 'chk_map_create' => $row['chk_map_create'] ?? 0,
+ 'chk_mobile' => $row['chk_mobile'] ?? 0,
+ 'chk_synchro' => $row['chk_synchro'] ?? 1,
+ 'chk_api_adresse' => $row['chk_api_adresse'] ?? 0,
+ 'chk_maj_adresse' => $row['chk_maj_adresse'] ?? 0,
+ 'anomalie' => $row['anomalie'] ?? 0,
+ 'created_at' => $row['date_creat'],
+ 'fk_user_creat' => $row['fk_user_creat'],
+ 'updated_at' => $row['date_modif'],
+ 'fk_user_modif' => $row['fk_user_modif'],
+ 'chk_active' => $row['active'] ?? 1
+ ]);
+ $success++;
+ } catch (PDOException $e) {
+ $this->log(" ✗ Erreur insertion ope_pass {$row['rowid']}: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+ }
+
+ // Log progression
+ $processed = min(($i + 1) * $batchSize, $total);
+ $this->log(" → Progression: $processed / $total passages");
+ }
+
+ } catch (Exception $e) {
+ $this->log(" ✗ Erreur migration ope_pass: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+
+ // Compter les lignes en TARGET après migration
+ $targetCount = $this->countTargetRows('ope_pass', $entityId);
+ $sourceCount = count($rows ?? []);
+ $diff = $targetCount - $sourceCount;
+ $diffStr = $diff >= 0 ? "+$diff" : "$diff";
+
+ // Afficher les détails de récupération
+ if (isset($sectorFromMapping) && isset($sectorFromGPS) && isset($sectorFallback)) {
+ $this->log(" 🎯 Secteurs: mapping=$sectorFromMapping | GPS=$sectorFromGPS | fallback=$sectorFallback");
+ }
+ if (isset($userOriginal) && isset($userReplaced)) {
+ $this->log(" 👤 Users: originaux=$userOriginal | remplacés=$userReplaced");
+ }
+
+ $this->log(" ✓ Passages migrés avec succès: $success, Erreurs: $errors");
+ $this->log(" 📊 SOURCE: $sourceCount → TARGET: $targetCount (différence: $diffStr)");
+ return ['success' => $success, 'errors' => $errors];
+ }
+
+ /**
+ * Migration de la table ope_pass_histo
+ */
+ private function migrateOpePassHisto($entityId = null) {
+ $success = 0;
+ $errors = 0;
+
+ try {
+ $sql = "SELECT h.rowid, h.fk_pass, h.date_histo, h.sujet, h.remarque
+ FROM ope_pass_histo h";
+
+ if ($entityId) {
+ $operationIds = $this->getLastOperationIds($entityId);
+ if (empty($operationIds)) {
+ $this->log(" → Aucune opération trouvée pour l'entité");
+ return ['success' => 0, 'errors' => 0];
+ }
+ $operationIdsStr = implode(',', $operationIds);
+ $sql .= " INNER JOIN ope_pass p ON h.fk_pass = p.rowid
+ WHERE p.fk_operation IN ($operationIdsStr)";
+ }
+
+ $stmt = $this->sourceDb->prepare($sql);
+ $stmt->execute();
+
+ $rows = $stmt->fetchAll();
+ $this->log(" → " . count($rows) . " historique(s) de passage trouvé(s) dans SOURCE");
+
+ if (empty($rows)) {
+ $this->log(" → Aucun historique de passage à migrer");
+ return ['success' => 0, 'errors' => 0];
+ }
+
+ $insertSql = "INSERT INTO ope_pass_histo (
+ id, fk_pass, date_histo, sujet, remarque
+ ) VALUES (
+ :id, :fk_pass, :date_histo, :sujet, :remarque
+ ) ON DUPLICATE KEY UPDATE
+ sujet = VALUES(sujet),
+ remarque = VALUES(remarque)";
+
+ $insertStmt = $this->targetDb->prepare($insertSql);
+
+ foreach ($rows as $row) {
+ try {
+ $insertStmt->execute([
+ 'id' => $row['rowid'],
+ 'fk_pass' => $row['fk_pass'],
+ 'date_histo' => $row['date_histo'],
+ 'sujet' => $row['sujet'],
+ 'remarque' => $row['remarque']
+ ]);
+ $success++;
+ } catch (PDOException $e) {
+ $this->log(" ✗ Erreur insertion ope_pass_histo {$row['rowid']}: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+ }
+
+ } catch (Exception $e) {
+ $this->log(" ✗ Erreur migration ope_pass_histo: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+
+ // Compter les lignes en TARGET après migration
+ $targetCount = $this->countTargetRows('ope_pass_histo', $entityId);
+ $sourceCount = count($rows ?? []);
+ $diff = $targetCount - $sourceCount;
+ $diffStr = $diff >= 0 ? "+$diff" : "$diff";
+
+ $this->log(" ✓ Historiques de passages migrés avec succès: $success, Erreurs: $errors");
+ $this->log(" 📊 SOURCE: $sourceCount → TARGET: $targetCount (différence: $diffStr)");
+ return ['success' => $success, 'errors' => $errors];
+ }
+
+ /**
+ * Migration de la table medias
+ */
+ private function migrateMedias($entityId = null) {
+ $success = 0;
+ $errors = 0;
+
+ try {
+ // NOTE: SOURCE a support_rowid (pas support_id), type_fichier (pas file_type), hauteur/largeur (pas original_width/height)
+ // Beaucoup de champs n'existent pas dans SOURCE (fk_entite, fk_operation, file_category, etc.)
+ $sql = "SELECT m.rowid, m.support, m.support_rowid, m.fichier, m.type_fichier,
+ m.description, m.hauteur as original_height, m.largeur as original_width,
+ m.date_creat, m.fk_user_creat, m.date_modif, m.fk_user_modif
+ FROM medias m";
+
+ if ($entityId) {
+ $operationIds = $this->getLastOperationIds($entityId);
+ if (empty($operationIds)) {
+ $this->log(" → Aucune opération trouvée pour l'entité");
+ return ['success' => 0, 'errors' => 0];
+ }
+ $operationIdsStr = implode(',', $operationIds);
+ // Filtrer les médias liés aux ope_pass des 3 dernières opérations
+ $sql .= " WHERE (m.support = 'ope_pass' AND m.support_rowid IN (
+ SELECT rowid FROM ope_pass WHERE fk_operation IN ($operationIdsStr)
+ ))";
+ }
+
+ $stmt = $this->sourceDb->prepare($sql);
+ $stmt->execute();
+
+ $rows = $stmt->fetchAll();
+ $this->log(" → " . count($rows) . " media(s) trouvé(s) dans SOURCE");
+
+ if (empty($rows)) {
+ $this->log(" → Aucun media à migrer");
+ return ['success' => 0, 'errors' => 0];
+ }
+
+ $insertSql = "INSERT INTO medias (
+ id, support, support_id, fichier, file_type, file_category,
+ file_size, mime_type, original_name, fk_entite, fk_operation,
+ file_path, original_width, original_height, processed_width, processed_height,
+ is_processed, description,
+ created_at, fk_user_creat, updated_at, fk_user_modif
+ ) VALUES (
+ :id, :support, :support_id, :fichier, :file_type, :file_category,
+ :file_size, :mime_type, :original_name, :fk_entite, :fk_operation,
+ :file_path, :original_width, :original_height, :processed_width, :processed_height,
+ :is_processed, :description,
+ :created_at, :fk_user_creat, :updated_at, :fk_user_modif
+ ) ON DUPLICATE KEY UPDATE
+ file_path = VALUES(file_path),
+ updated_at = VALUES(updated_at)";
+
+ $insertStmt = $this->targetDb->prepare($insertSql);
+
+ foreach ($rows as $row) {
+ try {
+ $insertStmt->execute([
+ 'id' => $row['rowid'],
+ 'support' => $row['support'],
+ 'support_id' => $row['support_rowid'], // SOURCE utilise support_rowid
+ 'fichier' => $row['fichier'],
+ 'file_type' => $row['type_fichier'], // SOURCE utilise type_fichier
+ 'file_category' => null, // N'existe pas dans SOURCE
+ 'file_size' => null, // N'existe pas dans SOURCE
+ 'mime_type' => null, // N'existe pas dans SOURCE
+ 'original_name' => null, // N'existe pas dans SOURCE
+ 'fk_entite' => null, // N'existe pas dans SOURCE
+ 'fk_operation' => null, // N'existe pas dans SOURCE
+ 'file_path' => null, // N'existe pas dans SOURCE
+ 'original_width' => $row['original_width'], // Mappé depuis largeur
+ 'original_height' => $row['original_height'], // Mappé depuis hauteur
+ 'processed_width' => null, // N'existe pas dans SOURCE
+ 'processed_height' => null, // N'existe pas dans SOURCE
+ 'is_processed' => 0, // N'existe pas dans SOURCE
+ 'description' => $row['description'],
+ 'created_at' => $row['date_creat'],
+ 'fk_user_creat' => $row['fk_user_creat'],
+ 'updated_at' => $row['date_modif'],
+ 'fk_user_modif' => $row['fk_user_modif']
+ ]);
+ $success++;
+ } catch (PDOException $e) {
+ $this->log(" ✗ Erreur insertion media {$row['rowid']}: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+ }
+
+ } catch (Exception $e) {
+ $this->log(" ✗ Erreur migration medias: " . $e->getMessage(), 'ERROR');
+ $errors++;
+ }
+
+ // Compter les lignes en TARGET après migration
+ $targetCount = $this->countTargetRows('medias', $entityId);
+ $sourceCount = count($rows ?? []);
+ $diff = $targetCount - $sourceCount;
+ $diffStr = $diff >= 0 ? "+$diff" : "$diff";
+
+ $this->log(" ✓ Medias migrés avec succès: $success, Erreurs: $errors");
+ $this->log(" 📊 SOURCE: $sourceCount → TARGET: $targetCount (différence: $diffStr)");
+ return ['success' => $success, 'errors' => $errors];
+ }
+
+ /**
+ * Logging
+ */
+ private function log($message, $level = 'INFO') {
+ $timestamp = date('Y-m-d H:i:s');
+ $logMessage = "[{$timestamp}] [{$level}] {$message}\n";
+
+ // Afficher dans la console
+ echo $logMessage;
+
+ // Écrire dans le fichier de log
+ file_put_contents($this->logFile, $logMessage, FILE_APPEND);
+ }
+}
+
+// === Fonctions globales pour compatibilité avec les scripts existants ===
+
+/**
+ * Retourne la connexion à la base source (remplace celle de config.php)
+ */
+function getSourceConnection() {
+ return $GLOBALS['__MIGRATION_SOURCE_DB'] ?? null;
+}
+
+/**
+ * Retourne la connexion à la base cible (remplace celle de config.php)
+ */
+function getTargetConnection() {
+ return $GLOBALS['__MIGRATION_TARGET_DB'] ?? null;
+}
+
+/**
+ * Pas de tunnel SSH nécessaire (bases locales)
+ */
+function createSshTunnel() {
+ // Rien à faire, les bases sont locales
+ return true;
+}
+
+/**
+ * Pas de tunnel SSH à fermer
+ */
+function closeSshTunnel() {
+ // Rien à faire
+}
+
+/**
+ * Logging simple
+ */
+function logOperation($message, $level = 'INFO') {
+ echo "[{$level}] {$message}\n";
+}
+
+// === Parse des arguments CLI ===
+
+function parseArguments($argv) {
+ $args = [
+ 'source-db' => null,
+ 'target-db' => 'pra_geo',
+ 'mode' => 'global',
+ 'entity-id' => null,
+ 'log' => null,
+ 'delete-before' => true,
+ 'help' => false
+ ];
+
+ foreach ($argv as $arg) {
+ if (strpos($arg, '--') === 0) {
+ $parts = explode('=', substr($arg, 2), 2);
+ $key = $parts[0];
+ $value = $parts[1] ?? true;
+
+ if (array_key_exists($key, $args)) {
+ $args[$key] = $value;
+ }
+ }
+ }
+
+ return $args;
+}
+
+function showHelp() {
+ echo <<connect()) {
+ exit(1);
+}
+
+$success = $migration->migrate();
+
+exit($success ? 0 : 1);
diff --git a/api/scripts/php/migrate_from_backup_verbose.php b/api/scripts/php/migrate_from_backup_verbose.php
new file mode 100755
index 00000000..5c3cc687
--- /dev/null
+++ b/api/scripts/php/migrate_from_backup_verbose.php
@@ -0,0 +1,543 @@
+#!/usr/bin/env php
+ ['source' => 0, 'migrated' => 0],
+ 'users' => ['source' => 0, 'migrated' => 0],
+ 'operations' => ['source' => 0, 'migrated' => 0],
+ 'ope_sectors' => ['source' => 0, 'migrated' => 0],
+ 'sectors_adresses' => ['source' => 0, 'migrated' => 0],
+ 'ope_users' => ['source' => 0, 'migrated' => 0],
+ 'ope_users_sectors' => ['source' => 0, 'migrated' => 0],
+ 'ope_pass' => ['source' => 0, 'migrated' => 0],
+ 'ope_pass_histo' => ['source' => 0, 'migrated' => 0],
+ 'medias' => ['source' => 0, 'migrated' => 0],
+];
+
+// Fonctions utilitaires
+function println($message, $color = C_RESET) {
+ echo $color . $message . C_RESET . "\n";
+}
+
+function printBox($title, $color = C_BLUE) {
+ $width = 70;
+ $titleLen = strlen($title);
+ $padding = ($width - $titleLen - 2) / 2;
+
+ println(str_repeat("═", $width), $color);
+ println(str_repeat(" ", floor($padding)) . $title . str_repeat(" ", ceil($padding)), $color);
+ println(str_repeat("═", $width), $color);
+}
+
+function printStep($step, $substep = null) {
+ if ($substep) {
+ println(" ├─ " . $substep, C_CYAN);
+ } else {
+ println("\n" . C_BOLD . "▶ " . $step . C_RESET);
+ }
+}
+
+function printStat($label, $source, $migrated, $indent = " ") {
+ $status = ($source === $migrated) ? C_GREEN . "✓" : C_YELLOW . "⚠";
+ println($indent . "📊 {$label}: {$source} source → {$migrated} migré(s) {$status}" . C_RESET);
+}
+
+function connectDatabases($sourceDbName, $targetDbName) {
+ global $sourceDb, $targetDb;
+
+ printStep("Connexion aux bases de données");
+
+ try {
+ // Base source
+ $dsn = sprintf('mysql:host=%s;port=%d;dbname=%s;charset=utf8mb4',
+ DB_HOST, DB_PORT, $sourceDbName);
+ $sourceDb = new PDO($dsn, DB_USER_ROOT, DB_PASS_ROOT, [
+ PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION,
+ PDO::ATTR_DEFAULT_FETCH_MODE => PDO::FETCH_ASSOC,
+ ]);
+ printStep("Source connectée: {$sourceDbName}", true);
+
+ // Base cible
+ $dsn = sprintf('mysql:host=%s;port=%d;dbname=%s;charset=utf8mb4',
+ DB_HOST, DB_PORT, $targetDbName);
+ $targetDb = new PDO($dsn, DB_USER, DB_PASS, [
+ PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION,
+ PDO::ATTR_DEFAULT_FETCH_MODE => PDO::FETCH_ASSOC,
+ ]);
+ printStep("Cible connectée: {$targetDbName}", true);
+
+ return true;
+ } catch (PDOException $e) {
+ println("✗ Erreur connexion: " . $e->getMessage(), C_RED);
+ return false;
+ }
+}
+
+function getEntityInfo($entityId) {
+ global $sourceDb;
+
+ $stmt = $sourceDb->prepare("
+ SELECT rowid, libelle, cp, ville
+ FROM users_entites
+ WHERE rowid = ?
+ ");
+ $stmt->execute([$entityId]);
+ return $stmt->fetch();
+}
+
+function migrateReferenceTable($tableName) {
+ global $sourceDb, $targetDb;
+
+ printStep("Migration table: {$tableName}");
+
+ // Compter source
+ $count = $sourceDb->query("SELECT COUNT(*) FROM {$tableName}")->fetchColumn();
+ printStep("Source: {$count} enregistrements", true);
+
+ if ($count === 0) {
+ printStep("Aucune donnée à migrer", true);
+ return 0;
+ }
+
+ // Récupérer les données
+ $rows = $sourceDb->query("SELECT * FROM {$tableName}")->fetchAll();
+
+ // Préparer l'insertion
+ $columns = array_keys($rows[0]);
+ $placeholders = array_map(fn($col) => ":{$col}", $columns);
+
+ $sql = sprintf(
+ "INSERT INTO %s (%s) VALUES (%s) ON DUPLICATE KEY UPDATE %s",
+ $tableName,
+ implode(', ', $columns),
+ implode(', ', $placeholders),
+ implode(', ', array_map(fn($col) => "{$col} = VALUES({$col})", $columns))
+ );
+
+ $stmt = $targetDb->prepare($sql);
+
+ $success = 0;
+ foreach ($rows as $row) {
+ try {
+ $stmt->execute($row);
+ $success++;
+ } catch (PDOException $e) {
+ // Ignorer erreurs
+ }
+ }
+
+ printStep("Migré: {$success}/{$count}", true);
+ return $success;
+}
+
+function migrateEntite($entityId) {
+ global $sourceDb, $targetDb, $stats;
+
+ printStep("ÉTAPE 1: Migration de l'entité #{$entityId}");
+
+ // Récupérer l'entité source
+ $stmt = $sourceDb->prepare("
+ SELECT * FROM users_entites WHERE rowid = ?
+ ");
+ $stmt->execute([$entityId]);
+ $entity = $stmt->fetch();
+
+ if (!$entity) {
+ println(" ✗ Entité introuvable", C_RED);
+ return false;
+ }
+
+ $stats['entites']['source'] = 1;
+
+ println(" 📋 Entité: " . $entity['libelle']);
+ println(" 📍 Code postal: " . ($entity['cp'] ?? 'N/A'));
+ println(" 🏙️ Ville: " . ($entity['ville'] ?? 'N/A'));
+
+ // Chiffrer les données
+ $encryptedName = ApiService::encryptSearchableData($entity['libelle']);
+ $encryptedEmail = !empty($entity['email']) ? ApiService::encryptSearchableData($entity['email']) : '';
+ $encryptedPhone = !empty($entity['phone']) ? ApiService::encryptData($entity['phone']) : '';
+ $encryptedMobile = !empty($entity['mobile']) ? ApiService::encryptData($entity['mobile']) : '';
+
+ // Insérer dans la cible
+ $sql = "INSERT INTO entites (
+ id, encrypted_name, code_postal, ville, encrypted_email, encrypted_phone, encrypted_mobile,
+ fk_region, fk_type, chk_active, created_at, updated_at
+ ) VALUES (
+ :id, :name, :cp, :ville, :email, :phone, :mobile,
+ :region, :type, :active, :created, :updated
+ ) ON DUPLICATE KEY UPDATE
+ encrypted_name = VALUES(encrypted_name),
+ code_postal = VALUES(code_postal),
+ ville = VALUES(ville)";
+
+ $stmt = $targetDb->prepare($sql);
+ $stmt->execute([
+ 'id' => $entity['rowid'],
+ 'name' => $encryptedName,
+ 'cp' => $entity['cp'] ?? '',
+ 'ville' => $entity['ville'] ?? '',
+ 'email' => $encryptedEmail,
+ 'phone' => $encryptedPhone,
+ 'mobile' => $encryptedMobile,
+ 'region' => $entity['fk_region'] ?? 1,
+ 'type' => $entity['fk_type'] ?? 1,
+ 'active' => $entity['active'] ?? 1,
+ 'created' => $entity['date_creat'],
+ 'updated' => $entity['date_modif']
+ ]);
+
+ $stats['entites']['migrated'] = 1;
+
+ printStat("Entité", 1, 1);
+
+ return true;
+}
+
+function migrateUsers($entityId) {
+ global $sourceDb, $targetDb, $stats;
+
+ printStep("ÉTAPE 2: Migration des utilisateurs");
+
+ // Compter source
+ $count = $sourceDb->prepare("SELECT COUNT(*) FROM users WHERE fk_entite = ? AND active = 1");
+ $count->execute([$entityId]);
+ $sourceCount = $count->fetchColumn();
+
+ $stats['users']['source'] = $sourceCount;
+ println(" 📊 Source: {$sourceCount} utilisateurs actifs");
+
+ if ($sourceCount === 0) {
+ println(" ⚠️ Aucun utilisateur à migrer", C_YELLOW);
+ return 0;
+ }
+
+ // Récupérer les users
+ $stmt = $sourceDb->prepare("
+ SELECT * FROM users WHERE fk_entite = ? AND active = 1
+ ");
+ $stmt->execute([$entityId]);
+ $users = $stmt->fetchAll();
+
+ $success = 0;
+ foreach ($users as $user) {
+ try {
+ $encryptedName = ApiService::encryptSearchableData($user['nom']);
+ $encryptedUsername = !empty($user['username']) ? ApiService::encryptSearchableData($user['username']) : '';
+ $encryptedEmail = !empty($user['email']) ? ApiService::encryptSearchableData($user['email']) : '';
+ $encryptedPhone = !empty($user['telephone']) ? ApiService::encryptData($user['telephone']) : '';
+ $encryptedMobile = !empty($user['mobile']) ? ApiService::encryptData($user['mobile']) : '';
+
+ $sql = "INSERT INTO users (
+ id, fk_entite, fk_role, encrypted_name, first_name,
+ encrypted_user_name, user_pass_hash, encrypted_email,
+ encrypted_phone, encrypted_mobile, chk_active, created_at, updated_at
+ ) VALUES (
+ :id, :entity, :role, :name, :firstname,
+ :username, :pass, :email,
+ :phone, :mobile, :active, :created, :updated
+ ) ON DUPLICATE KEY UPDATE
+ encrypted_name = VALUES(encrypted_name),
+ encrypted_email = VALUES(encrypted_email)";
+
+ $stmt = $targetDb->prepare($sql);
+ $stmt->execute([
+ 'id' => $user['rowid'],
+ 'entity' => $entityId,
+ 'role' => $user['fk_role'] ?? 1,
+ 'name' => $encryptedName,
+ 'firstname' => $user['prenom'] ?? '',
+ 'username' => $encryptedUsername,
+ 'pass' => $user['password'] ?? '',
+ 'email' => $encryptedEmail,
+ 'phone' => $encryptedPhone,
+ 'mobile' => $encryptedMobile,
+ 'active' => 1,
+ 'created' => $user['date_creat'],
+ 'updated' => $user['date_modif']
+ ]);
+
+ $success++;
+ } catch (PDOException $e) {
+ // Ignorer
+ }
+ }
+
+ $stats['users']['migrated'] = $success;
+ printStat("Utilisateurs", $sourceCount, $success);
+
+ return $success;
+}
+
+function migrateOperations($entityId, $limit = 3) {
+ global $sourceDb, $targetDb, $stats;
+
+ printStep("ÉTAPE 3: Migration des opérations (limite: {$limit})");
+
+ // Compter toutes les opérations
+ $count = $sourceDb->prepare("SELECT COUNT(*) FROM operations WHERE fk_entite = ? AND active = 1");
+ $count->execute([$entityId]);
+ $totalCount = $count->fetchColumn();
+
+ println(" 📊 Total disponible: {$totalCount} opérations");
+ println(" 🎯 Limitation: {$limit} dernières opérations");
+
+ $stats['operations']['source'] = min($limit, $totalCount);
+
+ // Récupérer les N dernières opérations
+ $stmt = $sourceDb->prepare("
+ SELECT * FROM operations
+ WHERE fk_entite = ? AND active = 1
+ ORDER BY date_creat DESC
+ LIMIT ?
+ ");
+ $stmt->execute([$entityId, $limit]);
+ $operations = $stmt->fetchAll();
+
+ if (empty($operations)) {
+ println(" ⚠️ Aucune opération à migrer", C_YELLOW);
+ return [];
+ }
+
+ $migratedOps = [];
+ foreach ($operations as $op) {
+ try {
+ $sql = "INSERT INTO operations (
+ id, fk_entite, libelle, date_deb, date_fin,
+ chk_distinct_sectors, chk_active, created_at, updated_at
+ ) VALUES (
+ :id, :entity, :libelle, :datedeb, :datefin,
+ :distinct, :active, :created, :updated
+ ) ON DUPLICATE KEY UPDATE
+ libelle = VALUES(libelle)";
+
+ $stmt = $targetDb->prepare($sql);
+ $stmt->execute([
+ 'id' => $op['rowid'],
+ 'entity' => $entityId,
+ 'libelle' => $op['libelle'],
+ 'datedeb' => $op['date_deb'],
+ 'datefin' => $op['date_fin'],
+ 'distinct' => $op['chk_distinct_sectors'] ?? 0,
+ 'active' => 1,
+ 'created' => $op['date_creat'],
+ 'updated' => $op['date_modif']
+ ]);
+
+ $migratedOps[] = $op['rowid'];
+ $stats['operations']['migrated']++;
+
+ println(" ├─ Opération #{$op['rowid']}: " . $op['libelle'], C_GREEN);
+ } catch (PDOException $e) {
+ println(" ├─ ✗ Erreur opération #{$op['rowid']}: " . $e->getMessage(), C_RED);
+ }
+ }
+
+ printStat("Opérations", count($operations), count($migratedOps));
+
+ return $migratedOps;
+}
+
+function migrateOperationDetails($operationId, $entityId) {
+ global $sourceDb, $targetDb, $stats;
+
+ println("\n " . C_BOLD . "┌─ Détails opération #{$operationId}" . C_RESET);
+
+ // 1. Compter les passages
+ $passCount = $sourceDb->prepare("SELECT COUNT(*) FROM ope_pass WHERE fk_operation = ?");
+ $passCount->execute([$operationId]);
+ $nbPassages = $passCount->fetchColumn();
+
+ println(" │ 📊 Passages disponibles: {$nbPassages}");
+
+ // 2. Compter les ope_users
+ $opeUsersCount = $sourceDb->prepare("SELECT COUNT(*) FROM ope_users WHERE fk_operation = ?");
+ $opeUsersCount->execute([$operationId]);
+ $nbOpeUsers = $opeUsersCount->fetchColumn();
+
+ $stats['ope_users']['source'] += $nbOpeUsers;
+ println(" │ 👥 Associations users: {$nbOpeUsers}");
+
+ // 3. Compter les secteurs (via ope_users_sectors)
+ $sectorsCount = $sourceDb->prepare("
+ SELECT COUNT(DISTINCT ous.fk_sector)
+ FROM ope_users_sectors ous
+ WHERE ous.fk_operation = ?
+ ");
+ $sectorsCount->execute([$operationId]);
+ $nbSectors = $sectorsCount->fetchColumn();
+
+ println(" │ 🗺️ Secteurs distincts: {$nbSectors}");
+
+ println(" └─ " . C_CYAN . "Migration des données associées..." . C_RESET);
+
+ // Migration ope_users (simplifié pour l'exemple)
+ // ... (code de migration réel ici)
+
+ $stats['ope_pass']['source'] += $nbPassages;
+}
+
+// === MAIN ===
+
+function parseArguments($argv) {
+ $args = [
+ 'source-db' => null,
+ 'target-db' => 'pra_geo',
+ 'entity-id' => null,
+ 'limit-operations' => 3,
+ 'help' => false
+ ];
+
+ foreach ($argv as $arg) {
+ if (strpos($arg, '--') === 0) {
+ $parts = explode('=', substr($arg, 2), 2);
+ $key = $parts[0];
+ $value = $parts[1] ?? true;
+
+ if (array_key_exists($key, $args)) {
+ $args[$key] = $value;
+ }
+ }
+ }
+
+ return $args;
+}
+
+// Vérifier CLI
+if (php_sapi_name() !== 'cli') {
+ die("Ce script doit être exécuté en ligne de commande.\n");
+}
+
+$args = parseArguments($argv);
+
+if ($args['help'] || !$args['source-db'] || !$args['entity-id']) {
+ echo << $data) {
+ if ($data['source'] > 0 || $data['migrated'] > 0) {
+ printStat(ucfirst($table), $data['source'], $data['migrated'], " ");
+ }
+}
+
+println("\n✅ Migration terminée avec succès!", C_GREEN);
+exit(0);
diff --git a/api/scripts/php/verify_migration_structure.php b/api/scripts/php/verify_migration_structure.php
new file mode 100644
index 00000000..30f067be
--- /dev/null
+++ b/api/scripts/php/verify_migration_structure.php
@@ -0,0 +1,282 @@
+query("DESCRIBE `$tableName`");
+ $columns = [];
+ while ($row = $stmt->fetch(PDO::FETCH_ASSOC)) {
+ $columns[$row['Field']] = [
+ 'type' => $row['Type'],
+ 'null' => $row['Null'],
+ 'key' => $row['Key'],
+ 'default' => $row['Default'],
+ 'extra' => $row['Extra']
+ ];
+ }
+ return $columns;
+ } catch (PDOException $e) {
+ return null;
+ }
+}
+
+// Mappings de colonnes connus
+$columnMappings = [
+ // Mappings globaux
+ 'global' => [
+ 'rowid' => 'id',
+ 'active' => 'chk_active',
+ 'date_creat' => 'created_at',
+ 'date_modif' => 'updated_at',
+ ],
+ // Mappings spécifiques par table
+ 'users_entites' => [
+ 'table_target' => 'entites',
+ 'mappings' => [
+ 'libelle' => 'encrypted_name',
+ 'tel1' => 'encrypted_phone',
+ 'tel2' => 'encrypted_mobile',
+ 'email' => 'encrypted_email',
+ 'iban' => 'encrypted_iban',
+ 'bic' => 'encrypted_bic',
+ 'cp' => 'code_postal',
+ ]
+ ],
+ 'users' => [
+ 'mappings' => [
+ 'libelle' => 'encrypted_name',
+ 'username' => 'encrypted_user_name',
+ 'userpswd' => 'user_pass_hash',
+ 'userpass' => 'user_pass_hash',
+ 'prenom' => 'first_name',
+ 'nom_tournee' => 'sect_name',
+ 'telephone' => 'encrypted_phone',
+ 'mobile' => 'encrypted_mobile',
+ 'email' => 'encrypted_email',
+ 'alert_email' => 'chk_alert_email',
+ ]
+ ],
+ 'ope_pass' => [
+ 'mappings' => [
+ 'date_eve' => 'passed_at',
+ 'libelle' => 'encrypted_name',
+ 'email' => 'encrypted_email',
+ 'phone' => 'encrypted_phone',
+ 'recu' => 'nom_recu',
+ ]
+ ],
+ 'medias' => [
+ 'mappings' => [
+ 'support_rowid' => 'support_id',
+ ]
+ ],
+ 'x_villes' => [
+ 'mappings' => [
+ 'cp' => 'code_postal',
+ ]
+ ],
+];
+
+// Tables à vérifier (source => cible)
+$tablesToVerify = [
+ 'x_devises' => 'x_devises',
+ 'x_entites_types' => 'x_entites_types',
+ 'x_types_passages' => 'x_types_passages',
+ 'x_types_reglements' => 'x_types_reglements',
+ 'x_users_roles' => 'x_users_roles',
+ 'x_pays' => 'x_pays',
+ 'x_regions' => 'x_regions',
+ 'x_departements' => 'x_departements',
+ 'x_villes' => 'x_villes',
+ 'users_entites' => 'entites',
+ 'users' => 'users',
+ 'operations' => 'operations',
+ 'ope_users' => 'ope_users',
+ 'ope_users_sectors' => 'ope_users_sectors',
+ 'ope_pass' => 'ope_pass',
+ 'ope_pass_histo' => 'ope_pass_histo',
+ 'medias' => 'medias',
+ 'sectors_adresses' => 'sectors_adresses',
+];
+
+try {
+ printColor("\n╔══════════════════════════════════════════════════════════════╗", COLOR_BLUE);
+ printColor("║ VÉRIFICATION DES STRUCTURES DE MIGRATION ║", COLOR_BLUE);
+ printColor("╚══════════════════════════════════════════════════════════════╝", COLOR_BLUE);
+
+ // Connexion aux bases de données
+ printColor("\n[INFO] Connexion aux bases de données...", COLOR_BLUE);
+ $sourceDb = getSourceConnection();
+ $targetDb = getTargetConnection();
+ printColor("[OK] Connexions établies", COLOR_GREEN);
+
+ $totalIssues = 0;
+ $totalWarnings = 0;
+ $totalTables = count($tablesToVerify);
+
+ foreach ($tablesToVerify as $sourceTable => $targetTable) {
+ printColor("\n" . str_repeat("─", 70), COLOR_BLUE);
+ printColor("📊 Table: $sourceTable → $targetTable", COLOR_BLUE);
+ printColor(str_repeat("─", 70), COLOR_BLUE);
+
+ // Récupérer les colonnes
+ $sourceCols = getTableColumns($sourceDb, $sourceTable);
+ $targetCols = getTableColumns($targetDb, $targetTable);
+
+ if ($sourceCols === null) {
+ printColor("❌ ERREUR: Table source '$sourceTable' introuvable", COLOR_RED);
+ $totalIssues++;
+ continue;
+ }
+
+ if ($targetCols === null) {
+ printColor("❌ ERREUR: Table cible '$targetTable' introuvable", COLOR_RED);
+ $totalIssues++;
+ continue;
+ }
+
+ // Récupérer les mappings pour cette table
+ $tableMappings = $columnMappings['global'];
+ if (isset($columnMappings[$sourceTable]['mappings'])) {
+ $tableMappings = array_merge($tableMappings, $columnMappings[$sourceTable]['mappings']);
+ }
+
+ // Vérifier chaque colonne source
+ $unmappedSourceCols = [];
+ $mappedCols = 0;
+
+ foreach ($sourceCols as $sourceCol => $sourceInfo) {
+ // Chercher la colonne cible
+ $targetCol = $tableMappings[$sourceCol] ?? $sourceCol;
+
+ if (isset($targetCols[$targetCol])) {
+ $mappedCols++;
+ // Colonne existe et mappée correctement
+ } else {
+ // Vérifier si c'est une colonne qui doit être ignorée
+ $ignoredCols = ['dir0', 'dir1', 'dir2', 'type_fichier', 'position', 'hauteur', 'largeur',
+ 'niveaugris', 'lieudit', 'chk_habitat_vide', 'lot_nb_passages', 'departement',
+ 'fk_user', 'chk_api_adresse', 'num_adherent', 'libelle_naissance', 'josh',
+ 'email_secondaire', 'infos', 'ltt', 'lng', 'sector', 'dept_naissance',
+ 'commune_naissance', 'anciennete', 'fk_categorie', 'fk_sous_categorie',
+ 'adresse_1', 'adresse_2', 'cp', 'ville', 'matricule', 'fk_grade',
+ 'chk_adherent_ud', 'chk_adherent_ur', 'chk_adherent_fns', 'chk_archive',
+ 'chk_double_affectation', 'date_creat', 'appname', 'http_host', 'tva_intra',
+ 'rcs', 'siret', 'ape', 'couleur', 'prefecture', 'fk_titre_gerant',
+ 'gerant_prenom', 'gerant_nom', 'site_url', 'gerant_signature',
+ 'tampon_signature', 'banque_libelle', 'banque_adresse', 'banque_cp',
+ 'banque_ville', 'genbase', 'groupebase', 'userbase', 'passbase', 'demo',
+ 'lib_vert', 'lib_verts', 'lib_orange', 'lib_oranges', 'lib_rouge', 'lib_rouges',
+ 'lib_bleu', 'lib_bleus', 'icon_siege', 'icon_siege_color', 'btn_width',
+ 'nbmembres', 'nbconnex'];
+
+ if (in_array($sourceCol, $ignoredCols)) {
+ // Colonne volontairement non migrée
+ continue;
+ }
+
+ $unmappedSourceCols[] = $sourceCol;
+ }
+ }
+
+ // Vérifier les nouvelles colonnes dans la cible
+ $newTargetCols = [];
+ foreach ($targetCols as $targetCol => $targetInfo) {
+ // Vérifier si cette colonne existe dans la source
+ $sourceCol = array_search($targetCol, $tableMappings);
+ if ($sourceCol === false) {
+ $sourceCol = $targetCol; // Même nom
+ }
+
+ if (!isset($sourceCols[$sourceCol])) {
+ // Vérifier si c'est une colonne attendue (timestamp auto, etc.)
+ $autoColumns = ['created_at', 'updated_at', 'id'];
+ if (!in_array($targetCol, $autoColumns)) {
+ $newTargetCols[] = $targetCol;
+ }
+ }
+ }
+
+ // Affichage des résultats
+ printColor("✓ Colonnes source mappées: $mappedCols/" . count($sourceCols), COLOR_GREEN);
+
+ if (!empty($unmappedSourceCols)) {
+ printColor("⚠ Colonnes source NON mappées:", COLOR_YELLOW);
+ foreach ($unmappedSourceCols as $col) {
+ printColor(" - $col ({$sourceCols[$col]['type']})", COLOR_YELLOW);
+ }
+ $totalWarnings += count($unmappedSourceCols);
+ }
+
+ if (!empty($newTargetCols)) {
+ printColor("ℹ Nouvelles colonnes dans cible (seront NULL/défaut):", COLOR_YELLOW);
+ foreach ($newTargetCols as $col) {
+ $defaultValue = $targetCols[$col]['default'] ?? 'NULL';
+ $nullable = $targetCols[$col]['null'] === 'YES' ? '(nullable)' : '(NOT NULL)';
+ printColor(" - $col ({$targetCols[$col]['type']}) = $defaultValue $nullable", COLOR_YELLOW);
+ }
+ $totalWarnings += count($newTargetCols);
+ }
+
+ if (empty($unmappedSourceCols) && empty($newTargetCols)) {
+ printColor("✓ Aucun problème détecté", COLOR_GREEN);
+ }
+ }
+
+ // Résumé final
+ printColor("\n" . str_repeat("═", 70), COLOR_BLUE);
+ printColor("📈 RÉSUMÉ DE LA VÉRIFICATION", COLOR_BLUE);
+ printColor(str_repeat("═", 70), COLOR_BLUE);
+ printColor("Tables vérifiées: $totalTables", COLOR_BLUE);
+
+ if ($totalIssues > 0) {
+ printColor("❌ Erreurs critiques: $totalIssues", COLOR_RED);
+ } else {
+ printColor("✓ Aucune erreur critique", COLOR_GREEN);
+ }
+
+ if ($totalWarnings > 0) {
+ printColor("⚠ Avertissements: $totalWarnings", COLOR_YELLOW);
+ printColor(" (colonnes non mappées ou nouvelles colonnes)", COLOR_YELLOW);
+ } else {
+ printColor("✓ Aucun avertissement", COLOR_GREEN);
+ }
+
+ printColor("\n💡 Recommandations:", COLOR_BLUE);
+ printColor(" - Vérifiez que les colonnes non mappées sont intentionnelles", COLOR_RESET);
+ printColor(" - Les nouvelles colonnes cible utiliseront leurs valeurs par défaut", COLOR_RESET);
+ printColor(" - Consultez README-migration.md pour plus de détails", COLOR_RESET);
+
+ // Fermer le tunnel SSH
+ closeSshTunnel();
+
+ printColor("\n✓ Vérification terminée\n", COLOR_GREEN);
+
+} catch (Exception $e) {
+ printColor("\n❌ ERREUR CRITIQUE: " . $e->getMessage(), COLOR_RED);
+ closeSshTunnel();
+ exit(1);
+}
diff --git a/api/scripts/sql/add_unique_constraints_SIMPLE.sql b/api/scripts/sql/add_unique_constraints_SIMPLE.sql
new file mode 100644
index 00000000..473eaad5
--- /dev/null
+++ b/api/scripts/sql/add_unique_constraints_SIMPLE.sql
@@ -0,0 +1,34 @@
+-- ========================================
+-- Script SIMPLE d'ajout de contraintes UNIQUE
+-- Pour tables avec peu de données (pas de suppression de doublons)
+-- Date: 2025-10-10
+-- ========================================
+
+USE pra_geo;
+
+-- Vérifier d'abord s'il y a des doublons
+SELECT 'Vérification doublons ope_users...' as status;
+SELECT fk_operation, fk_user, COUNT(*) as count
+FROM ope_users
+GROUP BY fk_operation, fk_user
+HAVING count > 1;
+
+SELECT 'Vérification doublons ope_users_sectors...' as status;
+SELECT fk_operation, fk_user, fk_sector, COUNT(*) as count
+FROM ope_users_sectors
+GROUP BY fk_operation, fk_user, fk_sector
+HAVING count > 1;
+
+-- Ajouter les contraintes UNIQUE directement
+-- Si des doublons existent, MySQL retournera une erreur explicite
+ALTER TABLE ope_users
+ADD UNIQUE KEY `idx_operation_user` (`fk_operation`, `fk_user`);
+
+ALTER TABLE ope_users_sectors
+ADD UNIQUE KEY `idx_operation_user_sector` (`fk_operation`, `fk_user`, `fk_sector`);
+
+-- Vérification
+SHOW INDEX FROM ope_users WHERE Key_name = 'idx_operation_user';
+SHOW INDEX FROM ope_users_sectors WHERE Key_name = 'idx_operation_user_sector';
+
+SELECT 'TERMINÉ ✓' as status;
diff --git a/api/scripts/sql/add_unique_constraints_ope_tables.sql b/api/scripts/sql/add_unique_constraints_ope_tables.sql
new file mode 100644
index 00000000..39050765
--- /dev/null
+++ b/api/scripts/sql/add_unique_constraints_ope_tables.sql
@@ -0,0 +1,59 @@
+-- ========================================
+-- Script d'ajout de contraintes UNIQUE
+-- Pour éviter les doublons dans ope_users et ope_users_sectors
+-- Date: 2025-10-10
+-- ========================================
+
+USE pra_geo;
+
+-- ========================================
+-- 1. TABLE ope_users
+-- ========================================
+
+-- Vérifier et supprimer les doublons existants avant d'ajouter la contrainte
+-- (Garde la première occurrence, supprime les duplicatas)
+DELETE ou1 FROM ope_users ou1
+INNER JOIN ope_users ou2
+WHERE ou1.id > ou2.id
+ AND ou1.fk_operation = ou2.fk_operation
+ AND ou1.fk_user = ou2.fk_user;
+
+-- Ajouter la contrainte UNIQUE sur (fk_operation, fk_user)
+ALTER TABLE ope_users
+ADD UNIQUE KEY `idx_operation_user` (`fk_operation`, `fk_user`);
+
+-- ========================================
+-- 2. TABLE ope_users_sectors
+-- ========================================
+
+-- Vérifier et supprimer les doublons existants avant d'ajouter la contrainte
+-- (Garde la première occurrence, supprime les duplicatas)
+DELETE ous1 FROM ope_users_sectors ous1
+INNER JOIN ope_users_sectors ous2
+WHERE ous1.id > ous2.id
+ AND ous1.fk_operation = ous2.fk_operation
+ AND ous1.fk_user = ous2.fk_user
+ AND ous1.fk_sector = ous2.fk_sector;
+
+-- Ajouter la contrainte UNIQUE sur (fk_operation, fk_user, fk_sector)
+ALTER TABLE ope_users_sectors
+ADD UNIQUE KEY `idx_operation_user_sector` (`fk_operation`, `fk_user`, `fk_sector`);
+
+-- ========================================
+-- Vérification
+-- ========================================
+
+-- Vérifier les contraintes ajoutées
+SHOW INDEX FROM ope_users WHERE Key_name = 'idx_operation_user';
+SHOW INDEX FROM ope_users_sectors WHERE Key_name = 'idx_operation_user_sector';
+
+-- Compter les doublons restants (devrait retourner 0 lignes)
+SELECT fk_operation, fk_user, COUNT(*) as count
+FROM ope_users
+GROUP BY fk_operation, fk_user
+HAVING count > 1;
+
+SELECT fk_operation, fk_user, fk_sector, COUNT(*) as count
+FROM ope_users_sectors
+GROUP BY fk_operation, fk_user, fk_sector
+HAVING count > 1;
diff --git a/api/scripts/sql/add_unique_constraints_ope_tables_FAST.sql b/api/scripts/sql/add_unique_constraints_ope_tables_FAST.sql
new file mode 100644
index 00000000..ddc702d7
--- /dev/null
+++ b/api/scripts/sql/add_unique_constraints_ope_tables_FAST.sql
@@ -0,0 +1,70 @@
+-- ========================================
+-- Script OPTIMISÉ d'ajout de contraintes UNIQUE
+-- Pour tables avec beaucoup de données
+-- Date: 2025-10-10
+-- ========================================
+
+USE pra_geo;
+
+-- ========================================
+-- OPTION 1 : Compter les doublons d'abord
+-- ========================================
+
+SELECT 'Analyse des doublons dans ope_users...' as status;
+SELECT COUNT(*) as total_rows,
+ COUNT(DISTINCT fk_operation, fk_user) as unique_combinations,
+ COUNT(*) - COUNT(DISTINCT fk_operation, fk_user) as duplicates
+FROM ope_users;
+
+SELECT 'Analyse des doublons dans ope_users_sectors...' as status;
+SELECT COUNT(*) as total_rows,
+ COUNT(DISTINCT fk_operation, fk_user, fk_sector) as unique_combinations,
+ COUNT(*) - COUNT(DISTINCT fk_operation, fk_user, fk_sector) as duplicates
+FROM ope_users_sectors;
+
+-- ========================================
+-- OPTION 2 : Supprimer RAPIDEMENT les doublons
+-- Créer une table temporaire avec les IDs à garder
+-- ========================================
+
+-- Pour ope_users
+CREATE TEMPORARY TABLE ope_users_to_keep AS
+SELECT MIN(id) as id_to_keep, fk_operation, fk_user
+FROM ope_users
+GROUP BY fk_operation, fk_user;
+
+-- Supprimer tous les doublons (plus rapide avec NOT IN + subquery)
+DELETE FROM ope_users
+WHERE id NOT IN (SELECT id_to_keep FROM ope_users_to_keep);
+
+DROP TEMPORARY TABLE ope_users_to_keep;
+
+-- Pour ope_users_sectors
+CREATE TEMPORARY TABLE ope_users_sectors_to_keep AS
+SELECT MIN(id) as id_to_keep, fk_operation, fk_user, fk_sector
+FROM ope_users_sectors
+GROUP BY fk_operation, fk_user, fk_sector;
+
+DELETE FROM ope_users_sectors
+WHERE id NOT IN (SELECT id_to_keep FROM ope_users_sectors_to_keep);
+
+DROP TEMPORARY TABLE ope_users_sectors_to_keep;
+
+-- ========================================
+-- OPTION 3 : Ajouter les contraintes UNIQUE
+-- ========================================
+
+ALTER TABLE ope_users
+ADD UNIQUE KEY `idx_operation_user` (`fk_operation`, `fk_user`);
+
+ALTER TABLE ope_users_sectors
+ADD UNIQUE KEY `idx_operation_user_sector` (`fk_operation`, `fk_user`, `fk_sector`);
+
+-- ========================================
+-- Vérification finale
+-- ========================================
+
+SHOW INDEX FROM ope_users WHERE Key_name = 'idx_operation_user';
+SHOW INDEX FROM ope_users_sectors WHERE Key_name = 'idx_operation_user_sector';
+
+SELECT 'TERMINÉ - Contraintes UNIQUE ajoutées avec succès' as status;
diff --git a/api/scripts/sql/truncate_data_tables.sql b/api/scripts/sql/truncate_data_tables.sql
new file mode 100644
index 00000000..20ab9100
--- /dev/null
+++ b/api/scripts/sql/truncate_data_tables.sql
@@ -0,0 +1,181 @@
+-- =========================================================
+-- Script de vidage des tables de données (PRODUCTION)
+-- Option B : Vider TOUTES les tables SAUF x_* et entité 1
+-- Conserve les tables de référence x_*
+-- Conserve l'entité id=1 (super admins) et ses users/opérations
+-- Date: 2025-10-09
+-- =========================================================
+
+-- Désactiver les contraintes de clés étrangères temporairement
+SET FOREIGN_KEY_CHECKS = 0;
+
+-- =========================================================
+-- TABLES CHAT
+-- =========================================================
+TRUNCATE TABLE chat_read_receipts;
+TRUNCATE TABLE chat_messages;
+TRUNCATE TABLE chat_participants;
+TRUNCATE TABLE chat_rooms;
+
+-- =========================================================
+-- TABLES EMAIL
+-- =========================================================
+TRUNCATE TABLE email_queue;
+TRUNCATE TABLE email_counter;
+
+-- =========================================================
+-- TABLES SÉCURITÉ
+-- =========================================================
+TRUNCATE TABLE sec_failed_login_attempts;
+TRUNCATE TABLE sec_blocked_ips;
+TRUNCATE TABLE sec_alerts;
+TRUNCATE TABLE sec_performance_metrics;
+
+-- =========================================================
+-- TABLES STRIPE
+-- =========================================================
+TRUNCATE TABLE stripe_webhooks;
+TRUNCATE TABLE stripe_payment_history;
+TRUNCATE TABLE stripe_refunds;
+TRUNCATE TABLE stripe_terminal_readers;
+TRUNCATE TABLE stripe_android_certified_devices;
+-- NOTE: stripe_accounts conservé car lié à entites via FK
+
+-- =========================================================
+-- TABLES DONNÉES MÉTIER (conserver entité 1)
+-- =========================================================
+
+-- 1. Supprimer les devices des users (sauf entité 1)
+DELETE FROM user_devices
+WHERE fk_user IN (SELECT id FROM users WHERE fk_entite != 1);
+
+-- 2. Supprimer les sessions (sauf users entité 1)
+DELETE FROM z_sessions
+WHERE fk_user IN (SELECT id FROM users WHERE fk_entite != 1);
+
+-- 3. Supprimer les médias (sauf entité 1)
+DELETE FROM medias WHERE fk_entite != 1;
+
+-- 4. Supprimer les comptes Stripe (sauf entité 1)
+DELETE FROM stripe_accounts WHERE fk_entite != 1;
+
+-- 5. Supprimer l'historique des passages (sauf entité 1)
+DELETE FROM ope_pass_histo
+WHERE fk_pass IN (
+ SELECT id FROM ope_pass
+ WHERE fk_operation IN (
+ SELECT id FROM operations WHERE fk_entite != 1
+ )
+);
+
+-- 6. Supprimer les passages (sauf ceux des opérations de l'entité 1)
+DELETE FROM ope_pass
+WHERE fk_operation IN (
+ SELECT id FROM operations WHERE fk_entite != 1
+);
+
+-- 7. Supprimer les associations users-sectors (sauf entité 1)
+DELETE FROM ope_users_sectors
+WHERE fk_operation IN (
+ SELECT id FROM operations WHERE fk_entite != 1
+);
+
+-- 8. Supprimer les associations users-operations (sauf entité 1)
+DELETE FROM ope_users
+WHERE fk_operation IN (
+ SELECT id FROM operations WHERE fk_entite != 1
+);
+
+-- 9. Supprimer les adresses de secteurs (sauf entité 1)
+DELETE FROM sectors_adresses
+WHERE fk_sector IN (
+ SELECT id FROM ope_sectors WHERE fk_operation IN (
+ SELECT id FROM operations WHERE fk_entite != 1
+ )
+);
+
+-- 10. Supprimer les secteurs (sauf ceux de l'entité 1)
+DELETE FROM ope_sectors
+WHERE fk_operation IN (
+ SELECT id FROM operations WHERE fk_entite != 1
+);
+
+-- 11. Supprimer les opérations (sauf celles de l'entité 1)
+DELETE FROM operations WHERE fk_entite != 1;
+
+-- 12. Supprimer les utilisateurs (sauf ceux de l'entité 1)
+DELETE FROM users WHERE fk_entite != 1;
+
+-- 13. Supprimer les entités (sauf l'entité 1)
+DELETE FROM entites WHERE id != 1;
+
+-- 14. Vider la table params (paramètres globaux)
+TRUNCATE TABLE params;
+
+-- Réactiver les contraintes de clés étrangères
+SET FOREIGN_KEY_CHECKS = 1;
+
+-- =========================================================
+-- VÉRIFICATIONS POST-VIDAGE
+-- =========================================================
+
+SELECT '========================================' as '';
+SELECT '=== TABLES DE DONNÉES (après vidage) ===' as '';
+SELECT '========================================' as '';
+
+SELECT 'entites' as table_name, COUNT(*) as count FROM entites
+UNION ALL SELECT 'users', COUNT(*) FROM users
+UNION ALL SELECT 'operations', COUNT(*) FROM operations
+UNION ALL SELECT 'ope_sectors', COUNT(*) FROM ope_sectors
+UNION ALL SELECT 'ope_pass', COUNT(*) FROM ope_pass
+UNION ALL SELECT 'medias', COUNT(*) FROM medias
+UNION ALL SELECT 'user_devices', COUNT(*) FROM user_devices
+UNION ALL SELECT 'z_sessions', COUNT(*) FROM z_sessions;
+
+SELECT '' as '';
+SELECT '========================================' as '';
+SELECT '=== TABLES CHAT (doivent être vides) ===' as '';
+SELECT '========================================' as '';
+
+SELECT 'chat_rooms' as table_name, COUNT(*) as count FROM chat_rooms
+UNION ALL SELECT 'chat_messages', COUNT(*) FROM chat_messages
+UNION ALL SELECT 'chat_participants', COUNT(*) FROM chat_participants
+UNION ALL SELECT 'chat_read_receipts', COUNT(*) FROM chat_read_receipts;
+
+SELECT '' as '';
+SELECT '========================================' as '';
+SELECT '=== TABLES STRIPE (vides sauf accounts) ===' as '';
+SELECT '========================================' as '';
+
+SELECT 'stripe_accounts' as table_name, COUNT(*) as count FROM stripe_accounts
+UNION ALL SELECT 'stripe_webhooks', COUNT(*) FROM stripe_webhooks
+UNION ALL SELECT 'stripe_terminal_readers', COUNT(*) FROM stripe_terminal_readers
+UNION ALL SELECT 'stripe_android_certified_devices', COUNT(*) FROM stripe_android_certified_devices;
+
+SELECT '' as '';
+SELECT '========================================' as '';
+SELECT '=== ENTITÉ 1 (doit être conservée) ===' as '';
+SELECT '========================================' as '';
+
+SELECT id, encrypted_name, encrypted_email, chk_active FROM entites WHERE id = 1;
+
+SELECT '' as '';
+SELECT 'Nombre de users entité 1:' as info, COUNT(*) as count FROM users WHERE fk_entite = 1;
+SELECT 'Nombre d\'opérations entité 1:' as info, COUNT(*) as count FROM operations WHERE fk_entite = 1;
+
+SELECT '' as '';
+SELECT '========================================' as '';
+SELECT '=== TABLES x_* (doivent être remplies) ===' as '';
+SELECT '========================================' as '';
+
+SELECT 'x_devises' as table_name, COUNT(*) as count FROM x_devises
+UNION ALL SELECT 'x_pays', COUNT(*) FROM x_pays
+UNION ALL SELECT 'x_regions', COUNT(*) FROM x_regions
+UNION ALL SELECT 'x_departements', COUNT(*) FROM x_departements
+UNION ALL SELECT 'x_villes', COUNT(*) FROM x_villes
+UNION ALL SELECT 'x_departements_contours', COUNT(*) FROM x_departements_contours
+UNION ALL SELECT 'x_entites_types', COUNT(*) FROM x_entites_types
+UNION ALL SELECT 'x_types_passages', COUNT(*) FROM x_types_passages
+UNION ALL SELECT 'x_types_reglements', COUNT(*) FROM x_types_reglements
+UNION ALL SELECT 'x_users_roles', COUNT(*) FROM x_users_roles
+UNION ALL SELECT 'x_users_titres', COUNT(*) FROM x_users_titres;
diff --git a/api/scripts/test/generate_receipt_manual.php b/api/scripts/test/generate_receipt_manual.php
deleted file mode 100644
index 3a5a5335..00000000
--- a/api/scripts/test/generate_receipt_manual.php
+++ /dev/null
@@ -1,112 +0,0 @@
-#!/usr/bin/env php
-
- */
-
-declare(strict_types=1);
-
-// Simuler l'environnement web pour AppConfig en CLI
-if (php_sapi_name() === 'cli') {
- $_SERVER['SERVER_NAME'] = 'dapp.geosector.fr'; // DEV
- $_SERVER['HTTP_HOST'] = $_SERVER['SERVER_NAME'];
- $_SERVER['REMOTE_ADDR'] = '127.0.0.1';
-
- if (!function_exists('getallheaders')) {
- function getallheaders() {
- return [];
- }
- }
-}
-
-// Chargement de l'environnement
-require_once __DIR__ . '/../../vendor/autoload.php';
-require_once __DIR__ . '/../../src/Config/AppConfig.php';
-require_once __DIR__ . '/../../src/Core/Database.php';
-require_once __DIR__ . '/../../src/Services/LogService.php';
-require_once __DIR__ . '/../../src/Services/ReceiptService.php';
-
-// Vérifier qu'un ID de passage est fourni
-if ($argc < 2) {
- echo "Usage: php generate_receipt_manual.php \n";
- exit(1);
-}
-
-$passageId = (int)$argv[1];
-
-try {
- echo "=== Test de génération de reçu ===\n";
- echo "Passage ID: $passageId\n\n";
-
- // Initialisation de la configuration
- $appConfig = AppConfig::getInstance();
- $dbConfig = $appConfig->getDatabaseConfig();
-
- // Initialiser la base de données
- Database::init($dbConfig);
- $db = Database::getInstance();
-
- echo "✓ Connexion à la base de données OK\n";
-
- // Vérifier le passage
- $stmt = $db->prepare('SELECT id, fk_type, encrypted_email, nom_recu FROM ope_pass WHERE id = ?');
- $stmt->execute([$passageId]);
- $passage = $stmt->fetch(PDO::FETCH_ASSOC);
-
- if (!$passage) {
- echo "✗ Passage $passageId non trouvé\n";
- exit(1);
- }
-
- echo "✓ Passage trouvé\n";
- echo " - fk_type: " . $passage['fk_type'] . "\n";
- echo " - encrypted_email: " . (!empty($passage['encrypted_email']) ? 'OUI' : 'NON') . "\n";
- echo " - nom_recu: " . ($passage['nom_recu'] ?: 'vide') . "\n\n";
-
- // Déchiffrer l'email
- if (!empty($passage['encrypted_email'])) {
- $email = \ApiService::decryptSearchableData($passage['encrypted_email']);
- echo " - Email déchiffré: $email\n";
- echo " - Email valide: " . (filter_var($email, FILTER_VALIDATE_EMAIL) ? 'OUI' : 'NON') . "\n\n";
- } else {
- echo "✗ Aucun email chiffré trouvé\n";
- exit(1);
- }
-
- // Générer le reçu
- echo "Génération du reçu...\n";
- $receiptService = new \App\Services\ReceiptService();
- $result = $receiptService->generateReceiptForPassage($passageId);
-
- if ($result) {
- echo "✓ Reçu généré avec succès !\n\n";
-
- // Vérifier l'email dans la queue
- $stmt = $db->prepare('SELECT id, to_email, status, created_at FROM email_queue WHERE fk_pass = ? ORDER BY created_at DESC LIMIT 1');
- $stmt->execute([$passageId]);
- $queueEmail = $stmt->fetch(PDO::FETCH_ASSOC);
-
- if ($queueEmail) {
- echo "✓ Email ajouté à la queue\n";
- echo " - Queue ID: " . $queueEmail['id'] . "\n";
- echo " - Destinataire: " . $queueEmail['to_email'] . "\n";
- echo " - Status: " . $queueEmail['status'] . "\n";
- echo " - Créé: " . $queueEmail['created_at'] . "\n";
- } else {
- echo "✗ Aucun email trouvé dans la queue\n";
- }
- } else {
- echo "✗ Échec de la génération du reçu\n";
- echo "Consultez /var/www/geosector/api/logs/api.log pour plus de détails\n";
- }
-
-} catch (Exception $e) {
- echo "✗ ERREUR: " . $e->getMessage() . "\n";
- echo $e->getTraceAsString() . "\n";
- exit(1);
-}
-
-echo "\n=== Fin du test ===\n";
-exit(0);
diff --git a/api/src/Config/AppConfig.php b/api/src/Config/AppConfig.php
index c1c1faba..1860e34e 100755
--- a/api/src/Config/AppConfig.php
+++ b/api/src/Config/AppConfig.php
@@ -6,9 +6,9 @@ declare(strict_types=1);
* Configuration de l'application Geosector
*
* Ce fichier contient la configuration de l'application Geosector pour les trois environnements :
- * - Production (app.geosector.fr)
+ * - Production (app3.geosector.fr)
* - Recette (rapp.geosector.fr)
- * - Développement (app.geo.dev)
+ * - Développement (dapp.geosector.fr)
*
* Il inclut les paramètres de base de données, les informations SMTP,
* les clés de chiffrement et les configurations des services externes (Mapbox, Stripe, SMS OVH).
@@ -24,6 +24,25 @@ class AppConfig {
// Récupération du host directement depuis SERVER_NAME ou HTTP_HOST
$this->currentHost = $_SERVER['SERVER_NAME'] ?? $_SERVER['HTTP_HOST'] ?? '';
+ // Si on est en CLI (CRON, scripts), tenter de détecter via le marqueur d'environnement
+ if (empty($this->currentHost) && php_sapi_name() === 'cli') {
+ $markerFile = __DIR__ . '/../../.env_marker';
+ if (file_exists($markerFile)) {
+ $envMarker = trim(file_get_contents($markerFile));
+ switch ($envMarker) {
+ case 'production':
+ $this->currentHost = 'app3.geosector.fr';
+ break;
+ case 'recette':
+ $this->currentHost = 'rapp.geosector.fr';
+ break;
+ case 'development':
+ $this->currentHost = 'dapp.geosector.fr';
+ break;
+ }
+ }
+ }
+
// Récupérer les autres en-têtes pour une utilisation ultérieure si nécessaire
// getallheaders() n'existe pas en CLI, donc on vérifie
$this->headers = function_exists('getallheaders') ? getallheaders() : [];
@@ -81,10 +100,10 @@ class AppConfig {
];
// Configuration PRODUCTION
- $this->config['app.geosector.fr'] = array_merge($baseConfig, [
+ $this->config['app3.geosector.fr'] = array_merge($baseConfig, [
'env' => 'production',
'database' => [
- 'host' => '13.23.33.4', // Container maria4 sur IN4
+ 'host' => '13.23.33.4', // Container maria4 sur IN4 (51.159.7.190)
'name' => 'pra_geo',
'username' => 'pra_geo_user',
'password' => 'd2jAAGGWi8fxFrWgXjOA',
@@ -93,17 +112,23 @@ class AppConfig {
'host' => '13.23.33.4', // Container maria4 sur IN4
'name' => 'adresses',
'username' => 'adr_geo_user',
- 'password' => 'd66,AdrGeo.User',
+ 'password' => 'd66,AdrGeoPrd.User',
+ ],
+ 'buildings_database' => [
+ 'host' => '13.23.33.4', // Container maria4 sur IN4
+ 'name' => 'batiments',
+ 'username' => 'adr_geo_user',
+ 'password' => 'd66,AdrGeoPrd.User',
],
// Configuration Stripe PRODUCTION - Clés LIVE du CLIENT
'stripe' => [
'public_key_test' => 'pk_test_XXXXXX', // Non utilisé en PROD
'secret_key_test' => 'sk_test_XXXXXX', // Non utilisé en PROD
- 'public_key_live' => 'CLIENT_PK_LIVE_A_REMPLACER', // ← À REMPLACER avec pk_live_...
- 'secret_key_live' => 'CLIENT_SK_LIVE_A_REMPLACER', // ← À REMPLACER avec sk_live_...
+ 'public_key_live' => 'pk_live_51S5oMd1tQE0jBEomdRW82RvqAFjmqN45szbU08t8nDk4yc5QnhAJtPrP1IZJB48fF1pePUqrGsM5vyAhhoaWCT8d00nh51QIsU',
+ 'secret_key_live' => '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',
'webhook_secret_test' => 'whsec_test_XXXXXXXXXXXX',
- 'webhook_secret_live' => 'whsec_live_XXXXXXXXXXXX',
- 'api_version' => '2024-06-20',
+ 'webhook_secret_live' => 'whsec_gFnA6pR92RLdbAS2T6CSC18xsSdNBZHR',
+ 'api_version' => '2025-08-27.basil',
'application_fee_percent' => 0,
'application_fee_minimum' => 0,
'mode' => 'live', // ← MODE LIVE pour la production
@@ -114,17 +139,17 @@ class AppConfig {
$this->config['rapp.geosector.fr'] = array_merge($baseConfig, [
'env' => 'recette',
'database' => [
- // Configuration future avec maria3 (à activer après migration)
- // 'host' => '13.23.33.4', // Container maria3 sur IN3
- // 'name' => 'rca_geo',
- // 'username' => 'rca_geo_user',
- // 'password' => 'UPf3C0cQ805LypyM71iW',
+ // Configuration maria3 activée (migration effectuée le 16/10/2025)
+ 'host' => '13.23.33.4', // Container maria3 sur IN3
+ 'name' => 'rca_geo',
+ 'username' => 'rca_geo_user',
+ 'password' => 'UPf3C0cQ805LypyM71iW',
- // Configuration actuelle - base locale dans rca-geo
- 'host' => 'localhost',
- 'name' => 'geo_app',
- 'username' => 'geo_app_user_rec',
- 'password' => 'UPf3C0cQ805LypyM71iW', // À ajuster si nécessaire
+ // Configuration AVANT migration (base locale dans rca-geo) - DÉSACTIVÉE
+ // 'host' => 'localhost',
+ // 'name' => 'geo_app',
+ // 'username' => 'geo_app_user_rec',
+ // 'password' => 'UPf3C0cQ805LypyM71iW',
],
'addresses_database' => [
'host' => '13.23.33.4', // Container maria3 sur IN3
@@ -132,15 +157,21 @@ class AppConfig {
'username' => 'adr_geo_user',
'password' => 'd66,AdrGeoRec.User',
],
+ 'buildings_database' => [
+ 'host' => '13.23.33.4', // Container maria3 sur IN3
+ 'name' => 'batiments',
+ 'username' => 'adr_geo_user',
+ 'password' => 'd66,AdrGeoRec.User',
+ ],
// Configuration Stripe RECETTE - Clés TEST du CLIENT
'stripe' => [
- 'public_key_test' => 'pk_test_51S5oMd1tQE0jBEomd1u28D1bUujOcl87ASuGf9xulcz4rY27QfHrLBtQj20MVlWta4AGXsX0YMfeOJFE66AlGlkz00vG30U8Rr',
- 'secret_key_test' => 'sk_test_51S5oMd1tQE0jBEomAhzPBvUcCf0HX9ydK0xq7DagKnidp3JsovbQoVaTj24TKSUPvujQA3PP7IpIS8iWzAd15Rte00TETmbimh',
+ 'public_key_test' => 'pk_test_51S5oN00EZ9a0jvy2VSPjAYyCiJWci8lwfuakc0wpStt5YWq8RlQWyliICYIWHwTaejeW8uMSKA6KTfsfUAOvjRi500XPXWRFhJ',
+ 'secret_key_test' => 'sk_test_51S5oN00EZ9a0jvy2paTcHY91Alh5QIMJLJZJGJ188jXqte5AkxwymbLoLDiLcCn0uQH41WC75UM03HPDDp04gl7h00wfno08gE',
'public_key_live' => 'pk_live_XXXXXX', // Non utilisé en REC
'secret_key_live' => 'sk_live_XXXXXX', // Non utilisé en REC
- 'webhook_secret_test' => 'whsec_test_XXXXXXXXXXXX',
+ 'webhook_secret_test' => 'whsec_avExshr0MeWTI7wXP8478XVUkrbYG8hs',
'webhook_secret_live' => 'whsec_live_XXXXXXXXXXXX',
- 'api_version' => '2024-06-20',
+ 'api_version' => '2025-08-27.basil',
'application_fee_percent' => 0,
'application_fee_minimum' => 0,
'mode' => 'test', // ← MODE TEST pour la recette
@@ -151,17 +182,17 @@ class AppConfig {
$this->config['dapp.geosector.fr'] = array_merge($baseConfig, [
'env' => 'development',
'database' => [
- // Configuration future avec maria3 (à activer après migration)
- // 'host' => '13.23.33.4', // Container maria3 sur IN3
- // 'name' => 'dva_geo',
- // 'username' => 'dva_geo_user',
- // 'password' => 'CBq9tKHj6PGPZuTmAHV7',
+ // Configuration maria3 (migration effectuée le 07/10/2025)
+ 'host' => '13.23.33.4', // Container maria3 sur IN3
+ 'name' => 'dva_geo',
+ 'username' => 'dva_geo_user',
+ 'password' => 'CBq9tKHj6PGPZuTmAHV7',
- // Configuration actuelle - base locale dans dva-geo
- 'host' => 'localhost',
- 'name' => 'geo_app',
- 'username' => 'geo_app_user_dev',
- 'password' => 'CBq9tKHj6PGPZuTmAHV7', // À ajuster si nécessaire
+ // Configuration locale AVANT migration (sauvegarde)
+ // 'host' => 'localhost',
+ // 'name' => 'geo_app',
+ // 'username' => 'geo_app_user_dev',
+ // 'password' => 'CBq9tKHj6PGPZuTmAHV7',
],
'addresses_database' => [
'host' => '13.23.33.4', // Container maria3 sur IN3
@@ -169,6 +200,12 @@ class AppConfig {
'username' => 'adr_geo_user',
'password' => 'd66,AdrGeoDev.User',
],
+ 'buildings_database' => [
+ 'host' => '13.23.33.4', // Container maria3 sur IN3
+ 'name' => 'batiments',
+ 'username' => 'adr_geo_user',
+ 'password' => 'd66,AdrGeoDev.User',
+ ],
// Configuration Stripe DÉVELOPPEMENT - Clés TEST de Pierre (plateforme de test existante)
'stripe' => [
'public_key_test' => 'pk_test_51QwoVN00pblGEgsXkf8qlXmLGEpxDQcG0KLRpjrGLjJHd7AVZ4Iwd6ChgdjO0w0n3vRqwNCEW8KnHUe5eh3uIlkV00k07kCBmd',
@@ -177,7 +214,7 @@ class AppConfig {
'secret_key_live' => 'sk_live_XXXXXX', // Non utilisé en DEV
'webhook_secret_test' => 'whsec_test_XXXXXXXXXXXX',
'webhook_secret_live' => 'whsec_live_XXXXXXXXXXXX',
- 'api_version' => '2024-06-20',
+ 'api_version' => '2025-08-27.basil',
'application_fee_percent' => 0,
'application_fee_minimum' => 0,
'mode' => 'test', // ← MODE TEST pour le développement
@@ -197,13 +234,20 @@ class AppConfig {
// Si l'hôte n'existe pas dans la configuration, tenter une correction
if (!isset($this->config[$this->currentHost])) {
+ // Gestion des cas spéciaux (anciennes URLs)
+ if ($this->currentHost === 'app.geosector.fr') {
+ $this->currentHost = 'app3.geosector.fr';
+ }
+
// Essayer de faire correspondre avec l'un des hôtes connus
- $knownHosts = array_keys($this->config);
- foreach ($knownHosts as $host) {
- if (strpos($this->currentHost, str_replace(['app.', 'rapp.', 'dapp.'], '', $host)) !== false) {
- // Correspondance trouvée, utiliser cette configuration
- $this->currentHost = $host;
- break;
+ if (!isset($this->config[$this->currentHost])) {
+ $knownHosts = array_keys($this->config);
+ foreach ($knownHosts as $host) {
+ if (strpos($this->currentHost, str_replace(['app3.', 'rapp.', 'dapp.'], '', $host)) !== false) {
+ // Correspondance trouvée, utiliser cette configuration
+ $this->currentHost = $host;
+ break;
+ }
}
}
@@ -231,7 +275,7 @@ class AppConfig {
/**
* Retourne l'identifiant de l'application basé sur l'hôte
*
- * @return string L'identifiant de l'application (app.geosector.fr, rapp.geosector.fr, dapp.geosector.fr)
+ * @return string L'identifiant de l'application (app3.geosector.fr, rapp.geosector.fr, dapp.geosector.fr)
*/
public function getAppIdentifier(): string {
return $this->currentHost;
@@ -293,7 +337,7 @@ class AppConfig {
/**
* Retourne la configuration de la base de données
- *
+ *
* @return array Configuration de la base de données
*/
public function getDatabaseConfig(): array {
@@ -302,13 +346,22 @@ class AppConfig {
/**
* Retourne la configuration de la base de données des adresses
- *
+ *
* @return array Configuration de la base de données des adresses
*/
public function getAddressesDatabaseConfig(): array {
return $this->getCurrentConfig()['addresses_database'];
}
+ /**
+ * Retourne la configuration de la base de données des bâtiments
+ *
+ * @return array Configuration de la base de données des bâtiments
+ */
+ public function getBuildingsDatabaseConfig(): array {
+ return $this->getCurrentConfig()['buildings_database'];
+ }
+
/**
* Retourne la clé de chiffrement
*
@@ -410,13 +463,23 @@ class AppConfig {
/**
* Retourne l'adresse IP du client
- *
+ *
* @return string L'adresse IP du client
*/
public function getClientIp(): string {
return $this->clientIp;
}
+ /**
+ * Vérifie si la redirection vers RECETTE est activée
+ *
+ * @return bool True si la redirection est activée
+ */
+ public function shouldRedirectToRecette(): bool {
+ $value = getenv('REDIRECT_TO_REC');
+ return $value === 'true' || $value === '1';
+ }
+
/**
* Retourne la configuration des backups
*
diff --git a/api/src/Controllers/ChatController.php b/api/src/Controllers/ChatController.php
index 53200ebd..a76a61eb 100644
--- a/api/src/Controllers/ChatController.php
+++ b/api/src/Controllers/ChatController.php
@@ -7,13 +7,19 @@ namespace App\Controllers;
require_once __DIR__ . '/../Services/LogService.php';
require_once __DIR__ . '/../Services/ApiService.php';
-// Les classes sont déjà incluses via require_once, pas besoin de 'use' statements
+use PDO;
+use Database;
+use Request;
+use Response;
+use Session;
+use App\Services\LogService;
+use App\Services\ApiService;
class ChatController {
- private \PDO $db;
+ private PDO $db;
public function __construct() {
- $this->db = \Database::getInstance();
+ $this->db = Database::getInstance();
}
/**
@@ -24,8 +30,8 @@ class ChatController {
// L'authentification est déjà vérifiée par le Router pour les routes privées
try {
- $userId = \Session::getUserId();
- $entityId = \Session::getEntityId();
+ $userId = Session::getUserId();
+ $entityId = Session::getEntityId();
// Vérifier si c'est une synchronisation incrémentale
$updatedAfter = $_GET['updated_after'] ?? null;
@@ -186,7 +192,7 @@ class ChatController {
}
}
- \Response::json([
+ Response::json([
'status' => 'success',
'sync_timestamp' => $syncTimestamp,
'has_changes' => !empty($rooms),
@@ -194,11 +200,11 @@ class ChatController {
]);
} catch (\PDOException $e) {
- \LogService::log('Erreur lors de la récupération des conversations', [
+ LogService::log('Erreur lors de la récupération des conversations', [
'level' => 'error',
'error' => $e->getMessage()
]);
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Erreur serveur'
], 500);
@@ -213,9 +219,9 @@ class ChatController {
// L'authentification est déjà vérifiée par le Router pour les routes privées
try {
- $data = \Request::getJson();
- $userId = \Session::getUserId();
- $entityId = \Session::getEntityId();
+ $data = Request::getJson();
+ $userId = Session::getUserId();
+ $entityId = Session::getEntityId();
$userRole = $this->getUserRole($userId);
// Récupérer le temp_id s'il est fourni (pour la synchronisation offline)
@@ -223,7 +229,7 @@ class ChatController {
// Validation des données
if (!isset($data['type']) || !in_array($data['type'], ['private', 'group', 'broadcast'])) {
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Type de conversation invalide'
], 400);
@@ -233,7 +239,7 @@ class ChatController {
// Vérification des permissions pour broadcast
// Seuls les super admins (role = 9) peuvent créer des broadcasts
if ($data['type'] === 'broadcast' && $userRole != 9) {
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Seuls les super administrateurs peuvent créer des annonces'
], 403);
@@ -242,7 +248,7 @@ class ChatController {
// Validation des participants
if (!isset($data['participants']) || !is_array($data['participants']) || empty($data['participants'])) {
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Au moins un participant requis'
], 400);
@@ -251,7 +257,7 @@ class ChatController {
// Pour une conversation privée, limiter à 2 participants (incluant le créateur)
if ($data['type'] === 'private' && count($data['participants']) > 1) {
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Une conversation privée ne peut avoir que 2 participants'
], 400);
@@ -272,7 +278,7 @@ class ChatController {
if ($tempId !== null) {
$existingRoom['temp_id'] = $tempId;
}
- \Response::json([
+ Response::json([
'status' => 'success',
'room' => $existingRoom,
'existing' => true
@@ -351,7 +357,7 @@ class ChatController {
$this->db->commit();
- \LogService::log('Conversation créée', [
+ LogService::log('Conversation créée', [
'level' => 'info',
'room_id' => $roomId,
'type' => $data['type'],
@@ -367,7 +373,7 @@ class ChatController {
$room['temp_id'] = $tempId;
}
- \Response::json([
+ Response::json([
'status' => 'success',
'room' => $room
], 201);
@@ -378,20 +384,20 @@ class ChatController {
}
} catch (\PDOException $e) {
- \LogService::log('Erreur lors de la création de la conversation', [
+ LogService::log('Erreur lors de la création de la conversation', [
'level' => 'error',
'error' => $e->getMessage()
]);
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Erreur serveur'
], 500);
} catch (\Exception $e) {
- \LogService::log('Erreur lors de la création de la conversation', [
+ LogService::log('Erreur lors de la création de la conversation', [
'level' => 'error',
'error' => $e->getMessage()
]);
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => $e->getMessage()
], 400);
@@ -406,8 +412,8 @@ class ChatController {
// L'authentification est déjà vérifiée par le Router pour les routes privées
try {
- $data = \Request::getJson();
- $userId = \Session::getUserId();
+ $data = Request::getJson();
+ $userId = Session::getUserId();
// Récupérer le temp_id s'il est fourni (pour la synchronisation offline)
$tempId = $data['temp_id'] ?? null;
@@ -423,7 +429,7 @@ class ChatController {
$room = $stmt->fetch(\PDO::FETCH_ASSOC);
if (!$room) {
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Conversation non trouvée ou accès non autorisé'
], 404);
@@ -432,7 +438,7 @@ class ChatController {
// Vérifier les permissions
if ($room['created_by'] != $userId && !$room['is_admin']) {
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Seul le créateur ou un admin peut modifier la conversation'
], 403);
@@ -460,24 +466,24 @@ class ChatController {
$updatedRoom['temp_id'] = $tempId;
}
- \LogService::log('Conversation mise à jour', [
+ LogService::log('Conversation mise à jour', [
'level' => 'info',
'room_id' => $roomId,
'updated_by' => $userId
]);
- \Response::json([
+ Response::json([
'status' => 'success',
'room' => $updatedRoom
]);
} catch (\PDOException $e) {
- \LogService::log('Erreur lors de la mise à jour de la conversation', [
+ LogService::log('Erreur lors de la mise à jour de la conversation', [
'level' => 'error',
'room_id' => $roomId,
'error' => $e->getMessage()
]);
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Erreur serveur'
], 500);
@@ -493,7 +499,7 @@ class ChatController {
// L'authentification est déjà vérifiée par le Router pour les routes privées
try {
- $userId = \Session::getUserId();
+ $userId = Session::getUserId();
// Vérifier que la room existe et récupérer le créateur
$stmt = $this->db->prepare('
@@ -506,7 +512,7 @@ class ChatController {
// Vérifier que la room existe
if (!$room) {
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Conversation non trouvée'
], 404);
@@ -515,7 +521,7 @@ class ChatController {
// Vérifier que la room n'est pas déjà supprimée
if ($room['is_active'] == 0) {
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Cette conversation est déjà supprimée'
], 400);
@@ -524,7 +530,7 @@ class ChatController {
// Vérifier que l'utilisateur est le créateur
if ($room['created_by'] != $userId) {
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Seul le créateur de la conversation peut la supprimer'
], 403);
@@ -554,13 +560,13 @@ class ChatController {
$this->db->commit();
- \LogService::log('Conversation supprimée', [
+ LogService::log('Conversation supprimée', [
'level' => 'info',
'room_id' => $roomId,
'deleted_by' => $userId
]);
- \Response::json([
+ Response::json([
'status' => 'success',
'message' => 'Conversation supprimée avec succès'
]);
@@ -573,12 +579,12 @@ class ChatController {
}
} catch (\PDOException $e) {
- \LogService::log('Erreur lors de la suppression de la conversation', [
+ LogService::log('Erreur lors de la suppression de la conversation', [
'level' => 'error',
'room_id' => $roomId,
'error' => $e->getMessage()
]);
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Erreur serveur'
], 500);
@@ -593,11 +599,11 @@ class ChatController {
// L'authentification est déjà vérifiée par le Router pour les routes privées
try {
- $userId = \Session::getUserId();
+ $userId = Session::getUserId();
// Vérifier que l'utilisateur est participant
if (!$this->isUserInRoom($userId, $roomId)) {
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Accès non autorisé à cette conversation'
], 403);
@@ -658,7 +664,7 @@ class ChatController {
// Déchiffrer les noms
foreach ($messages as &$message) {
- $message['sender_name'] = \ApiService::decryptData($message['sender_name']);
+ $message['sender_name'] = ApiService::decryptData($message['sender_name']);
$message['is_read'] = (bool)$message['is_read'];
$message['is_mine'] = ($message['sender_id'] == $userId);
}
@@ -675,7 +681,7 @@ class ChatController {
// Compter les messages non lus restants (devrait être 0)
$unreadCount = $this->getUnreadCount($roomId, $userId);
- \Response::json([
+ Response::json([
'status' => 'success',
'messages' => $messages,
'has_more' => count($messages) === $limit,
@@ -684,12 +690,12 @@ class ChatController {
]);
} catch (\PDOException $e) {
- \LogService::log('Erreur lors de la récupération des messages', [
+ LogService::log('Erreur lors de la récupération des messages', [
'level' => 'error',
'room_id' => $roomId,
'error' => $e->getMessage()
]);
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Erreur serveur'
], 500);
@@ -704,15 +710,15 @@ class ChatController {
// L'authentification est déjà vérifiée par le Router pour les routes privées
try {
- $data = \Request::getJson();
- $userId = \Session::getUserId();
+ $data = Request::getJson();
+ $userId = Session::getUserId();
// Récupérer le temp_id s'il est fourni (pour la synchronisation offline)
$tempId = $data['temp_id'] ?? null;
// Vérifier que l'utilisateur est participant
if (!$this->isUserInRoom($userId, $roomId)) {
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Accès non autorisé à cette conversation'
], 403);
@@ -724,7 +730,7 @@ class ChatController {
if ($roomInfo && $roomInfo['type'] === 'broadcast') {
// Pour les broadcasts, seul le créateur peut écrire
if ($roomInfo['created_by'] != $userId) {
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Seul l\'administrateur peut poster dans une annonce'
], 403);
@@ -733,7 +739,7 @@ class ChatController {
} else {
// Pour les autres types, vérifier can_write
if (!$this->canUserWrite($userId, $roomId)) {
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Vous n\'avez pas la permission d\'écrire dans cette conversation'
], 403);
@@ -743,7 +749,7 @@ class ChatController {
// Validation du contenu
if (!isset($data['content']) || empty(trim($data['content']))) {
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Le message ne peut pas être vide'
], 400);
@@ -754,7 +760,7 @@ class ChatController {
// Limiter la longueur du message
if (mb_strlen($content, 'UTF-8') > 5000) {
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Message trop long (max 5000 caractères)'
], 400);
@@ -799,7 +805,7 @@ class ChatController {
$msgStmt->execute(['id' => $messageId]);
$message = $msgStmt->fetch(\PDO::FETCH_ASSOC);
- $message['sender_name'] = \ApiService::decryptData($message['sender_name']);
+ $message['sender_name'] = ApiService::decryptData($message['sender_name']);
$message['is_mine'] = true;
$message['is_read'] = false;
$message['read_count'] = 0;
@@ -809,25 +815,25 @@ class ChatController {
$message['temp_id'] = $tempId;
}
- \LogService::log('Message envoyé', [
+ LogService::log('Message envoyé', [
'level' => 'debug',
'room_id' => $roomId,
'message_id' => $messageId,
'sender_id' => $userId
]);
- \Response::json([
+ Response::json([
'status' => 'success',
'message' => $message
], 201);
} catch (\PDOException $e) {
- \LogService::log('Erreur lors de l\'envoi du message', [
+ LogService::log('Erreur lors de l\'envoi du message', [
'level' => 'error',
'room_id' => $roomId,
'error' => $e->getMessage()
]);
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Erreur serveur'
], 500);
@@ -842,8 +848,8 @@ class ChatController {
// L'authentification est déjà vérifiée par le Router pour les routes privées
try {
- $data = \Request::getJson();
- $userId = \Session::getUserId();
+ $data = Request::getJson();
+ $userId = Session::getUserId();
// Récupérer le temp_id s'il est fourni (pour la synchronisation offline)
$tempId = $data['temp_id'] ?? null;
@@ -858,7 +864,7 @@ class ChatController {
$message = $stmt->fetch(\PDO::FETCH_ASSOC);
if (!$message) {
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Message non trouvé'
], 404);
@@ -867,7 +873,7 @@ class ChatController {
// Vérifier que l'utilisateur est le sender du message
if ($message['sender_id'] != $userId) {
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Vous ne pouvez modifier que vos propres messages'
], 403);
@@ -876,7 +882,7 @@ class ChatController {
// Vérifier que le message n'est pas supprimé
if ($message['is_deleted']) {
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Ce message a été supprimé'
], 400);
@@ -885,7 +891,7 @@ class ChatController {
// Validation du contenu
if (!isset($data['content']) || empty(trim($data['content']))) {
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Le message ne peut pas être vide'
], 400);
@@ -896,7 +902,7 @@ class ChatController {
// Limiter la longueur du message
if (mb_strlen($content, 'UTF-8') > 5000) {
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Message trop long (max 5000 caractères)'
], 400);
@@ -931,7 +937,7 @@ class ChatController {
$msgStmt->execute(['id' => $messageId]);
$updatedMessage = $msgStmt->fetch(\PDO::FETCH_ASSOC);
- $updatedMessage['sender_name'] = \ApiService::decryptData($updatedMessage['sender_name']);
+ $updatedMessage['sender_name'] = ApiService::decryptData($updatedMessage['sender_name']);
$updatedMessage['is_mine'] = true;
// Ajouter le temp_id à la réponse si fourni
@@ -939,24 +945,24 @@ class ChatController {
$updatedMessage['temp_id'] = $tempId;
}
- \LogService::log('Message modifié', [
+ LogService::log('Message modifié', [
'level' => 'debug',
'message_id' => $messageId,
'sender_id' => $userId
]);
- \Response::json([
+ Response::json([
'status' => 'success',
'message' => $updatedMessage
]);
} catch (\PDOException $e) {
- \LogService::log('Erreur lors de la modification du message', [
+ LogService::log('Erreur lors de la modification du message', [
'level' => 'error',
'message_id' => $messageId,
'error' => $e->getMessage()
]);
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Erreur serveur'
], 500);
@@ -971,12 +977,12 @@ class ChatController {
// L'authentification est déjà vérifiée par le Router pour les routes privées
try {
- $data = \Request::getJson();
- $userId = \Session::getUserId();
+ $data = Request::getJson();
+ $userId = Session::getUserId();
// Vérifier que l'utilisateur est participant
if (!$this->isUserInRoom($userId, $roomId)) {
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Accès non autorisé à cette conversation'
], 403);
@@ -1041,18 +1047,18 @@ class ChatController {
]);
$result = $countStmt->fetch(\PDO::FETCH_ASSOC);
- \Response::json([
+ Response::json([
'status' => 'success',
'unread_count' => (int)$result['unread_count']
]);
} catch (\PDOException $e) {
- \LogService::log('Erreur lors du marquage comme lu', [
+ LogService::log('Erreur lors du marquage comme lu', [
'level' => 'error',
'room_id' => $roomId,
'error' => $e->getMessage()
]);
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Erreur serveur'
], 500);
@@ -1067,8 +1073,8 @@ class ChatController {
// L'authentification est déjà vérifiée par le Router pour les routes privées
try {
- $userId = \Session::getUserId();
- $entityId = \Session::getEntityId();
+ $userId = Session::getUserId();
+ $entityId = Session::getEntityId();
$userRole = $this->getUserRole($userId);
$sql = '
@@ -1122,11 +1128,11 @@ class ChatController {
foreach ($recipients as &$recipient) {
// Déchiffrer le nom
- $recipient['name'] = \ApiService::decryptData($recipient['name']);
+ $recipient['name'] = ApiService::decryptData($recipient['name']);
// Déchiffrer le nom de l'entité
$entiteName = $recipient['entite_name'] ?
- \ApiService::decryptData($recipient['entite_name']) :
+ ApiService::decryptData($recipient['entite_name']) :
'Sans entité';
// Créer une copie pour recipients_by_entity
@@ -1146,18 +1152,18 @@ class ChatController {
$recipientsDecrypted[] = $recipient;
}
- \Response::json([
+ Response::json([
'status' => 'success',
'recipients' => $recipientsDecrypted,
'recipients_by_entity' => $recipientsByEntity
]);
} catch (\PDOException $e) {
- \LogService::log('Erreur lors de la récupération des destinataires', [
+ LogService::log('Erreur lors de la récupération des destinataires', [
'level' => 'error',
'error' => $e->getMessage()
]);
- \Response::json([
+ Response::json([
'status' => 'error',
'message' => 'Erreur serveur'
], 500);
@@ -1225,7 +1231,7 @@ class ChatController {
$participants = $stmt->fetchAll(\PDO::FETCH_ASSOC);
foreach ($participants as &$participant) {
- $participant['name'] = \ApiService::decryptData($participant['name']);
+ $participant['name'] = ApiService::decryptData($participant['name']);
}
return $participants;
@@ -1349,7 +1355,7 @@ class ChatController {
// Déchiffrer les noms et convertir les booléens
foreach ($messages as &$message) {
- $message['sender_name'] = \ApiService::decryptData($message['sender_name']);
+ $message['sender_name'] = ApiService::decryptData($message['sender_name']);
$message['is_read'] = (bool)$message['is_read'];
$message['is_mine'] = (bool)$message['is_mine'];
}
@@ -1398,7 +1404,7 @@ class ChatController {
// Déchiffrer les noms et convertir les booléens
foreach ($messages as &$message) {
- $message['sender_name'] = \ApiService::decryptData($message['sender_name']);
+ $message['sender_name'] = ApiService::decryptData($message['sender_name']);
$message['is_read'] = (bool)$message['is_read'];
$message['is_mine'] = (bool)$message['is_mine'];
}
diff --git a/api/src/Controllers/EntiteController.php b/api/src/Controllers/EntiteController.php
index 8ce76cc5..b8c08689 100755
--- a/api/src/Controllers/EntiteController.php
+++ b/api/src/Controllers/EntiteController.php
@@ -5,6 +5,7 @@ declare(strict_types=1);
namespace App\Controllers;
require_once __DIR__ . '/../Services/LogService.php';
+require_once __DIR__ . '/../Services/EventLogService.php';
require_once __DIR__ . '/../Services/ApiService.php';
use PDO;
@@ -14,8 +15,10 @@ use AppConfig;
use Request;
use Response;
use Session;
-use LogService;
-use ApiService;
+use App\Services\LogService;
+use App\Services\EventLogService;
+use App\Services\ApiService;
+use App\Services\FileService;
use Exception;
class EntiteController {
@@ -74,13 +77,12 @@ class EntiteController {
throw new Exception('Erreur lors de la création de l\'entité');
}
- LogService::log('Création d\'une nouvelle entité GeoSector', [
- 'level' => 'info',
- 'entiteId' => $entiteId,
- 'name' => $name,
- 'postalCode' => $postalCode,
- 'cityName' => $cityName
- ]);
+ // Log de création de l'entité
+ EventLogService::logEntityCreated(
+ (int)$entiteId,
+ 1, // fk_type toujours à 1 dans cette méthode
+ $postalCode
+ );
return [
'id' => $entiteId,
@@ -220,12 +222,12 @@ class EntiteController {
throw new Exception('Erreur lors de la création de l\'entité');
}
- LogService::log('Création d\'une nouvelle entité GeoSector via getOrCreateEntiteByPostalCode', [
- 'level' => 'info',
- 'entiteId' => $entiteId,
- 'name' => $name,
- 'postalCode' => $postalCode
- ]);
+ // Log de création de l'entité
+ EventLogService::logEntityCreated(
+ $entiteId,
+ 1, // fk_type toujours à 1 dans cette méthode
+ $postalCode
+ );
return $entiteId;
} catch (Exception $e) {
@@ -559,10 +561,8 @@ class EntiteController {
$params[] = $data['gps_lng'];
}
- if (isset($data['stripe_id'])) {
- $updateFields[] = 'encrypted_stripe_id = ?';
- $params[] = ApiService::encryptData($data['stripe_id']);
- }
+ // Note: stripe_id ne peut plus être modifié ici
+ // Les données Stripe sont gérées via la table stripe_accounts
if (isset($data['chk_demo'])) {
$updateFields[] = 'chk_demo = ?';
@@ -629,12 +629,23 @@ class EntiteController {
return;
}
- LogService::log('Mise à jour d\'une entité GeoSector', [
- 'level' => 'info',
- 'userId' => $userId,
- 'entiteId' => $entiteId,
- 'isAdmin' => $isAdmin
- ]);
+ // Log de mise à jour de l'entité
+ $changes = [];
+ $encryptedFields = ['name', 'email', 'phone', 'mobile'];
+
+ foreach ($data as $key => $value) {
+ if (in_array($key, $encryptedFields)) {
+ // Champs sensibles : booléen uniquement
+ $changes['encrypted_' . $key] = true;
+ } else {
+ // Champs non sensibles : valeur
+ $changes[$key] = ['new' => $value];
+ }
+ }
+
+ if (!empty($changes)) {
+ EventLogService::logEntityUpdated((int)$entiteId, $changes);
+ }
Response::json([
'status' => 'success',
@@ -738,7 +749,7 @@ class EntiteController {
// Créer le dossier de destination
require_once __DIR__ . '/../Services/FileService.php';
- $fileService = new \FileService();
+ $fileService = new FileService();
$uploadPath = "/{$entiteId}/logo";
$fullPath = $fileService->createDirectory($entiteId, $uploadPath);
diff --git a/api/src/Controllers/FileController.php b/api/src/Controllers/FileController.php
index ade72e98..2572f633 100755
--- a/api/src/Controllers/FileController.php
+++ b/api/src/Controllers/FileController.php
@@ -14,8 +14,8 @@ use AppConfig;
use Request;
use Response;
use Session;
-use LogService;
-use ApiService;
+use App\Services\LogService;
+use App\Services\ApiService;
use Exception;
class FileController {
diff --git a/api/src/Controllers/HealthController.php b/api/src/Controllers/HealthController.php
new file mode 100644
index 00000000..3d8f9be8
--- /dev/null
+++ b/api/src/Controllers/HealthController.php
@@ -0,0 +1,115 @@
+db = Database::getInstance();
+ }
+ /**
+ * Vérifie la santé de l'API
+ * GET /api/health
+ *
+ * @return void
+ */
+ public function check(): void
+ {
+ $checks = [
+ 'api' => 'ok',
+ 'database' => $this->checkDatabase(),
+ 'directories' => $this->checkDirectories()
+ ];
+
+ // Déterminer le statut global
+ $status = in_array('error', $checks, true) ? 'error' : 'ok';
+ $httpCode = $status === 'ok' ? 200 : 503;
+
+ Response::json([
+ 'status' => $status,
+ 'checks' => $checks,
+ 'timestamp' => date('Y-m-d H:i:s'),
+ 'environment' => $this->getEnvironment()
+ ], $httpCode);
+ }
+
+ /**
+ * Vérifie la connexion à la base de données
+ *
+ * @return string 'ok' ou 'error'
+ */
+ private function checkDatabase(): string
+ {
+ try {
+ $db = Database::getInstance();
+ $stmt = $db->query("SELECT 1");
+ return $stmt ? 'ok' : 'error';
+ } catch (Exception $e) {
+ error_log("Health check database error: " . $e->getMessage());
+ return 'error';
+ }
+ }
+
+ /**
+ * Vérifie l'accessibilité des dossiers critiques
+ *
+ * @return string 'ok' ou 'error'
+ */
+ private function checkDirectories(): string
+ {
+ $basePath = __DIR__ . '/../../';
+ $requiredDirs = ['logs', 'uploads'];
+
+ foreach ($requiredDirs as $dir) {
+ $fullPath = $basePath . $dir;
+
+ if (!is_dir($fullPath)) {
+ error_log("Health check: Directory not found: $fullPath");
+ return 'error';
+ }
+
+ if (!is_writable($fullPath)) {
+ error_log("Health check: Directory not writable: $fullPath");
+ return 'error';
+ }
+ }
+
+ return 'ok';
+ }
+
+ /**
+ * Détecte l'environnement actuel
+ *
+ * @return string 'dev', 'recette' ou 'production'
+ */
+ private function getEnvironment(): string
+ {
+ $host = $_SERVER['HTTP_HOST'] ?? 'unknown';
+
+ if (str_contains($host, 'dapp.geosector.fr')) {
+ return 'dev';
+ } elseif (str_contains($host, 'rapp.geosector.fr')) {
+ return 'recette';
+ } elseif (str_contains($host, 'app3.geosector.fr') || str_contains($host, 'app.geosector.fr')) {
+ return 'production';
+ }
+
+ return 'unknown';
+ }
+}
diff --git a/api/src/Controllers/LoginController.php b/api/src/Controllers/LoginController.php
index 4d787777..bcf0374e 100755
--- a/api/src/Controllers/LoginController.php
+++ b/api/src/Controllers/LoginController.php
@@ -14,10 +14,12 @@ use AppConfig;
use Request;
use Response;
use Session;
-use LogService;
-use ApiService;
+use App\Services\LogService;
+use App\Services\ApiService;
+use App\Services\EventLogService;
require_once __DIR__ . '/../Services/LogService.php';
+require_once __DIR__ . '/../Services/EventLogService.php';
require_once __DIR__ . '/../Services/ApiService.php';
require_once __DIR__ . '/EntiteController.php';
require_once __DIR__ . '/../Services/Security/SecurityMonitor.php';
@@ -55,14 +57,6 @@ class LoginController {
// admin accessible uniquement aux fk_role>1 (admins amicale + super-admins)
$roleCondition = ($interface === 'user') ? 'AND fk_role IN (1, 2)' : 'AND fk_role>1';
- // Log pour le debug
- LogService::log('Tentative de connexion GeoSector', [
- 'level' => 'info',
- 'username' => $username,
- 'type' => $interface,
- 'role_condition' => $roleCondition
- ]);
-
// Requête optimisée: on récupère l'utilisateur et son entité en une seule fois avec LEFT JOIN
$stmt = $this->db->prepare(
'SELECT
@@ -83,11 +77,8 @@ class LoginController {
$clientIp = $_SERVER['REMOTE_ADDR'] ?? 'unknown';
$userAgent = $_SERVER['HTTP_USER_AGENT'] ?? null;
SecurityMonitor::recordFailedLogin($clientIp, $username, 'user_not_found', $userAgent);
-
- LogService::log('Tentative de connexion GeoSector échouée : utilisateur non trouvé', [
- 'level' => 'warning',
- 'username' => $username
- ]);
+
+ EventLogService::logLoginFailed($username, 'user_not_found', 1);
Response::json(['error' => 'Identifiants invalides'], 401);
return;
}
@@ -100,22 +91,15 @@ class LoginController {
$clientIp = $_SERVER['REMOTE_ADDR'] ?? 'unknown';
$userAgent = $_SERVER['HTTP_USER_AGENT'] ?? null;
SecurityMonitor::recordFailedLogin($clientIp, $username, 'invalid_password', $userAgent);
-
- LogService::log('Tentative de connexion GeoSector échouée : mot de passe incorrect', [
- 'level' => 'warning',
- 'username' => $username
- ]);
+
+ EventLogService::logLoginFailed($username, 'invalid_password', 1);
Response::json(['error' => 'Identifiants invalides'], 401);
return;
}
// Vérifier si l'utilisateur a une entité et si elle est active
if (!empty($user['fk_entite']) && (!isset($user['entite_chk_active']) || $user['entite_chk_active'] != 1)) {
- LogService::log('Tentative de connexion GeoSector échouée : entité non active', [
- 'level' => 'warning',
- 'username' => $username,
- 'entite_id' => $user['fk_entite']
- ]);
+ EventLogService::logLoginFailed($username, 'account_inactive', 1);
Response::json([
'status' => 'error',
'message' => 'Votre amicale n\'est pas activée. Veuillez contacter votre administrateur.'
@@ -307,16 +291,33 @@ class LoginController {
// Récupérer l'ID de l'opération active (première opération retournée)
$activeOperationId = $operations[0]['id'];
+ // Récupérer ope_user_id pour l'utilisateur connecté et l'opération active
+ $opeUserStmt = $this->db->prepare(
+ 'SELECT id FROM ope_users WHERE fk_user = ? AND fk_operation = ?'
+ );
+ $opeUserStmt->execute([$user['id'], $activeOperationId]);
+ $opeUser = $opeUserStmt->fetch(PDO::FETCH_ASSOC);
+
+ if ($opeUser) {
+ $userData['ope_user_id'] = $opeUser['id'];
+ } else {
+ $userData['ope_user_id'] = null;
+ }
+
// 2. Récupérer les secteurs selon l'interface et le rôle
if ($interface === 'user') {
// Interface utilisateur : seulement les secteurs affectés à l'utilisateur
- $sectorsStmt = $this->db->prepare(
- 'SELECT s.id, s.libelle, s.color, s.sector
- FROM ope_sectors s
- JOIN ope_users_sectors us ON s.id = us.fk_sector
- WHERE us.fk_operation = ? AND us.fk_user = ? AND us.chk_active = 1 AND s.chk_active = 1'
- );
- $sectorsStmt->execute([$activeOperationId, $user['id']]);
+ // Utiliser ope_user_id au lieu de users.id
+ $opeUserId = $userData['ope_user_id'];
+ if ($opeUserId) {
+ $sectorsStmt = $this->db->prepare(
+ 'SELECT s.id, s.libelle, s.color, s.sector
+ FROM ope_sectors s
+ JOIN ope_users_sectors us ON s.id = us.fk_sector
+ WHERE us.fk_operation = ? AND us.fk_user = ? AND us.chk_active = 1 AND s.chk_active = 1'
+ );
+ $sectorsStmt->execute([$activeOperationId, $opeUserId]);
+ }
} elseif ($interface === 'admin' && $user['fk_role'] == 2) {
// Interface admin avec rôle 2 : tous les secteurs distincts de l'opération
$sectorsStmt = $this->db->prepare(
@@ -344,11 +345,12 @@ class LoginController {
// 3. Récupérer les passages selon l'interface et le rôle
if ($interface === 'user' && !empty($sectors)) {
// Interface utilisateur : passages de l'utilisateur + passages à finaliser sur ses secteurs
- $userId = $user['id'];
+ // Utiliser ope_user_id au lieu de users.id
+ $opeUserId = $userData['ope_user_id'];
$sectorIds = array_column($sectors, 'id');
$sectorIdsString = implode(',', $sectorIds);
- if (!empty($sectorIdsString)) {
+ if (!empty($sectorIdsString) && $opeUserId) {
$passagesStmt = $this->db->prepare(
"SELECT id, fk_operation, fk_sector, fk_user, fk_type, fk_adresse, passed_at, numero, rue, rue_bis, ville, residence, fk_habitat, appt, niveau,
gps_lat, gps_lng, nom_recu, encrypted_name, remarque, encrypted_email, encrypted_phone, montant, fk_type_reglement, email_erreur, nb_passages
@@ -362,7 +364,7 @@ class LoginController {
)
ORDER BY passed_at DESC"
);
- $passagesStmt->execute([$activeOperationId, $userId, $userId]);
+ $passagesStmt->execute([$activeOperationId, $opeUserId, $opeUserId]);
}
} elseif ($interface === 'admin' && $user['fk_role'] == 2) {
// Interface admin avec rôle 2 : tous les passages de l'opération
@@ -423,13 +425,14 @@ class LoginController {
if (!empty($sectorIdsString)) {
$usersSectorsStmt = $this->db->prepare(
- "SELECT DISTINCT u.id, u.first_name, u.encrypted_name, u.sect_name, us.fk_sector
- FROM users u
- JOIN ope_users_sectors us ON u.id = us.fk_user
- WHERE us.fk_sector IN ($sectorIdsString)
- AND us.fk_operation = ?
- AND us.chk_active = 1
- AND u.chk_active = 1
+ "SELECT DISTINCT u.id as user_id, ou.id as ope_user_id, ou.first_name, u.encrypted_name, u.sect_name, us.fk_sector
+ FROM users u
+ JOIN ope_users ou ON u.id = ou.fk_user
+ JOIN ope_users_sectors us ON ou.id = us.fk_user AND ou.fk_operation = us.fk_operation
+ WHERE us.fk_sector IN ($sectorIdsString)
+ AND us.fk_operation = ?
+ AND us.chk_active = 1
+ AND u.chk_active = 1
AND u.id != ?" // Exclure l'utilisateur connecté
);
$usersSectorsStmt->execute([$activeOperationId, $user['id']]);
@@ -458,14 +461,27 @@ class LoginController {
// 6. Récupérer les membres (users de l'entité du user) si nécessaire
if ($interface === 'admin' && $user['fk_role'] == 2 && !empty($user['fk_entite'])) {
- $membresStmt = $this->db->prepare(
- 'SELECT id, fk_role, fk_entite, fk_titre, encrypted_name, first_name, sect_name,
- encrypted_user_name, encrypted_phone, encrypted_mobile, encrypted_email,
- date_naissance, date_embauche, chk_active
- FROM users
- WHERE fk_entite = ?'
- );
- $membresStmt->execute([$user['fk_entite']]);
+ // Si on a une opération active, on récupère aussi ope_user_id
+ if (isset($activeOperationId)) {
+ $membresStmt = $this->db->prepare(
+ 'SELECT u.id, u.fk_role, u.fk_entite, u.fk_titre, u.encrypted_name, u.first_name, u.sect_name,
+ u.encrypted_user_name, u.encrypted_phone, u.encrypted_mobile, u.encrypted_email,
+ u.date_naissance, u.date_embauche, u.chk_active, ou.id as ope_user_id
+ FROM users u
+ LEFT JOIN ope_users ou ON u.id = ou.fk_user AND ou.fk_operation = ?
+ WHERE u.fk_entite = ?'
+ );
+ $membresStmt->execute([$activeOperationId, $user['fk_entite']]);
+ } else {
+ $membresStmt = $this->db->prepare(
+ 'SELECT id, fk_role, fk_entite, fk_titre, encrypted_name, first_name, sect_name,
+ encrypted_user_name, encrypted_phone, encrypted_mobile, encrypted_email,
+ date_naissance, date_embauche, chk_active
+ FROM users
+ WHERE fk_entite = ?'
+ );
+ $membresStmt->execute([$user['fk_entite']]);
+ }
$membres = $membresStmt->fetchAll(PDO::FETCH_ASSOC);
if (!empty($membres)) {
@@ -474,6 +490,7 @@ class LoginController {
foreach ($membres as $membre) {
$membreItem = [
'id' => $membre['id'],
+ 'ope_user_id' => $membre['ope_user_id'] ?? null,
'fk_role' => $membre['fk_role'],
'fk_entite' => $membre['fk_entite'],
'fk_titre' => $membre['fk_titre'],
@@ -537,13 +554,15 @@ class LoginController {
if ($user['fk_role'] <= 2) {
// User normal ou admin avec fk_role=2: uniquement son amicale
$amicaleStmt = $this->db->prepare(
- 'SELECT e.id, e.encrypted_name as name, e.adresse1, e.adresse2, e.code_postal, e.ville,
- e.fk_region, r.libelle AS lib_region, e.fk_type, e.encrypted_phone as phone, e.encrypted_mobile as mobile,
- e.encrypted_email as email, e.gps_lat, e.gps_lng,
- e.encrypted_stripe_id as stripe_id, e.chk_demo, e.chk_mdp_manuel, e.chk_username_manuel,
- e.chk_copie_mail_recu, e.chk_accept_sms, e.chk_active, e.chk_stripe, e.chk_user_delete_pass
+ 'SELECT e.id, e.encrypted_name as name, e.adresse1, e.adresse2, e.code_postal, e.ville,
+ e.fk_region, r.libelle AS lib_region, e.fk_type, e.encrypted_phone as phone, e.encrypted_mobile as mobile,
+ e.encrypted_email as email, e.gps_lat, e.gps_lng,
+ sa.stripe_account_id as stripe_id, e.chk_demo, e.chk_mdp_manuel, e.chk_username_manuel,
+ e.chk_copie_mail_recu, e.chk_accept_sms, e.chk_active, e.chk_stripe, e.chk_user_delete_pass,
+ sa.stripe_location_id
FROM entites e
LEFT JOIN x_regions r ON e.fk_region = r.id
+ LEFT JOIN stripe_accounts sa ON e.id = sa.fk_entite
WHERE e.id = ? AND e.chk_active = 1'
);
$amicaleStmt->execute([$user['fk_entite']]);
@@ -551,13 +570,15 @@ class LoginController {
} else {
// Admin avec fk_role>2: toutes les amicales sauf id=1
$amicaleStmt = $this->db->prepare(
- 'SELECT e.id, e.encrypted_name as name, e.adresse1, e.adresse2, e.code_postal, e.ville,
- e.fk_region, r.libelle AS lib_region, e.fk_type, e.encrypted_phone as phone, e.encrypted_mobile as mobile,
- e.encrypted_email as email, e.gps_lat, e.gps_lng,
- e.encrypted_stripe_id as stripe_id, e.chk_demo, e.chk_mdp_manuel, e.chk_username_manuel,
- e.chk_copie_mail_recu, e.chk_accept_sms, e.chk_active, e.chk_stripe, e.chk_user_delete_pass
+ 'SELECT e.id, e.encrypted_name as name, e.adresse1, e.adresse2, e.code_postal, e.ville,
+ e.fk_region, r.libelle AS lib_region, e.fk_type, e.encrypted_phone as phone, e.encrypted_mobile as mobile,
+ e.encrypted_email as email, e.gps_lat, e.gps_lng,
+ sa.stripe_account_id as stripe_id, e.chk_demo, e.chk_mdp_manuel, e.chk_username_manuel,
+ e.chk_copie_mail_recu, e.chk_accept_sms, e.chk_active, e.chk_stripe, e.chk_user_delete_pass,
+ sa.stripe_location_id
FROM entites e
LEFT JOIN x_regions r ON e.fk_region = r.id
+ LEFT JOIN stripe_accounts sa ON e.id = sa.fk_entite
WHERE e.id != 1 AND e.chk_active = 1'
);
$amicaleStmt->execute();
@@ -872,6 +893,9 @@ class LoginController {
// Ajouter les données du chat à la réponse
$response['chat'] = $chatData;
+ // Log de connexion réussie
+ EventLogService::logLoginSuccess($user['id'], $user['fk_entite'] ?? null, $username);
+
// Envoi de la réponse
Response::json($response);
} catch (PDOException $e) {
@@ -918,14 +942,6 @@ class LoginController {
// Déterminer le roleCondition selon le mode (même logique que login)
$roleCondition = ($mode === 'user') ? 'AND fk_role IN (1, 2)' : 'AND fk_role>1';
- // Log pour le debug
- LogService::log('Rafraîchissement session GeoSector', [
- 'level' => 'info',
- 'userId' => $userId,
- 'mode' => $mode,
- 'role_condition' => $roleCondition
- ]);
-
// 4. Requête pour récupérer l'utilisateur et son entité (même requête que login)
$stmt = $this->db->prepare(
'SELECT
@@ -1074,15 +1090,32 @@ class LoginController {
$activeOperationId = $operations[0]['id'];
+ // Récupérer ope_user_id pour l'utilisateur connecté et l'opération active
+ $opeUserStmt = $this->db->prepare(
+ 'SELECT id FROM ope_users WHERE fk_user = ? AND fk_operation = ?'
+ );
+ $opeUserStmt->execute([$user['id'], $activeOperationId]);
+ $opeUser = $opeUserStmt->fetch(PDO::FETCH_ASSOC);
+
+ if ($opeUser) {
+ $userData['ope_user_id'] = $opeUser['id'];
+ } else {
+ $userData['ope_user_id'] = null;
+ }
+
// Récupérer les secteurs selon le mode et le rôle
if ($mode === 'user') {
- $sectorsStmt = $this->db->prepare(
- 'SELECT s.id, s.libelle, s.color, s.sector
- FROM ope_sectors s
- JOIN ope_users_sectors us ON s.id = us.fk_sector
- WHERE us.fk_operation = ? AND us.fk_user = ? AND us.chk_active = 1 AND s.chk_active = 1'
- );
- $sectorsStmt->execute([$activeOperationId, $user['id']]);
+ // Utiliser ope_user_id au lieu de users.id
+ $opeUserId = $userData['ope_user_id'];
+ if ($opeUserId) {
+ $sectorsStmt = $this->db->prepare(
+ 'SELECT s.id, s.libelle, s.color, s.sector
+ FROM ope_sectors s
+ JOIN ope_users_sectors us ON s.id = us.fk_sector
+ WHERE us.fk_operation = ? AND us.fk_user = ? AND us.chk_active = 1 AND s.chk_active = 1'
+ );
+ $sectorsStmt->execute([$activeOperationId, $opeUserId]);
+ }
} elseif ($mode === 'admin' && $user['fk_role'] == 2) {
$sectorsStmt = $this->db->prepare(
'SELECT DISTINCT s.id, s.libelle, s.color, s.sector
@@ -1106,10 +1139,12 @@ class LoginController {
// Récupérer les passages selon le mode et le rôle
if ($mode === 'user' && !empty($sectors)) {
+ // Utiliser ope_user_id au lieu de users.id
+ $opeUserId = $userData['ope_user_id'];
$sectorIds = array_column($sectors, 'id');
$sectorIdsString = implode(',', $sectorIds);
- if (!empty($sectorIdsString)) {
+ if (!empty($sectorIdsString) && $opeUserId) {
$passagesStmt = $this->db->prepare(
"SELECT id, fk_operation, fk_sector, fk_user, fk_type, fk_adresse, passed_at, numero, rue, rue_bis, ville, residence, fk_habitat, appt, niveau,
gps_lat, gps_lng, nom_recu, encrypted_name, remarque, encrypted_email, encrypted_phone, montant, fk_type_reglement, email_erreur, nb_passages
@@ -1123,7 +1158,7 @@ class LoginController {
)
ORDER BY passed_at DESC"
);
- $passagesStmt->execute([$activeOperationId, $user['id'], $user['id']]);
+ $passagesStmt->execute([$activeOperationId, $opeUserId, $opeUserId]);
}
} elseif ($mode === 'admin' && $user['fk_role'] == 2) {
$passagesStmt = $this->db->prepare(
@@ -1177,9 +1212,10 @@ class LoginController {
if (!empty($sectorIdsString)) {
$usersSectorsStmt = $this->db->prepare(
- "SELECT DISTINCT u.id, u.first_name, u.encrypted_name, u.sect_name, us.fk_sector
+ "SELECT DISTINCT u.id as user_id, ou.id as ope_user_id, ou.first_name, u.encrypted_name, u.sect_name, us.fk_sector
FROM users u
- JOIN ope_users_sectors us ON u.id = us.fk_user
+ JOIN ope_users ou ON u.id = ou.fk_user
+ JOIN ope_users_sectors us ON ou.id = us.fk_user AND ou.fk_operation = us.fk_operation
WHERE us.fk_sector IN ($sectorIdsString)
AND us.fk_operation = ?
AND us.chk_active = 1
@@ -1209,20 +1245,34 @@ class LoginController {
// Récupérer les membres si nécessaire
$membresData = [];
if ($mode === 'admin' && $user['fk_role'] == 2 && !empty($user['fk_entite'])) {
- $membresStmt = $this->db->prepare(
- 'SELECT id, fk_role, fk_entite, fk_titre, encrypted_name, first_name, sect_name,
- encrypted_user_name, encrypted_phone, encrypted_mobile, encrypted_email,
- date_naissance, date_embauche, chk_active
- FROM users
- WHERE fk_entite = ?'
- );
- $membresStmt->execute([$user['fk_entite']]);
+ // Si on a une opération active, on récupère aussi ope_user_id
+ if (isset($activeOperationId)) {
+ $membresStmt = $this->db->prepare(
+ 'SELECT u.id, u.fk_role, u.fk_entite, u.fk_titre, u.encrypted_name, u.first_name, u.sect_name,
+ u.encrypted_user_name, u.encrypted_phone, u.encrypted_mobile, u.encrypted_email,
+ u.date_naissance, u.date_embauche, u.chk_active, ou.id as ope_user_id
+ FROM users u
+ LEFT JOIN ope_users ou ON u.id = ou.fk_user AND ou.fk_operation = ?
+ WHERE u.fk_entite = ?'
+ );
+ $membresStmt->execute([$activeOperationId, $user['fk_entite']]);
+ } else {
+ $membresStmt = $this->db->prepare(
+ 'SELECT id, fk_role, fk_entite, fk_titre, encrypted_name, first_name, sect_name,
+ encrypted_user_name, encrypted_phone, encrypted_mobile, encrypted_email,
+ date_naissance, date_embauche, chk_active
+ FROM users
+ WHERE fk_entite = ?'
+ );
+ $membresStmt->execute([$user['fk_entite']]);
+ }
$membres = $membresStmt->fetchAll(PDO::FETCH_ASSOC);
if (!empty($membres)) {
foreach ($membres as $membre) {
$membreItem = [
'id' => $membre['id'],
+ 'ope_user_id' => $membre['ope_user_id'] ?? null,
'fk_role' => $membre['fk_role'],
'fk_entite' => $membre['fk_entite'],
'fk_titre' => $membre['fk_titre'],
@@ -1279,10 +1329,12 @@ class LoginController {
'SELECT e.id, e.encrypted_name as name, e.adresse1, e.adresse2, e.code_postal, e.ville,
e.fk_region, r.libelle AS lib_region, e.fk_type, e.encrypted_phone as phone, e.encrypted_mobile as mobile,
e.encrypted_email as email, e.gps_lat, e.gps_lng,
- e.encrypted_stripe_id as stripe_id, e.chk_demo, e.chk_mdp_manuel, e.chk_username_manuel,
- e.chk_copie_mail_recu, e.chk_accept_sms, e.chk_active, e.chk_stripe, e.chk_user_delete_pass
+ sa.stripe_account_id as stripe_id, e.chk_demo, e.chk_mdp_manuel, e.chk_username_manuel,
+ e.chk_copie_mail_recu, e.chk_accept_sms, e.chk_active, e.chk_stripe, e.chk_user_delete_pass,
+ sa.stripe_location_id
FROM entites e
LEFT JOIN x_regions r ON e.fk_region = r.id
+ LEFT JOIN stripe_accounts sa ON e.id = sa.fk_entite
WHERE e.id = ? AND e.chk_active = 1'
);
$amicaleStmt->execute([$user['fk_entite']]);
@@ -1292,10 +1344,12 @@ class LoginController {
'SELECT e.id, e.encrypted_name as name, e.adresse1, e.adresse2, e.code_postal, e.ville,
e.fk_region, r.libelle AS lib_region, e.fk_type, e.encrypted_phone as phone, e.encrypted_mobile as mobile,
e.encrypted_email as email, e.gps_lat, e.gps_lng,
- e.encrypted_stripe_id as stripe_id, e.chk_demo, e.chk_mdp_manuel, e.chk_username_manuel,
- e.chk_copie_mail_recu, e.chk_accept_sms, e.chk_active, e.chk_stripe, e.chk_user_delete_pass
+ sa.stripe_account_id as stripe_id, e.chk_demo, e.chk_mdp_manuel, e.chk_username_manuel,
+ e.chk_copie_mail_recu, e.chk_accept_sms, e.chk_active, e.chk_stripe, e.chk_user_delete_pass,
+ sa.stripe_location_id
FROM entites e
LEFT JOIN x_regions r ON e.fk_region = r.id
+ LEFT JOIN stripe_accounts sa ON e.id = sa.fk_entite
WHERE e.id != 1 AND e.chk_active = 1'
);
$amicaleStmt->execute();
@@ -1830,13 +1884,13 @@ class LoginController {
}
*/
- // 5. Vérification de l'existence du code postal dans la table entites
- $checkPostalStmt = $this->db->prepare('SELECT id FROM entites WHERE code_postal = ?');
- $checkPostalStmt->execute([$postalCode]);
+ // 5. Vérification de l'existence du code postal + ville dans la table entites
+ $checkPostalStmt = $this->db->prepare('SELECT id FROM entites WHERE code_postal = ? AND ville = ?');
+ $checkPostalStmt->execute([$postalCode, $cityName]);
if ($checkPostalStmt->fetch()) {
Response::json([
'status' => 'error',
- 'message' => 'Une amicale est déjà inscrite à ce code postal'
+ 'message' => 'Une amicale est déjà inscrite pour ce code postal et cette ville'
], 409);
return;
}
@@ -2073,16 +2127,15 @@ class LoginController {
// Méthodes auxiliaires
public function logout(): void {
- $userId = Session::getUserId() ?? null;
- $userEmail = Session::getUserEmail() ?? 'anonyme';
+ $userId = Session::getUserId();
+ $entityId = Session::getEntityId();
Session::logout();
- LogService::log('Déconnexion GeoSector réussie', [
- 'level' => 'info',
- 'userId' => $userId,
- 'email' => $userEmail
- ]);
+ // Log de déconnexion
+ if ($userId) {
+ EventLogService::logLogout($userId, $entityId, 0);
+ }
// Retourner une réponse standardisée
Response::json([
@@ -2106,12 +2159,20 @@ class LoginController {
// Formater la ville et le code postal pour la recherche
$citySearch = urlencode($cityName . ' ' . $postalCode);
+ // Créer un contexte avec timeout de 2 secondes
+ $context = stream_context_create([
+ 'http' => [
+ 'timeout' => 2,
+ 'ignore_errors' => true
+ ]
+ ]);
+
foreach ($keywords as $keyword) {
// Construire l'URL de recherche pour l'API adresse.gouv.fr
$searchUrl = "https://api-adresse.data.gouv.fr/search/?q=" . urlencode($keyword) . "+$citySearch&limit=5";
- // Effectuer la requête HTTP
- $response = @file_get_contents($searchUrl);
+ // Effectuer la requête HTTP avec timeout
+ $response = @file_get_contents($searchUrl, false, $context);
if ($response === false) {
LogService::log('Erreur lors de la requête à l\'API adresse.gouv.fr', [
@@ -2159,9 +2220,19 @@ class LoginController {
}
}
- // Si aucune caserne n'a été trouvée avec les mots-clés, utiliser les coordonnées du centre de la ville
+ // Si aucune caserne trouvée, chercher simplement ville + code postal avec timeout
+ $citySearch = urlencode($cityName . ' ' . $postalCode);
$cityUrl = "https://api-adresse.data.gouv.fr/search/?q=$citySearch&limit=1";
- $cityResponse = @file_get_contents($cityUrl);
+
+ // Créer un contexte avec timeout de 2 secondes
+ $context = stream_context_create([
+ 'http' => [
+ 'timeout' => 2,
+ 'ignore_errors' => true
+ ]
+ ]);
+
+ $cityResponse = @file_get_contents($cityUrl, false, $context);
if ($cityResponse !== false) {
$cityData = json_decode($cityResponse, true);
@@ -2169,7 +2240,7 @@ class LoginController {
if ($cityData && isset($cityData['features'][0]['geometry']['coordinates'])) {
$coordinates = $cityData['features'][0]['geometry']['coordinates'];
- LogService::log('Utilisation des coordonnées du centre de la ville', [
+ LogService::log('Coordonnées GPS récupérées pour l\'adresse', [
'level' => 'info',
'city' => $cityName,
'postalCode' => $postalCode
@@ -2183,6 +2254,12 @@ class LoginController {
}
// Aucune coordonnée trouvée
+ LogService::log('Aucune coordonnée GPS trouvée (timeout ou adresse invalide)', [
+ 'level' => 'warning',
+ 'city' => $cityName,
+ 'postalCode' => $postalCode
+ ]);
+
return null;
}
}
diff --git a/api/src/Controllers/LoginController.php.backup_with_sql_fix b/api/src/Controllers/LoginController.php.backup_with_sql_fix
deleted file mode 100755
index 77097be2..00000000
--- a/api/src/Controllers/LoginController.php.backup_with_sql_fix
+++ /dev/null
@@ -1,1497 +0,0 @@
-db = Database::getInstance();
- $this->appConfig = AppConfig::getInstance();
- }
-
- public function login(): void {
- try {
- $data = Request::getJson();
-
- if (!isset($data['username'], $data['password'], $data['type'])) {
- LogService::log('Tentative de connexion GeoSector échouée : données manquantes', [
- 'level' => 'warning',
- 'username' => $data['username'] ?? 'non fourni'
- ]);
- Response::json(['error' => 'Nom d\'utilisateur et mot de passe requis'], 400);
- return;
- }
-
- $interface = trim($data['type']);
- $username = trim($data['username']);
- $encryptedUsername = ApiService::encryptSearchableData($username);
-
- // Récupérer le type d'utilisateur
- // admin accessible uniquement aux fk_role>1
- // user accessible uniquement aux fk_role=1
- $roleCondition = ($interface === 'user') ? 'AND fk_role=1' : 'AND fk_role>1';
-
- // Log pour le debug
- LogService::log('Tentative de connexion GeoSector', [
- 'level' => 'info',
- 'username' => $username,
- 'type' => $interface,
- 'role_condition' => $roleCondition
- ]);
-
- // Requête optimisée: on récupère l'utilisateur et son entité en une seule fois avec LEFT JOIN
- $stmt = $this->db->prepare(
- 'SELECT
- u.id, u.encrypted_email, u.encrypted_user_name, u.encrypted_name, u.user_pass_hash,
- u.first_name, u.fk_role, u.fk_entite, u.fk_titre, u.chk_active, u.sect_name,
- u.date_naissance, u.date_embauche, u.encrypted_phone, u.encrypted_mobile,
- e.id AS entite_id, e.encrypted_name AS entite_encrypted_name,
- e.adresse1, e.code_postal, e.ville, e.gps_lat, e.gps_lng, e.chk_active AS entite_chk_active
- FROM users u
- LEFT JOIN entites e ON u.fk_entite = e.id
- WHERE u.encrypted_user_name = ? AND u.chk_active != 0 ' . $roleCondition
- );
- $stmt->execute([$encryptedUsername]);
- $user = $stmt->fetch(PDO::FETCH_ASSOC);
-
- if (!$user) {
- // Enregistrer la tentative échouée
- $clientIp = $_SERVER['REMOTE_ADDR'] ?? 'unknown';
- $userAgent = $_SERVER['HTTP_USER_AGENT'] ?? null;
- SecurityMonitor::recordFailedLogin($clientIp, $username, 'user_not_found', $userAgent);
-
- LogService::log('Tentative de connexion GeoSector échouée : utilisateur non trouvé', [
- 'level' => 'warning',
- 'username' => $username
- ]);
- Response::json(['error' => 'Identifiants invalides'], 401);
- return;
- }
-
- // Vérification du mot de passe
- $passwordValid = password_verify($data['password'], $user['user_pass_hash']);
-
- if (!$passwordValid) {
- // Enregistrer la tentative échouée
- $clientIp = $_SERVER['REMOTE_ADDR'] ?? 'unknown';
- $userAgent = $_SERVER['HTTP_USER_AGENT'] ?? null;
- SecurityMonitor::recordFailedLogin($clientIp, $username, 'invalid_password', $userAgent);
-
- LogService::log('Tentative de connexion GeoSector échouée : mot de passe incorrect', [
- 'level' => 'warning',
- 'username' => $username
- ]);
- Response::json(['error' => 'Identifiants invalides'], 401);
- return;
- }
-
- // Vérifier si l'utilisateur a une entité et si elle est active
- if (!empty($user['fk_entite']) && (!isset($user['entite_chk_active']) || $user['entite_chk_active'] != 1)) {
- LogService::log('Tentative de connexion GeoSector échouée : entité non active', [
- 'level' => 'warning',
- 'username' => $username,
- 'entite_id' => $user['fk_entite']
- ]);
- Response::json([
- 'status' => 'error',
- 'message' => 'Votre amicale n\'est pas activée. Veuillez contacter votre administrateur.'
- ], 403);
- return;
- }
-
- // Mise à jour de last_login et activation du compte si nécessaire
- $updateQuery = 'UPDATE users SET
- updated_at = NOW()' .
- ($user['chk_active'] == -1 ? ', chk_active = 1' : '') .
- ($user['chk_active'] == 2 ? ', chk_active = 1' : '') .
- ' WHERE id = ?';
-
- $updateStmt = $this->db->prepare($updateQuery);
- $updateStmt->execute([$user['id']]);
-
- // Déchiffrement du nom
- $decryptedName = ApiService::decryptData($user['encrypted_name']);
-
- // Déchiffrement de l'email si disponible
- $email = '';
- if (!empty($user['encrypted_email'])) {
- $email = ApiService::decryptSearchableData($user['encrypted_email']);
-
- // Si le déchiffrement échoue, renvoyer une erreur
- if (empty($email)) {
- LogService::log('Déchiffrement email échoué', [
- 'level' => 'error',
- 'message' => 'Déchiffrement de l\'email échoué',
- 'encrypted_email' => $user['encrypted_email'],
- 'user_id' => $user['id']
- ]);
-
- Response::json([
- 'status' => 'error',
- 'message' => 'Erreur de déchiffrement de l\'email. Exécutez le script de migration pour résoudre ce problème.',
- 'debug_info' => [
- 'encrypted_email' => $user['encrypted_email'],
- 'user_id' => $user['id']
- ]
- ], 500);
- return;
- }
- }
-
- // Création de la session
- $sessionData = [
- 'id' => $user['id'],
- 'username' => $username,
- 'email' => $email,
- 'name' => $decryptedName,
- 'first_name' => $user['first_name'] ?? '',
- 'fk_role' => $user['fk_role'] ?? '0',
- 'fk_entite' => $user['fk_entite'] ?? '0',
- ];
- Session::login($sessionData);
-
- // Vérifier et exécuter l'initialisation des contours départementaux pour d6soft
- if ($username === 'd6soft') {
- require_once __DIR__ . '/../../scripts/init_departements_contours.php';
- $initLog = \DepartementContoursInitializer::runIfNeeded($this->db, $username);
-
- if ($initLog !== null) {
- // Logger l'initialisation
- LogService::log('Initialisation des contours départementaux', [
- 'level' => 'info',
- 'username' => $username,
- 'log_count' => count($initLog)
- ]);
-
- // Logger aussi les dernières lignes du log pour diagnostic
- $lastLines = array_slice($initLog, -5);
- foreach ($lastLines as $line) {
- if (strpos($line, '✗') !== false || strpos($line, 'terminé') !== false) {
- LogService::log('Import contours: ' . $line, [
- 'level' => 'info',
- 'username' => $username
- ]);
- }
- }
- }
- }
-
- // Préparation des données utilisateur pour la réponse (uniquement les champs du user)
- $userData = [
- 'id' => $user['id'],
- 'fk_entite' => $user['fk_entite'] ?? null,
- 'fk_role' => $user['fk_role'] ?? '0',
- 'fk_titre' => $user['fk_titre'] ?? null,
- 'first_name' => $user['first_name'] ?? '',
- 'sect_name' => $user['sect_name'] ?? '',
- 'date_naissance' => $user['date_naissance'] ?? null,
- 'date_embauche' => $user['date_embauche'] ?? null,
- 'username' => $username,
- 'name' => $decryptedName
- ];
-
- // Déchiffrement du téléphone
- if (!empty($user['encrypted_phone'])) {
- $userData['phone'] = ApiService::decryptData($user['encrypted_phone']);
- } else {
- $userData['phone'] = '';
- }
-
- // Déchiffrement du mobile
- if (!empty($user['encrypted_mobile'])) {
- $userData['mobile'] = ApiService::decryptData($user['encrypted_mobile']);
- } else {
- $userData['mobile'] = '';
- }
-
- // L'email est déjà déchiffré plus haut dans le code
- $userData['email'] = $email;
-
- // Suivant l'interface et le role de l'utilisateur, on lui charge toutes ses données utiles
-
- // operations :
- // Si $interface='user' : on ne récupère que la dernière opération active
- // Si $interface='admin' et si $user['fk_role']=2 : on récupère les 3 dernières opérations dont celle active
- // Dans tous les autres cas, operations: []
-
- // secteurs :
- // On récupère les secteurs de l'opération active trouvée, sinon secteurs: []
-
- // passages :
- // On récupère les passages du ou des secteurs trouvés, sinon passages: []
-
- // users_sectors :
- // On récupère les users affectés aux secteurs partagés de l'utilisateur, si pas de secteurs, users_passages: []
-
- // clients :
- // Si $interface="admin" et si $user['fk_role']=9
- // On récupère les entités au complet sauf la entite.id=1 dans un group clients contenant id, name, adresse1, adresse2, code_postal, ville, fk_region, lib_region, fk_type, phone, mobile, email, gps_lat, gps_lng, chk_active
-
- // Suivant l'interface et le role de l'utilisateur, on lui charge toutes ses données utiles
- $operationsData = [];
- $sectorsData = [];
- $passagesData = [];
- $usersSectorsData = [];
-
- // 1. Récupération des opérations selon les critères
- $operationLimit = 0;
- $activeOperationOnly = false;
-
- if ($interface === 'user') {
- // Interface utilisateur : seulement la dernière opération active
- $operationLimit = 1;
- $activeOperationOnly = true;
- } elseif ($interface === 'admin' && $user['fk_role'] == 2) {
- // Interface admin avec rôle 2 : les 3 dernières opérations dont l'active
- $operationLimit = 3;
- } elseif ($interface === 'admin' && $user['fk_role'] > 2) {
- // Interface admin avec rôle > 2 : les 10 dernières opérations dont l'active
- $operationLimit = 10;
- } else {
- // Autres cas : pas d'opérations
- $operationLimit = 0;
- }
-
- if ($operationLimit > 0 && !empty($user['fk_entite'])) {
- $operationQuery = "SELECT id, fk_entite, libelle, date_deb, date_fin, chk_active
- FROM operations
- WHERE fk_entite = ?";
-
- if ($activeOperationOnly) {
- $operationQuery .= " AND chk_active = 1";
- }
-
- $operationQuery .= " ORDER BY id DESC LIMIT " . $operationLimit;
-
- $operationStmt = $this->db->prepare($operationQuery);
- $operationStmt->execute([$user['fk_entite']]);
- $operations = $operationStmt->fetchAll(PDO::FETCH_ASSOC);
-
- if (!empty($operations)) {
- // Formater les données des opérations
- foreach ($operations as $operation) {
- $operationsData[] = [
- 'id' => $operation['id'],
- 'fk_entite' => $operation['fk_entite'],
- 'libelle' => $operation['libelle'],
- 'date_deb' => $operation['date_deb'],
- 'date_fin' => $operation['date_fin'],
- 'chk_active' => $operation['chk_active']
- ];
- }
-
- // Récupérer l'ID de l'opération active (première opération retournée)
- $activeOperationId = $operations[0]['id'];
-
- // 2. Récupérer les secteurs selon l'interface et le rôle
- if ($interface === 'user') {
- // Interface utilisateur : seulement les secteurs affectés à l'utilisateur
- $sectorsStmt = $this->db->prepare(
- 'SELECT s.id, s.libelle, s.color, s.sector
- FROM ope_sectors s
- JOIN ope_users_sectors us ON s.id = us.fk_sector
- WHERE us.fk_operation = ? AND us.fk_user = ? AND us.chk_active = 1 AND s.chk_active = 1'
- );
- $sectorsStmt->execute([$activeOperationId, $user['id']]);
- } elseif ($interface === 'admin' && $user['fk_role'] == 2) {
- // Interface admin avec rôle 2 : tous les secteurs distincts de l'opération
- $sectorsStmt = $this->db->prepare(
- 'SELECT DISTINCT s.id, s.libelle, s.color, s.sector
- FROM ope_sectors s
- WHERE s.fk_operation = ? AND s.chk_active = 1'
- );
- $sectorsStmt->execute([$activeOperationId]);
- } else {
- // Autres cas : pas de secteurs
- $sectors = [];
- $sectorsData = [];
- }
-
- // Récupération des secteurs si une requête a été préparée
- if (isset($sectorsStmt)) {
- $sectors = $sectorsStmt->fetchAll(PDO::FETCH_ASSOC);
- } else {
- $sectors = [];
- }
-
- if (!empty($sectors)) {
- $sectorsData = $sectors;
- }
-
- // 3. Récupérer les passages selon l'interface et le rôle
- if ($interface === 'user') {
- // Interface utilisateur :
- // 1. Passages effectués par cet utilisateur (fk_type != 2)
- // 2. Passages à finaliser (fk_type = 2) sur ses secteurs
-
- $userId = $user['id'];
- $sectorIds = !empty($sectors) ? array_column($sectors, 'id') : [];
- $sectorIdsString = !empty($sectorIds) ? implode(',', $sectorIds) : '0';
-
- // Requête hybride pour récupérer les deux types de passages
- $passagesStmt = $this->db->prepare(
- "SELECT id, fk_operation, fk_sector, fk_user, fk_type, fk_adresse, passed_at, numero, rue, rue_bis, ville, residence, fk_habitat, appt, niveau,
- gps_lat, gps_lng, nom_recu, encrypted_name, remarque, encrypted_email, encrypted_phone, montant, fk_type_reglement, email_erreur, nb_passages
- FROM ope_pass
- WHERE fk_operation = ?
- AND chk_active = 1
- AND (
- (fk_user = ? AND fk_type != 2) -- Passages effectués par l'utilisateur (sauf à finaliser)
- OR
- (fk_sector IN ($sectorIdsString) AND fk_type = 2) -- Passages à finaliser sur ses secteurs
- )
- ORDER BY passed_at DESC"
- );
- $passagesStmt->execute([$activeOperationId, $userId]);
- } elseif ($interface === 'admin' && $user['fk_role'] == 2) {
- // Interface admin avec rôle 2 : tous les passages de l'opération
- $passagesStmt = $this->db->prepare(
- "SELECT id, fk_operation, fk_sector, fk_user, fk_type, fk_adresse, passed_at, numero, rue, rue_bis, ville, residence, fk_habitat, appt, niveau,
- gps_lat, gps_lng, nom_recu, encrypted_name, remarque, encrypted_email, encrypted_phone, montant, fk_type_reglement, email_erreur, nb_passages
- FROM ope_pass
- WHERE fk_operation = ? AND chk_active = 1"
- );
- $passagesStmt->execute([$activeOperationId]);
- } else {
- // Autres cas : pas de passages
- $passages = [];
- $passagesData = [];
- }
-
- // Récupération des passages si une requête a été préparée
- if (isset($passagesStmt)) {
- $passages = $passagesStmt->fetchAll(PDO::FETCH_ASSOC);
- } else {
- $passages = [];
- }
-
- if (!empty($passages)) {
- // Déchiffrer les données sensibles
- foreach ($passages as &$passage) {
- // Déchiffrement du nom
- $passage['name'] = '';
- if (!empty($passage['encrypted_name'])) {
- $passage['name'] = ApiService::decryptData($passage['encrypted_name']);
- }
- unset($passage['encrypted_name']);
-
- // Déchiffrement de l'email
- $passage['email'] = '';
- if (!empty($passage['encrypted_email'])) {
- $decryptedEmail = ApiService::decryptSearchableData($passage['encrypted_email']);
- if ($decryptedEmail) {
- $passage['email'] = $decryptedEmail;
- }
- }
- unset($passage['encrypted_email']);
-
- // Déchiffrement du téléphone
- $passage['phone'] = '';
- if (!empty($passage['encrypted_phone'])) {
- $passage['phone'] = ApiService::decryptData($passage['encrypted_phone']);
- }
- unset($passage['encrypted_phone']);
- }
- $passagesData = $passages;
- }
-
- // 4. Récupérer les utilisateurs des secteurs partagés
- if (($interface === 'user' || ($interface === 'admin' && $user['fk_role'] == 2)) && !empty($sectors)) {
- $sectorIds = array_column($sectors, 'id');
- $sectorIdsString = implode(',', $sectorIds);
-
- if (!empty($sectorIdsString)) {
- $usersSectorsStmt = $this->db->prepare(
- "SELECT DISTINCT u.id, u.first_name, u.encrypted_name, u.sect_name, us.fk_sector
- FROM users u
- JOIN ope_users_sectors us ON u.id = us.fk_user
- WHERE us.fk_sector IN ($sectorIdsString)
- AND us.fk_operation = ?
- AND us.chk_active = 1
- AND u.chk_active = 1
- AND u.id != ?" // Exclure l'utilisateur connecté
- );
- $usersSectorsStmt->execute([$activeOperationId, $user['id']]);
- $usersSectors = $usersSectorsStmt->fetchAll(PDO::FETCH_ASSOC);
-
- if (!empty($usersSectors)) {
- // Déchiffrer les noms des utilisateurs
- foreach ($usersSectors as &$userSector) {
- if (!empty($userSector['encrypted_name'])) {
- $userSector['name'] = ApiService::decryptData($userSector['encrypted_name']);
- unset($userSector['encrypted_name']);
- }
- }
- $usersSectorsData = $usersSectors;
- }
- }
- } else {
- // Autres cas : pas d'utilisateurs de secteurs
- $usersSectorsData = [];
- }
- }
- }
- }
-
- // 5. Section clients gérée plus bas pour les super-administrateurs
-
- // 6. Récupérer les membres (users de l'entité du user) si nécessaire
- if ($interface === 'admin' && $user['fk_role'] == 2 && !empty($user['fk_entite'])) {
- $membresStmt = $this->db->prepare(
- 'SELECT id, fk_role, fk_entite, fk_titre, encrypted_name, first_name, sect_name,
- encrypted_user_name, encrypted_phone, encrypted_mobile, encrypted_email,
- date_naissance, date_embauche, chk_active
- FROM users
- WHERE fk_entite = ?'
- );
- $membresStmt->execute([$user['fk_entite']]);
- $membres = $membresStmt->fetchAll(PDO::FETCH_ASSOC);
-
- if (!empty($membres)) {
- $membresData = [];
-
- foreach ($membres as $membre) {
- $membreItem = [
- 'id' => $membre['id'],
- 'fk_role' => $membre['fk_role'],
- 'fk_entite' => $membre['fk_entite'],
- 'fk_titre' => $membre['fk_titre'],
- 'first_name' => $membre['first_name'] ?? '',
- 'sect_name' => $membre['sect_name'] ?? '',
- 'date_naissance' => $membre['date_naissance'] ?? null,
- 'date_embauche' => $membre['date_embauche'] ?? null,
- 'chk_active' => $membre['chk_active']
- ];
-
- // Déchiffrement du nom
- if (!empty($membre['encrypted_name'])) {
- $membreItem['name'] = ApiService::decryptData($membre['encrypted_name']);
- } else {
- $membreItem['name'] = '';
- }
-
- // Déchiffrement du nom d'utilisateur
- if (!empty($membre['encrypted_user_name'])) {
- $membreItem['username'] = ApiService::decryptSearchableData($membre['encrypted_user_name']);
- } else {
- $membreItem['username'] = '';
- }
-
- // Déchiffrement du téléphone
- if (!empty($membre['encrypted_phone'])) {
- $membreItem['phone'] = ApiService::decryptData($membre['encrypted_phone']);
- } else {
- $membreItem['phone'] = '';
- }
-
- // Déchiffrement du mobile
- if (!empty($membre['encrypted_mobile'])) {
- $membreItem['mobile'] = ApiService::decryptData($membre['encrypted_mobile']);
- } else {
- $membreItem['mobile'] = '';
- }
-
- // Déchiffrement de l'email
- if (!empty($membre['encrypted_email'])) {
- $decryptedEmail = ApiService::decryptSearchableData($membre['encrypted_email']);
- if ($decryptedEmail) {
- $membreItem['email'] = $decryptedEmail;
- }
- } else {
- $membreItem['email'] = '';
- }
-
- $membresData[] = $membreItem;
- }
-
- // Les membres seront ajoutés à la racine de la réponse plus tard
- // (après la préparation de la réponse)
- }
- }
-
- // 7. Récupérer les amicales selon le rôle de l'utilisateur
- $amicalesData = [];
-
- if (!empty($user['fk_entite'])) {
- if ($user['fk_role'] <= 2) {
- // User normal ou admin avec fk_role=2: uniquement son amicale
- $amicaleStmt = $this->db->prepare(
- 'SELECT e.id, e.encrypted_name as name, e.adresse1, e.adresse2, e.code_postal, e.ville,
- e.fk_region, r.libelle AS lib_region, e.fk_type, e.encrypted_phone as phone, e.encrypted_mobile as mobile,
- e.encrypted_email as email, e.gps_lat, e.gps_lng,
- e.encrypted_stripe_id as stripe_id, e.chk_demo, e.chk_mdp_manuel, e.chk_username_manuel,
- e.chk_copie_mail_recu, e.chk_accept_sms, e.chk_active, e.chk_stripe, e.chk_user_delete_pass, e.chk_lot_actif
- FROM entites e
- LEFT JOIN x_regions r ON e.fk_region = r.id
- WHERE e.id = ? AND e.chk_active = 1'
- );
- $amicaleStmt->execute([$user['fk_entite']]);
- $amicales = $amicaleStmt->fetchAll(PDO::FETCH_ASSOC);
- } else {
- // Admin avec fk_role>2: toutes les amicales sauf id=1
- $amicaleStmt = $this->db->prepare(
- 'SELECT e.id, e.encrypted_name as name, e.adresse1, e.adresse2, e.code_postal, e.ville,
- e.fk_region, r.libelle AS lib_region, e.fk_type, e.encrypted_phone as phone, e.encrypted_mobile as mobile,
- e.encrypted_email as email, e.gps_lat, e.gps_lng,
- e.encrypted_stripe_id as stripe_id, e.chk_demo, e.chk_mdp_manuel, e.chk_username_manuel,
- e.chk_copie_mail_recu, e.chk_accept_sms, e.chk_active, e.chk_stripe, e.chk_user_delete_pass, e.chk_lot_actif
- FROM entites e
- LEFT JOIN x_regions r ON e.fk_region = r.id
- WHERE e.id != 1 AND e.chk_active = 1'
- );
- $amicaleStmt->execute();
- $amicales = $amicaleStmt->fetchAll(PDO::FETCH_ASSOC);
- }
-
- if (!empty($amicales)) {
- foreach ($amicales as &$amicale) {
- // Déchiffrement du nom
- if (!empty($amicale['name'])) {
- $amicale['name'] = ApiService::decryptData($amicale['name']);
- }
-
- // Déchiffrement de l'email si disponible
- if (!empty($amicale['email'])) {
- $decryptedEmail = ApiService::decryptSearchableData($amicale['email']);
- if ($decryptedEmail) {
- $amicale['email'] = $decryptedEmail;
- }
- }
-
- // Déchiffrement du téléphone
- if (!empty($amicale['phone'])) {
- $amicale['phone'] = ApiService::decryptData($amicale['phone']);
- }
-
- // Déchiffrement du mobile
- if (!empty($amicale['mobile'])) {
- $amicale['mobile'] = ApiService::decryptData($amicale['mobile']);
- }
-
- // Déchiffrement du stripe_id
- if (!empty($amicale['stripe_id'])) {
- $amicale['stripe_id'] = ApiService::decryptData($amicale['stripe_id']);
- }
- }
- $amicalesData = $amicales;
- }
- }
-
- // 8. Récupérer les entités de type 1 pour les utilisateurs avec fk_role > 2
- $entitesData = [];
-
- if ($user['fk_role'] > 2) {
- // Admin avec fk_role > 2: toutes les entités de type 1
- $entitesStmt = $this->db->prepare(
- 'SELECT e.id, e.encrypted_name as name, e.adresse1, e.adresse2, e.code_postal, e.ville,
- e.fk_region, r.libelle AS lib_region, e.fk_type, e.encrypted_phone as phone, e.encrypted_mobile as mobile,
- e.encrypted_email as email, e.gps_lat, e.gps_lng,
- e.encrypted_stripe_id as stripe_id, e.chk_demo, e.chk_mdp_manuel, e.chk_username_manuel,
- e.chk_copie_mail_recu, e.chk_accept_sms, e.chk_active, e.chk_stripe, e.chk_user_delete_pass, e.chk_lot_actif
- FROM entites e
- LEFT JOIN x_regions r ON e.fk_region = r.id
- WHERE e.fk_type = 1 AND e.chk_active = 1'
- );
- $entitesStmt->execute();
- $entites = $entitesStmt->fetchAll(PDO::FETCH_ASSOC);
-
- if (!empty($entites)) {
- foreach ($entites as &$entite) {
- // Déchiffrement du nom
- if (!empty($entite['name'])) {
- $entite['name'] = ApiService::decryptData($entite['name']);
- }
-
- // Déchiffrement de l'email si disponible
- if (!empty($entite['email'])) {
- $decryptedEmail = ApiService::decryptSearchableData($entite['email']);
- if ($decryptedEmail) {
- $entite['email'] = $decryptedEmail;
- }
- }
-
- // Déchiffrement du téléphone
- if (!empty($entite['phone'])) {
- $entite['phone'] = ApiService::decryptData($entite['phone']);
- }
-
- // Déchiffrement du mobile
- if (!empty($entite['mobile'])) {
- $entite['mobile'] = ApiService::decryptData($entite['mobile']);
- }
-
- // Déchiffrement du stripe_id
- if (!empty($entite['stripe_id'])) {
- $entite['stripe_id'] = ApiService::decryptData($entite['stripe_id']);
- }
- }
- $entitesData = $entites;
- }
- }
-
- // Préparation de la réponse
- $response = [
- 'status' => 'success',
- 'message' => 'Connexion réussie',
- 'session_id' => session_id(),
- 'session_expiry' => date('c', strtotime('+24 hours')), // Ajoute une expiration de 24h
- 'user' => $userData
- ];
-
- // Ajout des amicales à la racine de la réponse si disponibles
- if (!empty($amicalesData)) {
- // Récupérer le logo de l'entité de l'utilisateur si elle existe
- $logoData = null;
- if (!empty($user['fk_entite'])) {
- $logoStmt = $this->db->prepare('
- SELECT id, fichier, file_path, file_type, mime_type, processed_width, processed_height
- FROM medias
- WHERE support = ? AND support_id = ? AND file_category = ?
- ORDER BY created_at DESC
- LIMIT 1
- ');
- $logoStmt->execute(['entite', $user['fk_entite'], 'logo']);
- $logo = $logoStmt->fetch(PDO::FETCH_ASSOC);
-
- if ($logo && file_exists($logo['file_path'])) {
- // Lire le fichier et l'encoder en base64
- $imageData = file_get_contents($logo['file_path']);
- if ($imageData !== false) {
- $base64 = base64_encode($imageData);
- // Format data URL pour usage direct dans Flutter
- $dataUrl = 'data:' . $logo['mime_type'] . ';base64,' . $base64;
-
- $logoData = [
- 'id' => $logo['id'],
- 'data_url' => $dataUrl, // Image encodée en base64
- 'file_name' => $logo['fichier'],
- 'mime_type' => $logo['mime_type'],
- 'width' => $logo['processed_width'],
- 'height' => $logo['processed_height']
- ];
- }
- }
- }
-
- // Si c'est un tableau avec un seul élément, on envoie directement l'objet
- // pour que le client reçoive un objet et non un tableau avec un seul objet
- if (count($amicalesData) === 1) {
- $response['amicale'] = $amicalesData[0];
- // Ajouter le logo à l'amicale si disponible
- if ($logoData !== null) {
- $response['amicale']['logo'] = $logoData;
- }
- } else {
- $response['amicale'] = $amicalesData;
- // Pour plusieurs amicales, ajouter le logo à celle de l'utilisateur
- if ($logoData !== null && !empty($user['fk_entite'])) {
- foreach ($response['amicale'] as &$amicale) {
- if ($amicale['id'] == $user['fk_entite']) {
- $amicale['logo'] = $logoData;
- break;
- }
- }
- }
- }
- }
-
- // Ajout des entités à la racine de la réponse sous le nom "clients" (vide pour fk_role <= 2)
- $response['clients'] = $entitesData;
-
- // Ajout des membres à la racine de la réponse si disponibles
- if (!empty($membresData)) {
- $response['membres'] = $membresData;
- }
-
- // Ajout des opérations à la racine de la réponse si disponibles
- if (!empty($operationsData)) {
- $response['operations'] = $operationsData;
- }
-
- // Ajout des secteurs à la racine de la réponse si disponibles
- if (!empty($sectorsData)) {
- $response['sectors'] = $sectorsData;
- }
-
- // Ajout des passages à la racine de la réponse si disponibles
- if (!empty($passagesData)) {
- $response['passages'] = $passagesData;
- }
-
- // Ajout des utilisateurs des secteurs à la racine de la réponse si disponibles
- if (!empty($usersSectorsData)) {
- $response['users_sectors'] = $usersSectorsData;
- }
-
- // 5. Section clients gérée plus bas pour les super-administrateurs
-
- // 9. Récupérer les régions selon le rôle de l'utilisateur
- $regionsData = [];
-
- if ($user['fk_role'] <= 2 && !empty($user['fk_entite'])) {
- // User normal ou admin avec fk_role=2: uniquement sa région basée sur le code postal de son amicale
- $amicaleStmt = $this->db->prepare('SELECT code_postal FROM entites WHERE id = ?');
- $amicaleStmt->execute([$user['fk_entite']]);
- $amicale = $amicaleStmt->fetch(PDO::FETCH_ASSOC);
-
- if (!empty($amicale) && !empty($amicale['code_postal'])) {
- $departement = substr($amicale['code_postal'], 0, 2);
-
- $regionStmt = $this->db->prepare(
- 'SELECT id, fk_pays, libelle, libelle_long, table_osm, departements, chk_active
- FROM x_regions
- WHERE FIND_IN_SET(?, departements) > 0 AND chk_active = 1'
- );
- $regionStmt->execute([$departement]);
- $regions = $regionStmt->fetchAll(PDO::FETCH_ASSOC);
-
- if (!empty($regions)) {
- $regionsData = $regions;
- }
- }
- } else {
- // Admin avec fk_role>2: toutes les régions
- $regionStmt = $this->db->prepare(
- 'SELECT id, fk_pays, libelle, libelle_long, table_osm, departements, chk_active
- FROM x_regions
- WHERE chk_active = 1'
- );
- $regionStmt->execute();
- $regions = $regionStmt->fetchAll(PDO::FETCH_ASSOC);
-
- if (!empty($regions)) {
- $regionsData = $regions;
- }
- }
-
- // Ajout des régions à la racine de la réponse si disponibles
- if (!empty($regionsData)) {
- $response['regions'] = $regionsData;
- }
-
- // Ajout des informations du module chat
- $chatData = [];
-
- // Récupérer le nombre total de conversations de l'utilisateur
- $roomCountStmt = $this->db->prepare('
- SELECT COUNT(DISTINCT r.id) as total_rooms
- FROM chat_rooms r
- INNER JOIN chat_participants p ON r.id = p.room_id
- WHERE p.user_id = :user_id
- AND p.left_at IS NULL
- AND r.is_active = 1
- ');
- $roomCountStmt->execute(['user_id' => $user['id']]);
- $roomCount = $roomCountStmt->fetch(PDO::FETCH_ASSOC);
- $chatData['total_rooms'] = (int)($roomCount['total_rooms'] ?? 0);
-
- // Récupérer le nombre de messages non lus
- $unreadStmt = $this->db->prepare('
- SELECT COUNT(*) as unread_count
- FROM chat_messages m
- INNER JOIN chat_participants p ON m.room_id = p.room_id
- WHERE p.user_id = :user_id
- AND p.left_at IS NULL
- AND m.sender_id != :sender_id
- AND m.sent_at > COALESCE(p.last_read_at, p.joined_at)
- AND m.is_deleted = 0
- ');
- $unreadStmt->execute([
- 'user_id' => $user['id'],
- 'sender_id' => $user['id']
- ]);
- $unreadResult = $unreadStmt->fetch(PDO::FETCH_ASSOC);
- $chatData['unread_messages'] = (int)($unreadResult['unread_count'] ?? 0);
-
- // Récupérer la dernière conversation active (optionnel, pour affichage rapide)
- $lastRoomStmt = $this->db->prepare('
- SELECT
- r.id,
- r.title,
- r.type,
- (SELECT m.content
- FROM chat_messages m
- WHERE m.room_id = r.id
- AND m.is_deleted = 0
- ORDER BY m.sent_at DESC
- LIMIT 1) as last_message,
- (SELECT m.sent_at
- FROM chat_messages m
- WHERE m.room_id = r.id
- AND m.is_deleted = 0
- ORDER BY m.sent_at DESC
- LIMIT 1) as last_message_at
- FROM chat_rooms r
- INNER JOIN chat_participants p ON r.id = p.room_id
- WHERE p.user_id = :user_id
- AND p.left_at IS NULL
- AND r.is_active = 1
- ORDER BY COALESCE(
- (SELECT MAX(m.sent_at) FROM chat_messages m WHERE m.room_id = r.id),
- r.created_at
- ) DESC
- LIMIT 1
- ');
- $lastRoomStmt->execute(['user_id' => $user['id']]);
- $lastRoom = $lastRoomStmt->fetch(PDO::FETCH_ASSOC);
-
- if ($lastRoom) {
- $chatData['last_active_room'] = [
- 'id' => $lastRoom['id'],
- 'title' => $lastRoom['title'],
- 'type' => $lastRoom['type'],
- 'last_message' => $lastRoom['last_message'],
- 'last_message_at' => $lastRoom['last_message_at']
- ];
- }
-
- // Indicateur si le chat est disponible pour cet utilisateur
- $chatData['chat_enabled'] = true; // Peut être conditionné selon le rôle ou l'entité
-
- // Ajouter les données du chat à la réponse
- $response['chat'] = $chatData;
-
- // Envoi de la réponse
- Response::json($response);
- } catch (PDOException $e) {
- LogService::log('Erreur base de données lors de la connexion GeoSector', [
- 'level' => 'error',
- 'error' => $e->getMessage(),
- 'code' => $e->getCode()
- ]);
- Response::json([
- 'status' => 'error',
- 'message' => 'Erreur serveur'
- ], 500);
- } catch (Exception $e) {
- LogService::log('Erreur inattendue lors de la connexion GeoSector', [
- 'level' => 'error',
- 'error' => $e->getMessage()
- ]);
- Response::json([
- 'status' => 'error',
- 'message' => 'Une erreur inattendue est survenue'
- ], 500);
- }
- }
-
- public function lostPassword(): void {
- try {
- $data = Request::getJson();
-
- if (!isset($data['email']) || empty($data['email'])) {
- Response::json([
- 'status' => 'error',
- 'message' => 'Email requis'
- ], 400);
- return;
- }
-
- $email = trim($data['email']);
-
- // Validation de l'email
- if (!filter_var($email, FILTER_VALIDATE_EMAIL)) {
- Response::json([
- 'status' => 'error',
- 'message' => 'Format d\'email invalide'
- ], 400);
- return;
- }
-
- // Chiffrement de l'email pour la recherche
- $encryptedEmail = ApiService::encryptSearchableData($email);
-
- // Recherche de TOUS les utilisateurs avec cet email (actifs ou non)
- $stmt = $this->db->prepare('
- SELECT id, encrypted_name, encrypted_user_name, chk_active
- FROM users
- WHERE encrypted_email = ?
- ');
- $stmt->execute([$encryptedEmail]);
- $users = $stmt->fetchAll(PDO::FETCH_ASSOC);
-
- if (empty($users)) {
- Response::json([
- 'status' => 'error',
- 'message' => 'Aucun compte trouvé avec cet email'
- ], 404);
- return;
- }
-
- // Génération d'un nouveau mot de passe unique pour tous les comptes
- $newPassword = ApiService::generateSecurePassword();
- $passwordHash = password_hash($newPassword, PASSWORD_DEFAULT);
-
- // Mise à jour du mot de passe pour TOUS les comptes avec cet email
- $updateStmt = $this->db->prepare('
- UPDATE users
- SET user_pass_hash = ?, updated_at = NOW()
- WHERE encrypted_email = ?
- ');
- $updateStmt->execute([$passwordHash, $encryptedEmail]);
-
- // Récupération du nombre de comptes mis à jour
- $updatedCount = $updateStmt->rowCount();
-
- // Collecte des usernames et du premier nom pour l'email
- $usernames = [];
- $firstName = '';
- foreach ($users as $user) {
- $username = ApiService::decryptSearchableData($user['encrypted_user_name']);
- if ($username) {
- $usernames[] = $username;
- }
- // Utiliser le premier nom trouvé pour personnaliser l'email
- if (empty($firstName) && !empty($user['encrypted_name'])) {
- $firstName = ApiService::decryptData($user['encrypted_name']);
- }
- }
-
- // Si aucun nom n'a été trouvé, utiliser "Utilisateur"
- if (empty($firstName)) {
- $firstName = 'Utilisateur';
- }
-
- // Envoi d'un seul email avec le nouveau mot de passe et la liste des comptes affectés
- $emailData = [
- 'username' => implode(', ', $usernames), // Liste tous les usernames concernés
- 'password' => $newPassword
- ];
-
- $emailSent = ApiService::sendEmail(
- $email,
- $firstName,
- 'lostpwd',
- $emailData
- );
-
- if ($emailSent) {
- LogService::log('Réinitialisation mot de passe GeoSector réussie', [
- 'level' => 'info',
- 'email' => $email,
- 'comptes_modifies' => $updatedCount,
- 'usernames' => $usernames
- ]);
-
- $message = $updatedCount > 1
- ? sprintf('Un nouveau mot de passe a été envoyé pour les %d comptes associés à votre adresse email', $updatedCount)
- : 'Un nouveau mot de passe a été envoyé à votre adresse email';
-
- Response::json([
- 'status' => 'success',
- 'message' => $message
- ]);
- } else {
- LogService::log('Échec envoi email réinitialisation mot de passe GeoSector', [
- 'level' => 'error',
- 'email' => $email,
- 'comptes_modifies' => $updatedCount
- ]);
-
- Response::json([
- 'status' => 'error',
- 'message' => 'Impossible d\'envoyer l\'email. Veuillez contacter l\'administrateur.'
- ], 500);
- }
- } catch (Exception $e) {
- LogService::log('Erreur lors de la réinitialisation du mot de passe GeoSector', [
- 'level' => 'error',
- 'error' => $e->getMessage()
- ]);
-
- Response::json([
- 'status' => 'error',
- 'message' => 'Une erreur est survenue. Veuillez réessayer.'
- ], 500);
- }
- }
-
- public function register(): void {
- try {
- $data = Request::getJson();
-
- // 1. Validation des données de base
- if (
- !isset($data['email'], $data['name'], $data['amicale_name'], $data['postal_code'], $data['city_name']) ||
- empty($data['email']) || empty($data['name']) || empty($data['amicale_name']) || empty($data['postal_code'])
- ) {
- Response::json([
- 'status' => 'error',
- 'message' => 'Tous les champs sont requis'
- ], 400);
- return;
- }
-
- // 2. Validation du token et du captcha
- if (!isset($data['token']) || empty($data['token'])) {
- Response::json([
- 'status' => 'error',
- 'message' => 'Token de sécurité manquant'
- ], 400);
- return;
- }
-
- // Vérification que le token est un timestamp valide et récent
- // Le frontend envoie un timestamp en millisecondes, donc on le convertit en secondes
- $tokenTimestamp = intval($data['token']) / 1000; // Conversion millisecondes -> secondes
- $currentTime = time();
- $twoHoursAgo = $currentTime - 7200; // 2 heures = 7200 secondes (plus permissif)
-
- // Tolérance de 5 minutes pour les décalages d'horloge
- $futureTime = $currentTime + 300; // 5 minutes = 300 secondes
-
- // Log pour le débogage
- LogService::log('Vérification du token', [
- 'level' => 'info',
- 'token_ms' => $data['token'],
- 'token_sec' => $tokenTimestamp,
- 'current_time' => $currentTime,
- 'two_hours_ago' => $twoHoursAgo,
- 'future_time' => $futureTime
- ]);
-
- // Vérification plus permissive
- if ($tokenTimestamp < $twoHoursAgo || $tokenTimestamp > $futureTime) {
- LogService::log('Tentative d\'inscription avec un token invalide', [
- 'level' => 'warning',
- 'token' => $data['token'],
- 'token_sec' => $tokenTimestamp,
- 'current_time' => $currentTime,
- 'email' => $data['email'] ?? 'non fourni'
- ]);
-
- Response::json([
- 'status' => 'error',
- 'message' => 'Session expirée, veuillez rafraîchir la page et réessayer'
- ], 400);
- return;
- }
-
- if (
- !isset($data['captcha_answer'], $data['captcha_expected']) ||
- $data['captcha_answer'] != $data['captcha_expected']
- ) {
- LogService::log('Tentative d\'inscription avec un captcha invalide', [
- 'level' => 'warning',
- 'captcha_answer' => $data['captcha_answer'] ?? 'non fourni',
- 'captcha_expected' => $data['captcha_expected'] ?? 'non fourni',
- 'email' => $data['email'] ?? 'non fourni'
- ]);
-
- Response::json([
- 'status' => 'error',
- 'message' => 'Vérification anti-robot échouée'
- ], 400);
- return;
- }
-
- $email = trim($data['email']);
- $name = trim($data['name']);
- $amicaleName = trim($data['amicale_name']);
- $postalCode = trim($data['postal_code']);
- $cityName = trim($data['city_name'] ?? '');
-
- // 3. Validation de l'email
- if (!filter_var($email, FILTER_VALIDATE_EMAIL)) {
- Response::json([
- 'status' => 'error',
- 'message' => 'Format d\'email invalide'
- ], 400);
- return;
- }
-
- // 4. Vérification de l'existence de l'email
- // DÉSACTIVÉ : Le client souhaite permettre plusieurs comptes avec le même email
- $encryptedEmail = ApiService::encryptSearchableData($email);
- /*
- $checkStmt = $this->db->prepare('SELECT id FROM users WHERE encrypted_email = ?');
- $checkStmt->execute([$encryptedEmail]);
- if ($checkStmt->fetch()) {
- Response::json([
- 'status' => 'error',
- 'message' => 'Cet email est déjà utilisé'
- ], 409);
- return;
- }
- */
-
- // 5. Vérification de l'existence du code postal dans la table entites
- $checkPostalStmt = $this->db->prepare('SELECT id FROM entites WHERE code_postal = ?');
- $checkPostalStmt->execute([$postalCode]);
- if ($checkPostalStmt->fetch()) {
- Response::json([
- 'status' => 'error',
- 'message' => 'Une amicale est déjà inscrite à ce code postal'
- ], 409);
- return;
- }
-
- // 6. Recherche de la région correspondant au code postal
- $departement = substr($postalCode, 0, 2);
- $regionStmt = $this->db->prepare('
- SELECT id FROM x_regions
- WHERE FIND_IN_SET(?, departements) > 0 AND chk_active = 1
- LIMIT 1
- ');
- $regionStmt->execute([$departement]);
- $region = $regionStmt->fetch(PDO::FETCH_ASSOC);
- $regionId = $region ? $region['id'] : null;
-
- // 7. Chiffrement des données sensibles
- $encryptedName = ApiService::encryptData($name);
- $encryptedAmicaleName = ApiService::encryptData($amicaleName);
- $encryptedEmail = ApiService::encryptSearchableData($email);
-
- // 8. Création de l'entité (amicale)
- $this->db->beginTransaction();
- try {
- // Insertion de la nouvelle entité
- $entiteStmt = $this->db->prepare('
- INSERT INTO entites (
- encrypted_name,
- code_postal,
- ville,
- fk_type,
- fk_region,
- encrypted_email,
- chk_demo,
- chk_active,
- created_at
- ) VALUES (?, ?, ?, 1, ?, ?, 1, 1, NOW())
- ');
- $entiteStmt->execute([
- $encryptedAmicaleName,
- $postalCode,
- $cityName,
- $regionId,
- $encryptedEmail
- ]);
- $entiteId = $this->db->lastInsertId();
-
- if (!$entiteId) {
- throw new Exception('Erreur lors de la création de l\'entité');
- }
-
- // Recherche des coordonnées GPS de la caserne de pompiers
- try {
- $gpsCoordinates = $this->findFireStationCoordinates($postalCode, $cityName);
-
- if ($gpsCoordinates) {
- // Mise à jour des coordonnées GPS de l'entité
- $updateGpsStmt = $this->db->prepare('
- UPDATE entites
- SET gps_lat = ?, gps_lng = ?
- WHERE id = ?
- ');
- $updateGpsStmt->execute([
- $gpsCoordinates['lat'],
- $gpsCoordinates['lng'],
- $entiteId
- ]);
-
- LogService::log('Coordonnées GPS de la caserne de pompiers ajoutées', [
- 'level' => 'info',
- 'entiteId' => $entiteId,
- 'postalCode' => $postalCode,
- 'cityName' => $cityName,
- 'lat' => $gpsCoordinates['lat'],
- 'lng' => $gpsCoordinates['lng']
- ]);
- } else {
- LogService::log('Aucune caserne de pompiers trouvée', [
- 'level' => 'warning',
- 'entiteId' => $entiteId,
- 'postalCode' => $postalCode,
- 'cityName' => $cityName
- ]);
- }
- } catch (Exception $e) {
- // On ne bloque pas l'inscription si la recherche de coordonnées échoue
- LogService::log('Erreur lors de la recherche des coordonnées GPS', [
- 'level' => 'error',
- 'entiteId' => $entiteId,
- 'postalCode' => $postalCode,
- 'cityName' => $cityName,
- 'error' => $e->getMessage()
- ]);
- }
-
- // 9. Génération du nom d'utilisateur et du mot de passe
- $username = ApiService::generateUserName($this->db, $name, $postalCode, $cityName);
- $encryptedUsername = ApiService::encryptSearchableData($username);
- $password = ApiService::generateSecurePassword();
- $passwordHash = password_hash($password, PASSWORD_DEFAULT);
-
- // 10. Création de l'utilisateur administrateur
- $userStmt = $this->db->prepare('
- INSERT INTO users (
- encrypted_user_name,
- encrypted_email,
- user_pass_hash,
- encrypted_name,
- fk_role,
- created_at,
- chk_active,
- fk_entite
- ) VALUES (?, ?, ?, ?, 2, NOW(), 1, ?)
- ');
- $userStmt->execute([
- $encryptedUsername,
- $encryptedEmail,
- $passwordHash,
- $encryptedName,
- $entiteId
- ]);
- $userId = $this->db->lastInsertId();
-
- $this->db->commit();
-
- // Log du succès de l'inscription
- LogService::log('Inscription GeoSector réussie', [
- 'level' => 'info',
- 'userId' => $userId,
- 'username' => $username,
- 'email' => $email,
- 'role' => 2,
- 'entiteId' => $entiteId,
- 'amicaleName' => $amicaleName,
- 'postalCode' => $postalCode,
- 'cityName' => $cityName
- ]);
-
- // 11. Envoi des emails
- // Premier email : bienvenue avec UNIQUEMENT le nom d'utilisateur (sans mot de passe)
- // Création d'un mot de passe temporaire pour le template (ne sera pas affiché)
- $tempPassword = "********";
- $welcomeResult = ApiService::sendEmail(
- $email,
- $name,
- 'welcome',
- ['username' => $username, 'password' => $tempPassword]
- );
-
- // Email de notification aux administrateurs (sans le nom d'utilisateur ni le mot de passe)
- $notificationMessage = "Nouvelle inscription GeoSector:\n\n" .
- "Nom: $name\n" .
- "Email: $email\n" .
- "Amicale: $amicaleName\n" .
- "Code postal: $postalCode\n" .
- "Ville: $cityName\n";
-
- ApiService::sendEmail(
- "contactgeosector@gmail.com",
- "Admin GeoSector",
- 'alert',
- ['subject' => 'Nouvelle inscription GeoSector', 'message' => $notificationMessage]
- );
-
- ApiService::sendEmail(
- "contact@geosector.fr",
- "Admin GeoSector",
- 'alert',
- ['subject' => 'Nouvelle inscription GeoSector', 'message' => $notificationMessage]
- );
-
- // Attendre un court délai avant d'envoyer le second email (pour éviter les filtres anti-spam)
- sleep(2);
-
- // Second email : UNIQUEMENT le mot de passe
- $passwordResult = ApiService::sendEmail(
- $email,
- $name,
- 'lostpwd',
- ['username' => $username, 'password' => $password]
- );
-
- // Réponse selon le résultat de l'envoi d'email
- if ($welcomeResult === 0 || $passwordResult === 0) {
- Response::json([
- 'status' => 'warning',
- 'message' => 'Compte créé avec succès mais impossible de vous envoyer tous les emails. ' .
- 'Rendez-vous sur la page de login et choisissez mot de passe perdu pour recevoir votre mot de passe.'
- ], 201);
- } else {
- Response::json([
- 'status' => 'success',
- 'message' => 'Votre compte a bien été créé et vous recevrez par email votre identifiant et mot de passe'
- ], 201);
- }
- } catch (Exception $e) {
- $this->db->rollBack();
- LogService::log('Erreur lors de la création du compte GeoSector', [
- 'level' => 'error',
- 'error' => $e->getMessage(),
- 'email' => $email,
- 'amicaleName' => $amicaleName,
- 'postalCode' => $postalCode
- ]);
-
- Response::json([
- 'status' => 'error',
- 'message' => $e->getMessage()
- ], 500);
- return;
- }
- } catch (PDOException $e) {
- LogService::log('Erreur serveur lors de l\'inscription GeoSector', [
- 'level' => 'error',
- 'error' => $e->getMessage(),
- 'code' => $e->getCode(),
- 'trace' => $e->getTraceAsString()
- ]);
- Response::json([
- 'status' => 'error',
- 'message' => 'Erreur lors de la création du compte. Veuillez réessayer.'
- ], 500);
- } catch (Exception $e) {
- LogService::log('Erreur inattendue lors de l\'inscription GeoSector', [
- 'level' => 'error',
- 'error' => $e->getMessage()
- ]);
- Response::json([
- 'status' => 'error',
- 'message' => 'Une erreur inattendue est survenue. Veuillez réessayer.'
- ], 500);
- }
- }
-
- // Méthodes auxiliaires
-
- public function logout(): void {
- $userId = Session::getUserId() ?? null;
- $userEmail = Session::getUserEmail() ?? 'anonyme';
-
- Session::logout();
-
- LogService::log('Déconnexion GeoSector réussie', [
- 'level' => 'info',
- 'userId' => $userId,
- 'email' => $userEmail
- ]);
-
- // Retourner une réponse standardisée
- Response::json([
- 'status' => 'success',
- 'message' => 'Déconnexion réussie'
- ]);
- }
-
- /**
- * Recherche les coordonnées GPS d'une caserne de pompiers dans une ville donnée
- *
- * @param string $postalCode Le code postal de la ville
- * @param string $cityName Le nom de la ville
- * @return array|null Tableau associatif contenant les coordonnées GPS (lat, lng) ou null si aucune caserne trouvée
- * @throws Exception En cas d'erreur lors de la requête API
- */
- private function findFireStationCoordinates(string $postalCode, string $cityName): ?array {
- // Mots-clés pour rechercher une caserne de pompiers
- $keywords = ['pompiers', 'sdis', 'sapeurs-pompiers', 'caserne', 'centre de secours'];
-
- // Formater la ville et le code postal pour la recherche
- $citySearch = urlencode($cityName . ' ' . $postalCode);
-
- foreach ($keywords as $keyword) {
- // Construire l'URL de recherche pour l'API adresse.gouv.fr
- $searchUrl = "https://api-adresse.data.gouv.fr/search/?q=" . urlencode($keyword) . "+$citySearch&limit=5";
-
- // Effectuer la requête HTTP
- $response = @file_get_contents($searchUrl);
-
- if ($response === false) {
- LogService::log('Erreur lors de la requête à l\'API adresse.gouv.fr', [
- 'level' => 'error',
- 'url' => $searchUrl
- ]);
- continue; // Essayer avec le mot-clé suivant
- }
-
- // Décoder la réponse JSON
- $data = json_decode($response, true);
-
- if (!$data || !isset($data['features']) || empty($data['features'])) {
- continue; // Aucun résultat, essayer avec le mot-clé suivant
- }
-
- // Parcourir les résultats pour trouver une caserne de pompiers
- foreach ($data['features'] as $feature) {
- $properties = $feature['properties'] ?? [];
- $name = strtolower($properties['name'] ?? '');
- $label = strtolower($properties['label'] ?? '');
-
- // Vérifier si le résultat correspond à une caserne de pompiers
- if (
- strpos($name, 'pompier') !== false ||
- strpos($name, 'sdis') !== false ||
- strpos($label, 'pompier') !== false ||
- strpos($label, 'sdis') !== false ||
- strpos($name, 'caserne') !== false ||
- strpos($label, 'caserne') !== false ||
- strpos($name, 'centre de secours') !== false ||
- strpos($label, 'centre de secours') !== false
- ) {
- // Extraire les coordonnées GPS
- $coordinates = $feature['geometry']['coordinates'] ?? null;
-
- if ($coordinates && count($coordinates) >= 2) {
- // L'API retourne les coordonnées au format [longitude, latitude]
- return [
- 'lng' => $coordinates[0],
- 'lat' => $coordinates[1]
- ];
- }
- }
- }
- }
-
- // Si aucune caserne n'a été trouvée avec les mots-clés, utiliser les coordonnées du centre de la ville
- $cityUrl = "https://api-adresse.data.gouv.fr/search/?q=$citySearch&limit=1";
- $cityResponse = @file_get_contents($cityUrl);
-
- if ($cityResponse !== false) {
- $cityData = json_decode($cityResponse, true);
-
- if ($cityData && isset($cityData['features'][0]['geometry']['coordinates'])) {
- $coordinates = $cityData['features'][0]['geometry']['coordinates'];
-
- LogService::log('Utilisation des coordonnées du centre de la ville', [
- 'level' => 'info',
- 'city' => $cityName,
- 'postalCode' => $postalCode
- ]);
-
- return [
- 'lng' => $coordinates[0],
- 'lat' => $coordinates[1]
- ];
- }
- }
-
- // Aucune coordonnée trouvée
- return null;
- }
-}
diff --git a/api/src/Controllers/MigrationController.php b/api/src/Controllers/MigrationController.php
new file mode 100644
index 00000000..7c4a5002
--- /dev/null
+++ b/api/src/Controllers/MigrationController.php
@@ -0,0 +1,544 @@
+db = Database::getInstance();
+ $this->appConfig = AppConfig::getInstance();
+ $this->migrationService = new MigrationService();
+ }
+
+ /**
+ * Teste les connexions aux bases de données source et cible
+ *
+ * GET /api/migrations/test-connections
+ *
+ * @return void
+ */
+ public function testConnections(): void {
+ try {
+ $result = $this->migrationService->testConnections();
+
+ Response::json([
+ 'status' => 'success',
+ 'connections' => $result
+ ]);
+ } catch (Exception $e) {
+ LogService::log('Erreur lors du test des connexions', [
+ 'level' => 'error',
+ 'error' => $e->getMessage()
+ ]);
+
+ Response::json([
+ 'status' => 'error',
+ 'message' => $e->getMessage()
+ ], 500);
+ }
+ }
+
+ /**
+ * Liste les entités disponibles à migrer depuis la base source
+ *
+ * GET /api/migrations/entities/available
+ *
+ * @return void
+ */
+ public function getAvailableEntities(): void {
+ try {
+ $entities = $this->migrationService->getAvailableEntities();
+
+ Response::json([
+ 'status' => 'success',
+ 'count' => count($entities),
+ 'entities' => $entities
+ ]);
+ } catch (Exception $e) {
+ LogService::log('Erreur lors de la récupération des entités disponibles', [
+ 'level' => 'error',
+ 'error' => $e->getMessage()
+ ]);
+
+ Response::json([
+ 'status' => 'error',
+ 'message' => $e->getMessage()
+ ], 500);
+ }
+ }
+
+ /**
+ * Récupère les détails d'une entité source
+ *
+ * GET /api/migrations/entities/:id
+ *
+ * @param int $id ID de l'entité dans la base source
+ * @return void
+ */
+ public function getEntityDetails(int $id): void {
+ try {
+ $entity = $this->migrationService->getEntityDetails($id);
+
+ if (!$entity) {
+ Response::json([
+ 'status' => 'error',
+ 'message' => 'Entité non trouvée'
+ ], 404);
+ return;
+ }
+
+ Response::json([
+ 'status' => 'success',
+ 'entity' => $entity
+ ]);
+ } catch (Exception $e) {
+ LogService::log('Erreur lors de la récupération des détails de l\'entité', [
+ 'level' => 'error',
+ 'error' => $e->getMessage(),
+ 'entity_id' => $id
+ ]);
+
+ Response::json([
+ 'status' => 'error',
+ 'message' => $e->getMessage()
+ ], 500);
+ }
+ }
+
+ /**
+ * Migre une entité complète ou par étapes
+ *
+ * POST /api/migrations/entity
+ * Body: {
+ * "entity_id": 45,
+ * "steps": ["users", "operations"], // Optionnel
+ * "dry_run": false, // Optionnel
+ * "truncate": false // Optionnel
+ * }
+ *
+ * @return void
+ */
+ public function migrateEntity(): void {
+ try {
+ $data = Request::getJsonBody();
+
+ // Validation
+ if (!isset($data['entity_id'])) {
+ Response::json([
+ 'status' => 'error',
+ 'message' => 'Le champ entity_id est requis'
+ ], 400);
+ return;
+ }
+
+ $entityId = (int) $data['entity_id'];
+ $steps = $data['steps'] ?? null;
+ $dryRun = $data['dry_run'] ?? false;
+ $truncate = $data['truncate'] ?? false;
+
+ // Vérifier les permissions (admin uniquement)
+ $userRole = Session::get('fk_role');
+ if ($userRole != 3) { // 3 = admin
+ Response::json([
+ 'status' => 'error',
+ 'message' => 'Accès refusé. Cette action nécessite les droits administrateur.'
+ ], 403);
+ return;
+ }
+
+ LogService::log('Début de migration d\'entité', [
+ 'level' => 'info',
+ 'entity_id' => $entityId,
+ 'steps' => $steps,
+ 'dry_run' => $dryRun,
+ 'truncate' => $truncate,
+ 'user_id' => Session::get('user_id')
+ ]);
+
+ // Exécuter la migration
+ $result = $this->migrationService->migrateEntity(
+ $entityId,
+ $steps,
+ $dryRun,
+ $truncate
+ );
+
+ Response::json([
+ 'status' => 'success',
+ 'entity_id' => $entityId,
+ 'entity_name' => $result['entity_name'],
+ 'migration_id' => $result['migration_id'],
+ 'steps_completed' => $result['steps_completed'],
+ 'total_duration_ms' => $result['total_duration_ms'],
+ 'summary' => $result['summary']
+ ]);
+ } catch (Exception $e) {
+ LogService::log('Erreur lors de la migration d\'entité', [
+ 'level' => 'error',
+ 'error' => $e->getMessage(),
+ 'entity_id' => $data['entity_id'] ?? null
+ ]);
+
+ Response::json([
+ 'status' => 'error',
+ 'message' => $e->getMessage()
+ ], 500);
+ }
+ }
+
+ /**
+ * Migre une étape spécifique pour une entité
+ *
+ * POST /api/migrations/entity/step
+ * Body: {
+ * "entity_id": 45,
+ * "step": "users",
+ * "dry_run": false,
+ * "options": {}
+ * }
+ *
+ * @return void
+ */
+ public function migrateEntityStep(): void {
+ try {
+ $data = Request::getJsonBody();
+
+ // Validation
+ if (!isset($data['entity_id']) || !isset($data['step'])) {
+ Response::json([
+ 'status' => 'error',
+ 'message' => 'Les champs entity_id et step sont requis'
+ ], 400);
+ return;
+ }
+
+ $entityId = (int) $data['entity_id'];
+ $step = $data['step'];
+ $dryRun = $data['dry_run'] ?? false;
+ $options = $data['options'] ?? [];
+
+ // Vérifier les permissions (admin uniquement)
+ $userRole = Session::get('fk_role');
+ if ($userRole != 3) {
+ Response::json([
+ 'status' => 'error',
+ 'message' => 'Accès refusé. Cette action nécessite les droits administrateur.'
+ ], 403);
+ return;
+ }
+
+ LogService::log('Début de migration d\'étape', [
+ 'level' => 'info',
+ 'entity_id' => $entityId,
+ 'step' => $step,
+ 'dry_run' => $dryRun,
+ 'options' => $options,
+ 'user_id' => Session::get('user_id')
+ ]);
+
+ // Exécuter l'étape de migration
+ $result = $this->migrationService->migrateStep(
+ $entityId,
+ $step,
+ $dryRun,
+ $options
+ );
+
+ Response::json([
+ 'status' => 'success',
+ 'entity_id' => $entityId,
+ 'step' => $step,
+ 'records_migrated' => $result['records_migrated'],
+ 'duration_ms' => $result['duration_ms'],
+ 'warnings' => $result['warnings'] ?? [],
+ 'details' => $result['details'] ?? []
+ ]);
+ } catch (Exception $e) {
+ LogService::log('Erreur lors de la migration d\'étape', [
+ 'level' => 'error',
+ 'error' => $e->getMessage(),
+ 'entity_id' => $data['entity_id'] ?? null,
+ 'step' => $data['step'] ?? null
+ ]);
+
+ Response::json([
+ 'status' => 'error',
+ 'message' => $e->getMessage()
+ ], 500);
+ }
+ }
+
+ /**
+ * Récupère le statut de migration d'une entité
+ *
+ * GET /api/migrations/entity/:id/status
+ *
+ * @param int $id ID de l'entité
+ * @return void
+ */
+ public function getMigrationStatus(int $id): void {
+ try {
+ $status = $this->migrationService->getMigrationStatus($id);
+
+ Response::json([
+ 'status' => 'success',
+ 'entity_id' => $id,
+ 'migration_status' => $status
+ ]);
+ } catch (Exception $e) {
+ LogService::log('Erreur lors de la récupération du statut de migration', [
+ 'level' => 'error',
+ 'error' => $e->getMessage(),
+ 'entity_id' => $id
+ ]);
+
+ Response::json([
+ 'status' => 'error',
+ 'message' => $e->getMessage()
+ ], 500);
+ }
+ }
+
+ /**
+ * Récupère les logs de migration d'une entité
+ *
+ * GET /api/migrations/entity/:id/logs
+ *
+ * @param int $id ID de l'entité
+ * @return void
+ */
+ public function getMigrationLogs(int $id): void {
+ try {
+ $logs = $this->migrationService->getMigrationLogs($id);
+
+ Response::json([
+ 'status' => 'success',
+ 'entity_id' => $id,
+ 'logs' => $logs
+ ]);
+ } catch (Exception $e) {
+ LogService::log('Erreur lors de la récupération des logs de migration', [
+ 'level' => 'error',
+ 'error' => $e->getMessage(),
+ 'entity_id' => $id
+ ]);
+
+ Response::json([
+ 'status' => 'error',
+ 'message' => $e->getMessage()
+ ], 500);
+ }
+ }
+
+ /**
+ * Génère un rapport de migration pour une entité
+ *
+ * GET /api/migrations/entity/:id/report
+ *
+ * @param int $id ID de l'entité
+ * @return void
+ */
+ public function getMigrationReport(int $id): void {
+ try {
+ $report = $this->migrationService->generateMigrationReport($id);
+
+ Response::json([
+ 'status' => 'success',
+ 'entity_id' => $id,
+ 'report' => $report
+ ]);
+ } catch (Exception $e) {
+ LogService::log('Erreur lors de la génération du rapport de migration', [
+ 'level' => 'error',
+ 'error' => $e->getMessage(),
+ 'entity_id' => $id
+ ]);
+
+ Response::json([
+ 'status' => 'error',
+ 'message' => $e->getMessage()
+ ], 500);
+ }
+ }
+
+ /**
+ * Compare les données source vs cible pour une entité
+ *
+ * GET /api/migrations/entity/:id/compare
+ *
+ * @param int $id ID de l'entité
+ * @return void
+ */
+ public function compareEntityData(int $id): void {
+ try {
+ $comparison = $this->migrationService->compareEntityData($id);
+
+ Response::json([
+ 'status' => 'success',
+ 'entity_id' => $id,
+ 'comparison' => $comparison
+ ]);
+ } catch (Exception $e) {
+ LogService::log('Erreur lors de la comparaison des données', [
+ 'level' => 'error',
+ 'error' => $e->getMessage(),
+ 'entity_id' => $id
+ ]);
+
+ Response::json([
+ 'status' => 'error',
+ 'message' => $e->getMessage()
+ ], 500);
+ }
+ }
+
+ /**
+ * Vérifie l'intégrité des données migrées pour une entité
+ *
+ * GET /api/migrations/entity/:id/verify
+ *
+ * @param int $id ID de l'entité
+ * @return void
+ */
+ public function verifyMigration(int $id): void {
+ try {
+ $verification = $this->migrationService->verifyMigration($id);
+
+ Response::json([
+ 'status' => 'success',
+ 'entity_id' => $id,
+ 'verification' => $verification
+ ]);
+ } catch (Exception $e) {
+ LogService::log('Erreur lors de la vérification de la migration', [
+ 'level' => 'error',
+ 'error' => $e->getMessage(),
+ 'entity_id' => $id
+ ]);
+
+ Response::json([
+ 'status' => 'error',
+ 'message' => $e->getMessage()
+ ], 500);
+ }
+ }
+
+ /**
+ * Annule la migration d'une entité (rollback)
+ *
+ * DELETE /api/migrations/entity/:id
+ *
+ * @param int $id ID de l'entité
+ * @return void
+ */
+ public function rollbackEntity(int $id): void {
+ try {
+ // Vérifier les permissions (admin uniquement)
+ $userRole = Session::get('fk_role');
+ if ($userRole != 3) {
+ Response::json([
+ 'status' => 'error',
+ 'message' => 'Accès refusé. Cette action nécessite les droits administrateur.'
+ ], 403);
+ return;
+ }
+
+ LogService::log('Début de rollback d\'entité', [
+ 'level' => 'warning',
+ 'entity_id' => $id,
+ 'user_id' => Session::get('user_id')
+ ]);
+
+ $result = $this->migrationService->rollbackEntity($id);
+
+ Response::json([
+ 'status' => 'success',
+ 'entity_id' => $id,
+ 'message' => 'Migration annulée avec succès',
+ 'deleted_records' => $result['deleted_records']
+ ]);
+ } catch (Exception $e) {
+ LogService::log('Erreur lors du rollback de la migration', [
+ 'level' => 'error',
+ 'error' => $e->getMessage(),
+ 'entity_id' => $id
+ ]);
+
+ Response::json([
+ 'status' => 'error',
+ 'message' => $e->getMessage()
+ ], 500);
+ }
+ }
+
+ /**
+ * Supprime une étape spécifique de la migration
+ *
+ * DELETE /api/migrations/entity/:id/step/:step
+ *
+ * @param int $id ID de l'entité
+ * @param string $step Nom de l'étape
+ * @return void
+ */
+ public function rollbackStep(int $id, string $step): void {
+ try {
+ // Vérifier les permissions (admin uniquement)
+ $userRole = Session::get('fk_role');
+ if ($userRole != 3) {
+ Response::json([
+ 'status' => 'error',
+ 'message' => 'Accès refusé. Cette action nécessite les droits administrateur.'
+ ], 403);
+ return;
+ }
+
+ LogService::log('Début de rollback d\'étape', [
+ 'level' => 'warning',
+ 'entity_id' => $id,
+ 'step' => $step,
+ 'user_id' => Session::get('user_id')
+ ]);
+
+ $result = $this->migrationService->rollbackStep($id, $step);
+
+ Response::json([
+ 'status' => 'success',
+ 'entity_id' => $id,
+ 'step' => $step,
+ 'message' => 'Étape annulée avec succès',
+ 'deleted_records' => $result['deleted_records']
+ ]);
+ } catch (Exception $e) {
+ LogService::log('Erreur lors du rollback de l\'étape', [
+ 'level' => 'error',
+ 'error' => $e->getMessage(),
+ 'entity_id' => $id,
+ 'step' => $step
+ ]);
+
+ Response::json([
+ 'status' => 'error',
+ 'message' => $e->getMessage()
+ ], 500);
+ }
+ }
+}
diff --git a/api/src/Controllers/OperationController.php b/api/src/Controllers/OperationController.php
index 5bb9f0ed..4a83a945 100755
--- a/api/src/Controllers/OperationController.php
+++ b/api/src/Controllers/OperationController.php
@@ -5,6 +5,7 @@ declare(strict_types=1);
namespace App\Controllers;
require_once __DIR__ . '/../Services/LogService.php';
+require_once __DIR__ . '/../Services/EventLogService.php';
require_once __DIR__ . '/../Services/ExportService.php';
require_once __DIR__ . '/../Services/ApiService.php';
require_once __DIR__ . '/../Services/OperationDataService.php';
@@ -16,10 +17,11 @@ use AppConfig;
use Request;
use Response;
use Session;
-use LogService;
-use ExportService;
-use ApiService;
-use OperationDataService;
+use App\Services\LogService;
+use App\Services\EventLogService;
+use App\Services\ExportService;
+use App\Services\ApiService;
+use App\Services\OperationDataService;
use Exception;
use DateTime;
@@ -378,34 +380,37 @@ class OperationController {
$newSectId = (int)$this->db->lastInsertId();
$duplicatedSectors++;
- // Étape 4.3 : Dupliquer les users_sectors en vérifiant que fk_user existe dans ope_users
+ // Étape 4.3 : Dupliquer les users_sectors en convertissant ancien ope_users.id → nouvel ope_users.id
$stmt = $this->db->prepare('
INSERT INTO ope_users_sectors (fk_operation, fk_user, fk_sector, fk_user_creat)
- SELECT ?, ous.fk_user, ?, ?
+ SELECT ?, new_ou.id, ?, ?
FROM ope_users_sectors ous
- INNER JOIN ope_users ou ON ou.fk_user = ous.fk_user AND ou.fk_operation = ?
+ INNER JOIN ope_users old_ou ON old_ou.id = ous.fk_user AND old_ou.fk_operation = ?
+ INNER JOIN ope_users new_ou ON new_ou.fk_user = old_ou.fk_user AND new_ou.fk_operation = ?
WHERE ous.fk_operation = ? AND ous.fk_sector = ? AND ous.chk_active = 1
');
- $stmt->execute([$newOpeId, $newSectId, $userId, $newOpeId, $oldOpeId, $oldSectId]);
+ $stmt->execute([$newOpeId, $newSectId, $userId, $oldOpeId, $newOpeId, $oldOpeId, $oldSectId]);
$duplicatedUsersSectors += $stmt->rowCount();
- // Étape 4.4 : Dupliquer les passages avec les valeurs par défaut spécifiées
+ // Étape 4.4 : Dupliquer les passages en convertissant ancien ope_users.id → nouvel ope_users.id
$stmt = $this->db->prepare('
INSERT INTO ope_pass (
- fk_operation, fk_sector, fk_user, fk_adresse, numero, rue, rue_bis, ville,
+ fk_operation, fk_sector, fk_user, fk_adresse, numero, rue, rue_bis, ville,
fk_habitat, appt, niveau, residence, gps_lat, gps_lng, encrypted_name,
- fk_type, passed_at, montant, fk_type_reglement, chk_email_sent, chk_striped,
- docremis, nb_passages, chk_map_create, chk_mobile, chk_synchro, anomalie,
+ fk_type, passed_at, montant, fk_type_reglement, chk_email_sent, chk_striped,
+ docremis, nb_passages, chk_map_create, chk_mobile, chk_synchro, anomalie,
fk_user_creat, chk_active
)
- SELECT
- ?, ?, fk_user, fk_adresse, numero, rue, rue_bis, ville,
- fk_habitat, appt, niveau, residence, gps_lat, gps_lng, encrypted_name,
- 2, NULL, 0, 4, 0, 0, 0, 1, 0, 0, 1, 0, ?, 1
- FROM ope_pass
- WHERE fk_operation = ? AND fk_sector = ? AND chk_active = 1
+ SELECT
+ ?, ?, new_ou.id, op.fk_adresse, op.numero, op.rue, op.rue_bis, op.ville,
+ op.fk_habitat, op.appt, op.niveau, op.residence, op.gps_lat, op.gps_lng, op.encrypted_name,
+ 2, NULL, 0, 4, 0, 0, 0, 0, 0, 0, 1, 0, ?, 1
+ FROM ope_pass op
+ INNER JOIN ope_users old_ou ON old_ou.id = op.fk_user AND old_ou.fk_operation = ?
+ INNER JOIN ope_users new_ou ON new_ou.fk_user = old_ou.fk_user AND new_ou.fk_operation = ?
+ WHERE op.fk_operation = ? AND op.fk_sector = ? AND op.chk_active = 1
');
- $stmt->execute([$newOpeId, $newSectId, $userId, $oldOpeId, $oldSectId]);
+ $stmt->execute([$newOpeId, $newSectId, $userId, $oldOpeId, $newOpeId, $oldOpeId, $oldSectId]);
$duplicatedPassages += $stmt->rowCount();
}
@@ -455,19 +460,12 @@ class OperationController {
// Étape 7 : Préparer la réponse avec les groupes JSON
$response = OperationDataService::prepareOperationResponse($this->db, $newOpeId, $entiteId);
- LogService::log('Création opération terminée avec succès', [
- 'level' => 'info',
- 'userId' => $userId,
- 'entiteId' => $entiteId,
- 'newOpeId' => $newOpeId,
- 'oldOpeId' => $oldOpeId,
- 'stats' => [
- 'insertedUsers' => $insertedUsers,
- 'duplicatedSectors' => $duplicatedSectors,
- 'duplicatedUsersSectors' => $duplicatedUsersSectors,
- 'duplicatedPassages' => $duplicatedPassages
- ]
- ]);
+ // Log de création de l'opération
+ EventLogService::logOperationCreated(
+ $newOpeId,
+ $data['date_deb'],
+ $data['date_fin']
+ );
Response::json($response, 201);
} catch (Exception $e) {
@@ -621,12 +619,24 @@ class OperationController {
$operationId
]);
- LogService::log('Mise à jour d\'une opération', [
- 'level' => 'info',
- 'userId' => $userId,
- 'entiteId' => $entiteId,
- 'operationId' => $operationId
- ]);
+ // Log de mise à jour de l'opération
+ $changes = [];
+ if (isset($data['libelle']) || isset($data['name'])) {
+ $changes['libelle'] = ['new' => $libelle];
+ }
+ if (isset($data['date_deb'])) {
+ $changes['date_deb'] = ['new' => $data['date_deb']];
+ }
+ if (isset($data['date_fin'])) {
+ $changes['date_fin'] = ['new' => $data['date_fin']];
+ }
+ if (isset($data['chk_distinct_sectors'])) {
+ $changes['chk_distinct_sectors'] = ['new' => (int)$data['chk_distinct_sectors']];
+ }
+
+ if (!empty($changes)) {
+ EventLogService::logOperationUpdated($operationId, $changes);
+ }
Response::json([
'status' => 'success',
@@ -820,25 +830,8 @@ class OperationController {
// Valider la transaction
$this->db->commit();
- LogService::log('Suppression complète d\'une opération et de toutes ses données', [
- 'level' => 'info',
- 'userId' => $userId,
- 'userRole' => $userRole,
- 'userEntiteId' => $userEntiteId,
- 'operationEntiteId' => $operationEntiteId,
- 'operationId' => $operationId,
- 'operationActive' => $operationActive,
- 'deletedCounts' => [
- 'medias' => $deletedMedias,
- 'ope_pass_histo' => $deletedPassHisto,
- 'ope_pass' => $deletedPass,
- 'ope_users_sectors' => $deletedUsersSectors,
- 'sectors_adresses' => $deletedSectorsAdresses,
- 'ope_sectors' => $deletedSectors,
- 'ope_users' => $deletedUsers,
- 'operations' => 1
- ]
- ]);
+ // Log de suppression de l'opération (suppression physique)
+ EventLogService::logOperationDeleted($operationId, false);
// Préparer la réponse selon le statut de l'opération supprimée
$response = [
@@ -948,13 +941,14 @@ class OperationController {
// Récupérer les relations utilisateurs-secteurs
$stmt = $this->db->prepare('
- SELECT
+ SELECT
ous.id, ous.fk_operation, ous.fk_user, ous.fk_sector,
ous.created_at, ous.updated_at, ous.chk_active,
- u.encrypted_name as user_name, u.first_name as user_first_name,
+ u.encrypted_name as user_name, ou.first_name as user_first_name,
s.libelle as sector_name
FROM ope_users_sectors ous
- INNER JOIN users u ON u.id = ous.fk_user
+ INNER JOIN ope_users ou ON ou.id = ous.fk_user
+ INNER JOIN users u ON u.id = ou.fk_user
INNER JOIN ope_sectors s ON s.id = ous.fk_sector
WHERE ous.fk_operation = ? AND ous.chk_active = 1
ORDER BY s.libelle, u.encrypted_name
diff --git a/api/src/Controllers/PassageController.php b/api/src/Controllers/PassageController.php
index f5a3d1fa..da50e83d 100755
--- a/api/src/Controllers/PassageController.php
+++ b/api/src/Controllers/PassageController.php
@@ -5,6 +5,7 @@ declare(strict_types=1);
namespace App\Controllers;
require_once __DIR__ . '/../Services/LogService.php';
+require_once __DIR__ . '/../Services/EventLogService.php';
require_once __DIR__ . '/../Services/ApiService.php';
require_once __DIR__ . '/../Services/ReceiptService.php';
@@ -15,8 +16,9 @@ use AppConfig;
use Request;
use Response;
use Session;
-use LogService;
-use ApiService;
+use App\Services\LogService;
+use App\Services\EventLogService;
+use App\Services\ApiService;
use Exception;
use DateTime;
@@ -233,13 +235,14 @@ class PassageController {
p.fk_habitat, p.appt, p.niveau, p.residence, p.gps_lat, p.gps_lng,
p.encrypted_name, p.montant, p.fk_type_reglement, p.remarque,
p.encrypted_email, p.encrypted_phone, p.nom_recu, p.date_recu,
- p.chk_email_sent, p.stripe_payment_id, p.docremis, p.date_repasser, p.nb_passages,
+ p.chk_email_sent, p.stripe_payment_id, p.stripe_payment_link_id, p.docremis, p.date_repasser, p.nb_passages,
p.chk_mobile, p.anomalie, p.created_at, p.updated_at, p.chk_active,
o.libelle as operation_libelle,
- u.encrypted_name as user_name, u.first_name as user_first_name
+ u.encrypted_name as user_name, ou.first_name as user_first_name
FROM ope_pass p
INNER JOIN operations o ON p.fk_operation = o.id
- INNER JOIN users u ON p.fk_user = u.id
+ INNER JOIN ope_users ou ON p.fk_user = ou.id
+ INNER JOIN users u ON ou.fk_user = u.id
WHERE $whereClause AND p.chk_active = 1
ORDER BY p.created_at DESC
LIMIT ? OFFSET ?
@@ -324,13 +327,14 @@ class PassageController {
$passageId = (int)$id;
$stmt = $this->db->prepare('
- SELECT
- p.*,
+ SELECT
+ p.*,
o.libelle as operation_libelle,
- u.encrypted_name as user_name, u.first_name as user_first_name
+ u.encrypted_name as user_name, ou.first_name as user_first_name
FROM ope_pass p
INNER JOIN operations o ON p.fk_operation = o.id
- INNER JOIN users u ON p.fk_user = u.id
+ INNER JOIN ope_users ou ON p.fk_user = ou.id
+ INNER JOIN users u ON ou.fk_user = u.id
WHERE p.id = ? AND o.fk_entite = ? AND p.chk_active = 1
');
@@ -410,12 +414,13 @@ class PassageController {
p.id, p.fk_operation, p.fk_sector, p.fk_user, p.passed_at,
p.numero, p.rue, p.rue_bis, p.ville, p.gps_lat, p.gps_lng,
p.encrypted_name, p.montant, p.fk_type_reglement, p.remarque,
- p.encrypted_email, p.encrypted_phone, p.stripe_payment_id, p.chk_email_sent,
+ p.encrypted_email, p.encrypted_phone, p.stripe_payment_id, p.stripe_payment_link_id, p.chk_email_sent,
p.docremis, p.date_repasser, p.nb_passages, p.chk_mobile,
p.anomalie, p.created_at, p.updated_at,
- u.encrypted_name as user_name, u.first_name as user_first_name
+ u.encrypted_name as user_name, ou.first_name as user_first_name
FROM ope_pass p
- INNER JOIN users u ON p.fk_user = u.id
+ INNER JOIN ope_users ou ON p.fk_user = ou.id
+ INNER JOIN users u ON ou.fk_user = u.id
WHERE p.fk_operation = ? AND p.chk_active = 1
ORDER BY p.created_at DESC
LIMIT ? OFFSET ?
@@ -510,6 +515,24 @@ class PassageController {
return;
}
+ // Récupérer ope_users.id pour l'utilisateur du passage
+ // $data['fk_user'] contient users.id, on doit le convertir en ope_users.id
+ $passageUserId = (int)$data['fk_user'];
+ $stmtOpeUser = $this->db->prepare('
+ SELECT id FROM ope_users
+ WHERE fk_user = ? AND fk_operation = ?
+ ');
+ $stmtOpeUser->execute([$passageUserId, $operationId]);
+ $opeUserId = $stmtOpeUser->fetchColumn();
+
+ if (!$opeUserId) {
+ Response::json([
+ 'status' => 'error',
+ 'message' => 'Utilisateur non trouvé dans cette opération'
+ ], 404);
+ return;
+ }
+
// Chiffrement des données sensibles
$encryptedName = '';
if (isset($data['name']) && !empty(trim($data['name']))) {
@@ -527,7 +550,7 @@ class PassageController {
$insertData = [
'fk_operation' => $operationId,
'fk_sector' => isset($data['fk_sector']) ? (int)$data['fk_sector'] : 0,
- 'fk_user' => (int)$data['fk_user'],
+ 'fk_user' => $opeUserId,
'fk_adresse' => $data['fk_adresse'] ?? '',
'passed_at' => isset($data['passed_at']) ? $data['passed_at'] : null,
'fk_type' => isset($data['fk_type']) ? (int)$data['fk_type'] : 0,
@@ -569,12 +592,14 @@ class PassageController {
$passageId = $this->db->lastInsertId();
- LogService::log('Création d\'un nouveau passage', [
- 'level' => 'info',
- 'userId' => $userId,
- 'passageId' => $passageId,
- 'operationId' => $operationId
- ]);
+ // Log de création du passage
+ EventLogService::logPassageCreated(
+ (int)$passageId,
+ $insertData['fk_operation'],
+ $insertData['fk_sector'],
+ $insertData['montant'],
+ (string)$insertData['fk_type_reglement']
+ );
// Enregistrer la génération du reçu dans shutdown_function pour garantir son exécution
// Même si le worker FPM est tué après fastcgi_finish_request()
@@ -702,16 +727,33 @@ class PassageController {
return;
}
+ // Récupérer ope_users.id pour l'utilisateur connecté
+ $operationId = $passage['fk_operation'];
+ $stmtCurrentOpeUser = $this->db->prepare('
+ SELECT id FROM ope_users
+ WHERE fk_user = ? AND fk_operation = ?
+ ');
+ $stmtCurrentOpeUser->execute([$userId, $operationId]);
+ $currentOpeUserId = $stmtCurrentOpeUser->fetchColumn();
+
+ if (!$currentOpeUserId) {
+ Response::json([
+ 'status' => 'error',
+ 'message' => 'Utilisateur connecté non trouvé dans cette opération'
+ ], 404);
+ return;
+ }
+
// Si le passage était de type 2 et que l'utilisateur actuel est différent du créateur
// On force l'attribution du passage à l'utilisateur actuel
- if ((int)$passage['fk_type'] === 2 && (int)$passage['fk_user'] !== $userId) {
- $data['fk_user'] = $userId;
+ if ((int)$passage['fk_type'] === 2 && (int)$passage['fk_user'] !== $currentOpeUserId) {
+ $data['fk_user'] = $currentOpeUserId;
LogService::log('Attribution automatique d\'un passage type 2 à l\'utilisateur', [
'level' => 'info',
'passageId' => $passageId,
'ancien_user' => $passage['fk_user'],
- 'nouveau_user' => $userId
+ 'nouveau_user' => $currentOpeUserId
]);
}
@@ -722,7 +764,7 @@ class PassageController {
// Champs pouvant être mis à jour
$updatableFields = [
'fk_sector',
- 'fk_user',
+ // Note: fk_user est traité séparément pour conversion users.id -> ope_users.id
'fk_adresse',
'passed_at',
'fk_type',
@@ -740,6 +782,7 @@ class PassageController {
'fk_type_reglement',
'remarque',
'stripe_payment_id',
+ 'stripe_payment_link_id',
'nom_recu',
'date_recu',
'docremis',
@@ -756,6 +799,48 @@ class PassageController {
}
}
+ // Traitement spécial pour fk_user : conversion users.id -> ope_users.id
+ if (isset($data['fk_user'])) {
+ // Si $data['fk_user'] vient de l'attribution automatique, c'est déjà ope_users.id
+ // Sinon, on doit convertir users.id en ope_users.id
+ $providedUserId = (int)$data['fk_user'];
+
+ // Vérifier si c'est déjà un ope_users.id valide
+ $stmtCheckOpeUser = $this->db->prepare('
+ SELECT id FROM ope_users
+ WHERE id = ? AND fk_operation = ?
+ ');
+ $stmtCheckOpeUser->execute([$providedUserId, $operationId]);
+ $isOpeUserId = $stmtCheckOpeUser->fetchColumn();
+
+ if ($isOpeUserId) {
+ // C'est déjà un ope_users.id valide
+ $updateFields[] = "fk_user = ?";
+ $params[] = $providedUserId;
+ } else {
+ // C'est probablement un users.id, on le convertit
+ $stmtGetOpeUser = $this->db->prepare('
+ SELECT id FROM ope_users
+ WHERE fk_user = ? AND fk_operation = ?
+ ');
+ $stmtGetOpeUser->execute([$providedUserId, $operationId]);
+ $convertedOpeUserId = $stmtGetOpeUser->fetchColumn();
+
+ if ($convertedOpeUserId) {
+ $updateFields[] = "fk_user = ?";
+ $params[] = $convertedOpeUserId;
+ } else {
+ // Utilisateur non trouvé, on ignore cette mise à jour
+ LogService::log('Tentative de mise à jour avec un utilisateur invalide', [
+ 'level' => 'warning',
+ 'passageId' => $passageId,
+ 'provided_user_id' => $providedUserId,
+ 'operation_id' => $operationId
+ ]);
+ }
+ }
+ }
+
// Gestion des champs chiffrés
if (array_key_exists('name', $data)) {
$updateFields[] = "encrypted_name = ?";
@@ -791,11 +876,21 @@ class PassageController {
$stmt = $this->db->prepare($sql);
$stmt->execute($params);
- LogService::log('Mise à jour d\'un passage', [
- 'level' => 'info',
- 'userId' => $userId,
- 'passageId' => $passageId
- ]);
+ // Log de mise à jour du passage (changements simplifiés)
+ $changes = [];
+ foreach ($data as $key => $value) {
+ // Ne logger que les champs non sensibles
+ if (!in_array($key, ['name', 'email', 'phone', 'encrypted_name', 'encrypted_email', 'encrypted_phone'])) {
+ $changes[$key] = ['new' => $value];
+ } else {
+ // Indiquer qu'un champ chiffré a été modifié
+ $changes[$key] = true;
+ }
+ }
+
+ if (!empty($changes)) {
+ EventLogService::logPassageUpdated((int)$passageId, $changes);
+ }
// Enregistrer la génération du reçu dans shutdown_function pour garantir son exécution
// Même si le worker FPM est tué après fastcgi_finish_request()
@@ -944,7 +1039,7 @@ class PassageController {
}
$stmt = $this->db->prepare('
- SELECT p.id
+ SELECT p.id, p.fk_operation
FROM ope_pass p
INNER JOIN operations o ON p.fk_operation = o.id
WHERE p.id = ? AND o.fk_entite = ? AND p.chk_active = 1
@@ -962,18 +1057,19 @@ class PassageController {
// Désactiver le passage (soft delete)
$stmt = $this->db->prepare('
- UPDATE ope_pass
+ UPDATE ope_pass
SET chk_active = 0, updated_at = NOW(), fk_user_modif = ?
WHERE id = ?
');
$stmt->execute([$userId, $passageId]);
- LogService::log('Suppression d\'un passage', [
- 'level' => 'info',
- 'userId' => $userId,
- 'passageId' => $passageId
- ]);
+ // Log de suppression du passage
+ EventLogService::logPassageDeleted(
+ $passageId,
+ (int)$passage['fk_operation'],
+ true // soft delete
+ );
Response::json([
'status' => 'success',
diff --git a/api/src/Controllers/PasswordController.php b/api/src/Controllers/PasswordController.php
index 3d01fd7d..3a42980d 100644
--- a/api/src/Controllers/PasswordController.php
+++ b/api/src/Controllers/PasswordController.php
@@ -9,7 +9,7 @@ require_once __DIR__ . '/../Services/LogService.php';
use Request;
use Response;
-use LogService;
+use App\Services\LogService;
use App\Services\PasswordSecurityService;
/**
diff --git a/api/src/Controllers/SectorController.php b/api/src/Controllers/SectorController.php
index a777bfea..0d2711b2 100644
--- a/api/src/Controllers/SectorController.php
+++ b/api/src/Controllers/SectorController.php
@@ -3,14 +3,14 @@ namespace App\Controllers;
use Database;
use Response;
-use LogService;
-use ApiService;
-use AddressService;
-use DepartmentBoundaryService;
+use App\Services\LogService;
+use App\Services\EventLogService;
+use App\Services\ApiService;
+use App\Services\AddressService;
+use App\Services\DepartmentBoundaryService;
+require_once __DIR__ . '/../Services/EventLogService.php';
require_once __DIR__ . '/../Services/ApiService.php';
-require_once __DIR__ . '/../Services/AddressService.php';
-require_once __DIR__ . '/../Services/DepartmentBoundaryService.php';
class SectorController
{
@@ -193,14 +193,31 @@ class SectorController
// Affectation des users si fournis
if (!empty($users)) {
- $queryMember = "INSERT INTO ope_users_sectors (fk_operation, fk_user, fk_sector, created_at, fk_user_creat, chk_active)
- VALUES (:operation_id, :user_id, :sector_id, NOW(), :user_creat, 1)";
+ $queryMember = "INSERT INTO ope_users_sectors (fk_operation, fk_user, fk_sector, created_at, updated_at, fk_user_creat, chk_active)
+ VALUES (:operation_id, :user_id, :sector_id, NOW(), NOW(), :user_creat, 1)";
$stmtMember = $this->db->prepare($queryMember);
-
+
foreach ($users as $memberId) {
+ // $memberId est DÉJÀ ope_users.id (envoyé par Flutter)
+ // Vérifier que cet ope_users.id existe et appartient bien à l'opération
+ $stmtOpeUser = $this->db->prepare('
+ SELECT id FROM ope_users
+ WHERE id = ? AND fk_operation = ?
+ ');
+ $stmtOpeUser->execute([$memberId, $operationId]);
+ $opeUserId = $stmtOpeUser->fetchColumn();
+
+ if (!$opeUserId) {
+ $this->logService->warning('ope_users.id non trouvé pour cette opération', [
+ 'ope_users_id' => $memberId,
+ 'operation_id' => $operationId
+ ]);
+ continue;
+ }
+
$stmtMember->execute([
'operation_id' => $operationId,
- 'user_id' => $memberId,
+ 'user_id' => $opeUserId,
'sector_id' => $sectorId,
'user_creat' => $userId
]);
@@ -268,16 +285,24 @@ class SectorController
$passagesCreated = 0; // Initialiser le compteur de passages
try {
$addresses = $this->addressService->getAddressesInPolygon($coordinates, $entityId);
-
+
+ // Enrichir les adresses avec les données bâtiments
+ $addresses = $this->addressService->enrichAddressesWithBuildings($addresses, $entityId);
+
if (!empty($addresses)) {
- $queryAddress = "INSERT INTO sectors_adresses (fk_sector, fk_adresse, numero, rue, rue_bis, cp, ville, gps_lat, gps_lng)
- VALUES (:sector_id, :address_id, :numero, :rue, :rue_bis, :cp, :ville, :gps_lat, :gps_lng)";
+ $queryAddress = "INSERT INTO sectors_adresses (
+ fk_sector, fk_adresse, numero, rue, rue_bis, cp, ville, gps_lat, gps_lng,
+ fk_batiment, fk_habitat, nb_niveau, nb_log, residence, alt_sol
+ ) VALUES (
+ :sector_id, :address_id, :numero, :rue, :rue_bis, :cp, :ville, :gps_lat, :gps_lng,
+ :fk_batiment, :fk_habitat, :nb_niveau, :nb_log, :residence, :alt_sol
+ )";
$stmtAddress = $this->db->prepare($queryAddress);
-
+
foreach ($addresses as $address) {
// Extraire le rue_bis si présent (généralement vide)
$rueBis = '';
-
+
$stmtAddress->execute([
'sector_id' => $sectorId,
'address_id' => $address['id'],
@@ -287,60 +312,111 @@ class SectorController
'cp' => $address['code_postal'],
'ville' => $address['commune'],
'gps_lat' => $address['latitude'],
- 'gps_lng' => $address['longitude']
+ 'gps_lng' => $address['longitude'],
+ 'fk_batiment' => $address['fk_batiment'] ?? null,
+ 'fk_habitat' => $address['fk_habitat'] ?? 1,
+ 'nb_niveau' => $address['nb_niveau'] ?? null,
+ 'nb_log' => $address['nb_log'] ?? null,
+ 'residence' => $address['residence'] ?? '',
+ 'alt_sol' => $address['alt_sol'] ?? null
]);
}
// Créer les passages pour chaque adresse
if (!empty($users)) {
- $firstUserId = $users[0]; // Premier user pour l'affectation des passages
- $passageQuery = "INSERT INTO ope_pass (
- fk_operation, fk_sector, fk_user, fk_adresse,
- numero, rue, rue_bis, ville,
- gps_lat, gps_lng, fk_type, encrypted_name,
- created_at, fk_user_creat, chk_active
- ) VALUES (
- :operation_id, :sector_id, :user_id, :fk_adresse,
- :numero, :rue, :rue_bis, :ville,
- :gps_lat, :gps_lng, 2, '',
- NOW(), :user_creat, 1
- )";
- $passageStmt = $this->db->prepare($passageQuery);
-
- $passagesCreated = 0;
- foreach ($addresses as $address) {
- // Vérifier si cette adresse n'est pas déjà utilisée par un passage orphelin
- if (in_array($address['id'], $addressesToExclude)) {
- continue; // Passer à l'adresse suivante
- }
-
- try {
- // Extraire le rue_bis si présent (généralement vide)
- $rueBis = '';
-
- $passageStmt->execute([
- 'operation_id' => $operationId,
- 'sector_id' => $sectorId,
- 'user_id' => $firstUserId,
- 'fk_adresse' => $address['id'],
- 'numero' => $address['numero'],
- 'rue' => $address['voie'],
- 'rue_bis' => $rueBis,
- 'ville' => $address['commune'],
- 'gps_lat' => $address['latitude'],
- 'gps_lng' => $address['longitude'],
- 'user_creat' => $userId
- ]);
- $passagesCreated++;
- } catch (\Exception $e) {
- $this->logService->warning('Erreur lors de la création d\'un passage', [
- 'address_id' => $address['id'],
- 'error' => $e->getMessage()
- ]);
+ // Récupérer ope_users.id pour le premier utilisateur
+ // $users[0] est DÉJÀ ope_users.id (envoyé par Flutter)
+ $stmtFirstOpeUser = $this->db->prepare('
+ SELECT id FROM ope_users
+ WHERE id = ? AND fk_operation = ?
+ ');
+ $stmtFirstOpeUser->execute([$users[0], $operationId]);
+ $firstOpeUserId = $stmtFirstOpeUser->fetchColumn();
+
+ if (!$firstOpeUserId) {
+ $this->logService->warning('Premier ope_users.id non trouvé pour cette opération', [
+ 'ope_users_id' => $users[0],
+ 'operation_id' => $operationId
+ ]);
+ // Pas de création de passages sans utilisateur valide dans ope_users
+ } else {
+ $passageQuery = "INSERT INTO ope_pass (
+ fk_operation, fk_sector, fk_user, fk_adresse,
+ numero, rue, rue_bis, ville, residence, appt, fk_habitat,
+ gps_lat, gps_lng, fk_type, nb_passages, encrypted_name,
+ created_at, fk_user_creat, chk_active
+ ) VALUES (
+ :operation_id, :sector_id, :user_id, :fk_adresse,
+ :numero, :rue, :rue_bis, :ville, :residence, :appt, :fk_habitat,
+ :gps_lat, :gps_lng, 2, 0, '',
+ NOW(), :user_creat, 1
+ )";
+ $passageStmt = $this->db->prepare($passageQuery);
+
+ $passagesCreated = 0;
+ foreach ($addresses as $address) {
+ // Vérifier si cette adresse n'est pas déjà utilisée par un passage orphelin
+ if (in_array($address['id'], $addressesToExclude)) {
+ continue; // Passer à l'adresse suivante
+ }
+
+ try {
+ // Extraire le rue_bis si présent (généralement vide)
+ $rueBis = '';
+
+ // Déterminer le nombre de passages à créer
+ $fkHabitat = $address['fk_habitat'] ?? 1;
+ $nbLog = ($fkHabitat == 2 && isset($address['nb_log'])) ? (int)$address['nb_log'] : 1;
+ $residence = $address['residence'] ?? '';
+
+ // IMPORTANT : Uniformisation GPS pour les immeubles
+ // Tous les passages d'une même adresse partagent les mêmes coordonnées GPS
+ // Issues de la table adresses enrichie (gps_lat, gps_lng)
+ $gpsLat = $address['latitude'];
+ $gpsLng = $address['longitude'];
+
+ // Créer 1 passage pour maison individuelle, nb_log passages pour immeuble
+ for ($i = 1; $i <= $nbLog; $i++) {
+ $appt = ($fkHabitat == 2) ? (string)$i : ''; // Numéro d'appartement pour immeubles
+
+ $passageStmt->execute([
+ 'operation_id' => $operationId,
+ 'sector_id' => $sectorId,
+ 'user_id' => $firstOpeUserId,
+ 'fk_adresse' => $address['id'],
+ 'numero' => $address['numero'],
+ 'rue' => $address['voie'],
+ 'rue_bis' => $rueBis,
+ 'ville' => $address['commune'],
+ 'residence' => $residence,
+ 'appt' => $appt,
+ 'fk_habitat' => $fkHabitat,
+ 'gps_lat' => $gpsLat,
+ 'gps_lng' => $gpsLng,
+ 'user_creat' => $userId
+ ]);
+ $passagesCreated++;
+ }
+
+ // Log pour vérifier l'uniformisation GPS (surtout pour immeubles)
+ if ($fkHabitat == 2 && $nbLog > 1) {
+ $this->logService->info('[SectorController] Création passages immeuble avec GPS uniformisés', [
+ 'address_id' => $address['id'],
+ 'nb_passages' => $nbLog,
+ 'gps_lat' => $gpsLat,
+ 'gps_lng' => $gpsLng,
+ 'residence' => $residence
+ ]);
+ }
+ } catch (\Exception $e) {
+ $this->logService->warning('Erreur lors de la création d\'un passage', [
+ 'address_id' => $address['id'],
+ 'error' => $e->getMessage()
+ ]);
+ }
}
}
-
}
}
} catch (\Exception $e) {
@@ -351,9 +427,16 @@ class SectorController
'entity_id' => $entityId
]);
}
-
+
$this->db->commit();
-
+
+ // Log de création du secteur
+ EventLogService::logSectorCreated(
+ (int)$sectorId,
+ (int)$operationId,
+ $sectorData['libelle']
+ );
+
// Préparer les données de réponse
$responseData = [
'sector_id' => $sectorId
@@ -413,9 +496,10 @@ class SectorController
}
// Récupérer les users affectés
- $usersQuery = "SELECT u.id, u.first_name, u.sect_name, u.encrypted_name, ous.fk_sector
+ $usersQuery = "SELECT u.id, ou.id as ope_user_id, u.first_name, u.sect_name, u.encrypted_name, ous.fk_sector
FROM ope_users_sectors ous
- JOIN users u ON ous.fk_user = u.id
+ JOIN ope_users ou ON ous.fk_user = ou.id
+ JOIN users u ON ou.fk_user = u.id
WHERE ous.fk_sector = :sector_id";
$usersStmt = $this->db->prepare($usersQuery);
$usersStmt->execute(['sector_id' => $sectorId]);
@@ -425,7 +509,8 @@ class SectorController
$responseData['users_sectors'] = [];
foreach ($usersSectors as $userSector) {
$userData = [
- 'id' => $userSector['id'],
+ 'user_id' => $userSector['id'],
+ 'ope_user_id' => $userSector['ope_user_id'],
'first_name' => $userSector['first_name'] ?? '',
'sect_name' => $userSector['sect_name'] ?? '',
'fk_sector' => $userSector['fk_sector'],
@@ -498,24 +583,27 @@ class SectorController
try {
$data = json_decode(file_get_contents('php://input'), true);
$entityId = $_SESSION['entity_id'] ?? null;
-
+
if (!$entityId) {
Response::json(['status' => 'error', 'message' => 'Entité non définie'], 400);
return;
}
-
+
// Vérifier que le secteur appartient à l'entité
- $checkQuery = "SELECT s.id
+ $checkQuery = "SELECT s.id, s.fk_operation, s.libelle
FROM ope_sectors s
JOIN operations o ON s.fk_operation = o.id
WHERE s.id = :id AND o.fk_entite = :entity_id";
$checkStmt = $this->db->prepare($checkQuery);
$checkStmt->execute(['id' => $id, 'entity_id' => $entityId]);
-
- if (!$checkStmt->fetch()) {
+
+ $existingSector = $checkStmt->fetch();
+ if (!$existingSector) {
Response::json(['status' => 'error', 'message' => 'Secteur non trouvé'], 404);
return;
}
+
+ $operationId = $existingSector['fk_operation'];
$this->db->beginTransaction();
@@ -580,8 +668,8 @@ class SectorController
// Ajouter les nouvelles affectations
if (!empty($data['users'])) {
- $insertQuery = "INSERT INTO ope_users_sectors (fk_operation, fk_user, fk_sector, created_at, fk_user_creat, chk_active)
- VALUES (:operation_id, :user_id, :sector_id, NOW(), :user_creat, 1)";
+ $insertQuery = "INSERT INTO ope_users_sectors (fk_operation, fk_user, fk_sector, created_at, updated_at, fk_user_creat, chk_active)
+ VALUES (:operation_id, :user_id, :sector_id, NOW(), NOW(), :user_creat, 1)";
$this->logService->info('[UPDATE USERS] SQL - Requête INSERT préparée', [
'query' => $insertQuery
]);
@@ -591,9 +679,27 @@ class SectorController
$failedUsers = [];
foreach ($data['users'] as $memberId) {
try {
+ // $memberId est DÉJÀ ope_users.id (envoyé par Flutter)
+ // Vérifier que cet ope_users.id existe et appartient bien à l'opération
+ $stmtOpeUser = $this->db->prepare('
+ SELECT id FROM ope_users
+ WHERE id = ? AND fk_operation = ?
+ ');
+ $stmtOpeUser->execute([$memberId, $operationId]);
+ $opeUserId = $stmtOpeUser->fetchColumn();
+
+ if (!$opeUserId) {
+ $this->logService->warning('[UPDATE USERS] ope_users.id non trouvé pour cette opération', [
+ 'ope_users_id' => $memberId,
+ 'operation_id' => $operationId
+ ]);
+ $failedUsers[] = $memberId;
+ continue;
+ }
+
$params = [
'operation_id' => $operationId,
- 'user_id' => $memberId,
+ 'user_id' => $opeUserId,
'sector_id' => $id,
'user_creat' => $_SESSION['user_id'] ?? null
];
@@ -626,14 +732,25 @@ class SectorController
}
}
- // Gérer les passages si le secteur a changé
+ // Gérer les passages si le secteur a changé ET si chk_adresses_change = 1
$passageCounters = [
'passages_orphaned' => 0,
'passages_updated' => 0,
'passages_created' => 0,
'passages_kept' => 0
];
- if (isset($data['sector'])) {
+
+ // chk_adresses_change : 0=ne pas toucher aux adresses/passages, 1=recalculer (défaut)
+ $chkAdressesChange = $data['chk_adresses_change'] ?? 1;
+
+ if (isset($data['sector']) && $chkAdressesChange == 0) {
+ $this->logService->info('[UPDATE] Modification secteur sans recalcul adresses/passages', [
+ 'sector_id' => $id,
+ 'chk_adresses_change' => $chkAdressesChange
+ ]);
+ }
+
+ if (isset($data['sector']) && $chkAdressesChange == 1) {
// Mettre à jour les adresses du secteur AVANT de traiter les passages
try {
// Supprimer les anciennes adresses
@@ -660,17 +777,25 @@ class SectorController
]);
$addresses = $this->addressService->getAddressesInPolygon($coordinates, $entityId);
-
+
+ // Enrichir les adresses avec les données bâtiments
+ $addresses = $this->addressService->enrichAddressesWithBuildings($addresses, $entityId);
+
$this->logService->info('[UPDATE] Adresses récupérées', [
'sector_id' => $id,
'nb_addresses' => count($addresses)
]);
-
+
if (!empty($addresses)) {
- $queryAddress = "INSERT INTO sectors_adresses (fk_sector, fk_adresse, numero, rue, cp, ville, gps_lat, gps_lng)
- VALUES (:sector_id, :address_id, :numero, :rue, :cp, :ville, :gps_lat, :gps_lng)";
+ $queryAddress = "INSERT INTO sectors_adresses (
+ fk_sector, fk_adresse, numero, rue, cp, ville, gps_lat, gps_lng,
+ fk_batiment, fk_habitat, nb_niveau, nb_log, residence, alt_sol
+ ) VALUES (
+ :sector_id, :address_id, :numero, :rue, :cp, :ville, :gps_lat, :gps_lng,
+ :fk_batiment, :fk_habitat, :nb_niveau, :nb_log, :residence, :alt_sol
+ )";
$stmtAddress = $this->db->prepare($queryAddress);
-
+
foreach ($addresses as $address) {
$stmtAddress->execute([
'sector_id' => $id,
@@ -680,7 +805,13 @@ class SectorController
'cp' => $address['code_postal'],
'ville' => $address['commune'],
'gps_lat' => $address['latitude'],
- 'gps_lng' => $address['longitude']
+ 'gps_lng' => $address['longitude'],
+ 'fk_batiment' => $address['fk_batiment'] ?? null,
+ 'fk_habitat' => $address['fk_habitat'] ?? 1,
+ 'nb_niveau' => $address['nb_niveau'] ?? null,
+ 'nb_log' => $address['nb_log'] ?? null,
+ 'residence' => $address['residence'] ?? '',
+ 'alt_sol' => $address['alt_sol'] ?? null
]);
}
@@ -715,10 +846,29 @@ class SectorController
// Commit des modifications (users et/ou secteur)
$this->db->commit();
-
+
+ // Log de mise à jour du secteur
+ $changes = [];
+ if (isset($data['libelle'])) {
+ $changes['libelle'] = ['new' => $data['libelle']];
+ }
+ if (isset($data['color'])) {
+ $changes['color'] = ['new' => $data['color']];
+ }
+ if (isset($data['sector'])) {
+ $changes['sector'] = true; // Polygon modifié
+ }
+ if (isset($data['users'])) {
+ $changes['users'] = true; // Affectation modifiée
+ }
+
+ if (!empty($changes)) {
+ EventLogService::logSectorUpdated((int)$id, (int)$operationId, $changes);
+ }
+
// Récupérer le secteur mis à jour
$query = "
- SELECT
+ SELECT
s.id,
s.libelle,
s.color,
@@ -726,57 +876,61 @@ class SectorController
FROM ope_sectors s
WHERE s.id = :id
";
-
+
$stmt = $this->db->prepare($query);
$stmt->execute(['id' => $id]);
$sector = $stmt->fetch(\PDO::FETCH_ASSOC);
-
- // Récupérer tous les passages du secteur
- $passagesQuery = "SELECT id, fk_operation, fk_sector, fk_user, fk_type, fk_adresse, passed_at,
- numero, rue, rue_bis, ville, residence, fk_habitat, appt, niveau,
- gps_lat, gps_lng, nom_recu, encrypted_name, remarque, encrypted_email,
- encrypted_phone, montant, fk_type_reglement, email_erreur, nb_passages
- FROM ope_pass
- WHERE fk_sector = :sector_id
- ORDER BY id";
- $passagesStmt = $this->db->prepare($passagesQuery);
- $passagesStmt->execute(['sector_id' => $id]);
- $passages = $passagesStmt->fetchAll(\PDO::FETCH_ASSOC);
-
- // Déchiffrer les données sensibles des passages
+
+ // Récupérer les passages UNIQUEMENT si chk_adresses_change = 1
$passagesDecrypted = [];
- foreach ($passages as $passage) {
- // Déchiffrement du nom
- $passage['name'] = '';
- if (!empty($passage['encrypted_name'])) {
- $passage['name'] = ApiService::decryptData($passage['encrypted_name']);
- }
- unset($passage['encrypted_name']);
-
- // Déchiffrement de l'email
- $passage['email'] = '';
- if (!empty($passage['encrypted_email'])) {
- $decryptedEmail = ApiService::decryptSearchableData($passage['encrypted_email']);
- if ($decryptedEmail) {
- $passage['email'] = $decryptedEmail;
+ if ($chkAdressesChange == 1) {
+ // Récupérer tous les passages du secteur
+ $passagesQuery = "SELECT id, fk_operation, fk_sector, fk_user, fk_type, fk_adresse, passed_at,
+ numero, rue, rue_bis, ville, residence, fk_habitat, appt, niveau,
+ gps_lat, gps_lng, nom_recu, encrypted_name, remarque, encrypted_email,
+ encrypted_phone, montant, fk_type_reglement, email_erreur, nb_passages
+ FROM ope_pass
+ WHERE fk_sector = :sector_id
+ ORDER BY id";
+ $passagesStmt = $this->db->prepare($passagesQuery);
+ $passagesStmt->execute(['sector_id' => $id]);
+ $passages = $passagesStmt->fetchAll(\PDO::FETCH_ASSOC);
+
+ // Déchiffrer les données sensibles des passages
+ foreach ($passages as $passage) {
+ // Déchiffrement du nom
+ $passage['name'] = '';
+ if (!empty($passage['encrypted_name'])) {
+ $passage['name'] = ApiService::decryptData($passage['encrypted_name']);
}
+ unset($passage['encrypted_name']);
+
+ // Déchiffrement de l'email
+ $passage['email'] = '';
+ if (!empty($passage['encrypted_email'])) {
+ $decryptedEmail = ApiService::decryptSearchableData($passage['encrypted_email']);
+ if ($decryptedEmail) {
+ $passage['email'] = $decryptedEmail;
+ }
+ }
+ unset($passage['encrypted_email']);
+
+ // Déchiffrement du téléphone
+ $passage['phone'] = '';
+ if (!empty($passage['encrypted_phone'])) {
+ $passage['phone'] = ApiService::decryptData($passage['encrypted_phone']);
+ }
+ unset($passage['encrypted_phone']);
+
+ $passagesDecrypted[] = $passage;
}
- unset($passage['encrypted_email']);
-
- // Déchiffrement du téléphone
- $passage['phone'] = '';
- if (!empty($passage['encrypted_phone'])) {
- $passage['phone'] = ApiService::decryptData($passage['encrypted_phone']);
- }
- unset($passage['encrypted_phone']);
-
- $passagesDecrypted[] = $passage;
}
// Récupérer les users affectés (avec READ UNCOMMITTED pour forcer la lecture des données fraîches)
- $usersQuery = "SELECT u.id, u.first_name, u.sect_name, u.encrypted_name, ous.fk_sector
+ $usersQuery = "SELECT u.id, ou.id as ope_user_id, u.first_name, u.sect_name, u.encrypted_name, ous.fk_sector
FROM ope_users_sectors ous
- JOIN users u ON ous.fk_user = u.id
+ JOIN ope_users ou ON ous.fk_user = ou.id
+ JOIN users u ON ou.fk_user = u.id
WHERE ous.fk_sector = :sector_id
ORDER BY u.id";
@@ -801,7 +955,8 @@ class SectorController
$usersDecrypted = [];
foreach ($usersSectors as $userSector) {
$userData = [
- 'id' => $userSector['id'],
+ 'user_id' => $userSector['id'],
+ 'ope_user_id' => $userSector['ope_user_id'],
'first_name' => $userSector['first_name'] ?? '',
'sect_name' => $userSector['sect_name'] ?? '',
'fk_sector' => $userSector['fk_sector'],
@@ -934,18 +1089,20 @@ class SectorController
}
// Vérifier que le secteur existe et récupérer ses informations
- $checkQuery = "SELECT s.id, s.libelle, o.fk_entite
+ $checkQuery = "SELECT s.id, s.libelle, s.fk_operation, o.fk_entite
FROM ope_sectors s
JOIN operations o ON s.fk_operation = o.id
WHERE s.id = :id";
$checkStmt = $this->db->prepare($checkQuery);
$checkStmt->execute(['id' => $id]);
$sector = $checkStmt->fetch();
-
+
if (!$sector || $sector['fk_entite'] != $entityId) {
Response::json(['status' => 'error', 'message' => 'Secteur non trouvé ou non autorisé'], 404);
return;
}
+
+ $operationId = $sector['fk_operation'];
$this->db->beginTransaction();
@@ -1001,9 +1158,16 @@ class SectorController
$deleteQuery = "DELETE FROM ope_sectors WHERE id = :id";
$deleteStmt = $this->db->prepare($deleteQuery);
$deleteStmt->execute(['id' => $id]);
-
+
$this->db->commit();
-
+
+ // Log de suppression du secteur (suppression physique = false)
+ EventLogService::logSectorDeleted(
+ (int)$id,
+ (int)$operationId,
+ false // suppression physique (DELETE)
+ );
+
// Déchiffrer les données sensibles des passages
$passagesDecrypted = [];
foreach ($passagesToUpdate as $passage) {
@@ -1249,8 +1413,11 @@ class SectorController
}
// 2. CRÉATION/MISE À JOUR DES PASSAGES POUR LES NOUVELLES ADRESSES (OPTIMISÉE)
- // Récupérer toutes les adresses du secteur depuis sectors_adresses
- $addressesQuery = "SELECT * FROM sectors_adresses WHERE fk_sector = :sector_id";
+ // Récupérer toutes les adresses du secteur depuis sectors_adresses (avec colonnes bâtiments)
+ $addressesQuery = "SELECT
+ fk_sector, fk_adresse, numero, rue, rue_bis, cp, ville, gps_lat, gps_lng,
+ fk_batiment, fk_habitat, nb_niveau, nb_log, residence, alt_sol
+ FROM sectors_adresses WHERE fk_sector = :sector_id";
$addressesStmt = $this->db->prepare($addressesQuery);
$addressesStmt->execute(['sector_id' => $sectorId]);
$addresses = $addressesStmt->fetchAll();
@@ -1268,93 +1435,121 @@ class SectorController
$firstUserId = $firstUser ? $firstUser['fk_user'] : null;
if ($firstUserId && !empty($addresses)) {
- $this->logService->info('[updatePassagesForSector] Optimisation passages', [
+ $this->logService->info('[updatePassagesForSector] Traitement des passages', [
'user_id' => $firstUserId,
'nb_addresses' => count($addresses)
]);
- // OPTIMISATION : Récupérer TOUS les passages existants en UNE requête
- $addressIds = array_filter(array_column($addresses, 'fk_adresse'));
-
- // Construire la requête pour récupérer tous les passages existants
+ // Récupérer TOUS les passages existants pour cette opération en UNE requête
$existingQuery = "
- SELECT id, fk_adresse, numero, rue, rue_bis, ville
+ SELECT id, fk_adresse, numero, rue, rue_bis, ville, residence, appt, fk_habitat,
+ fk_type, encrypted_name, created_at
FROM ope_pass
- WHERE fk_operation = :operation_id
- AND (";
-
- $params = ['operation_id' => $operationId];
- $conditions = [];
-
- // Condition pour les fk_adresse
- if (!empty($addressIds)) {
- $placeholders = [];
- foreach ($addressIds as $idx => $addrId) {
- $key = 'addr_' . $idx;
- $placeholders[] = ':' . $key;
- $params[$key] = $addrId;
- }
- $conditions[] = "fk_adresse IN (" . implode(',', $placeholders) . ")";
- }
-
- // Condition pour les données d'adresse (numero, rue, ville)
- $addressConditions = [];
- foreach ($addresses as $idx => $addr) {
- $numKey = 'num_' . $idx;
- $rueKey = 'rue_' . $idx;
- $bisKey = 'bis_' . $idx;
- $villeKey = 'ville_' . $idx;
-
- $addressConditions[] = "(numero = :$numKey AND rue = :$rueKey AND rue_bis = :$bisKey AND ville = :$villeKey)";
- $params[$numKey] = $addr['numero'];
- $params[$rueKey] = $addr['rue'];
- $params[$bisKey] = $addr['rue_bis'];
- $params[$villeKey] = $addr['ville'];
- }
-
- if (!empty($addressConditions)) {
- $conditions[] = "(" . implode(' OR ', $addressConditions) . ")";
- }
-
- $existingQuery .= implode(' OR ', $conditions) . ")";
+ WHERE fk_operation = :operation_id";
$existingStmt = $this->db->prepare($existingQuery);
- $existingStmt->execute($params);
+ $existingStmt->execute(['operation_id' => $operationId]);
$existingPassages = $existingStmt->fetchAll();
- // Indexer les passages existants pour recherche rapide
+ // Indexer les passages existants par clé : numero|rue|rue_bis|ville
$passagesByAddress = [];
- $passagesByData = [];
foreach ($existingPassages as $p) {
- if (!empty($p['fk_adresse'])) {
- $passagesByAddress[$p['fk_adresse']] = $p;
+ $addressKey = $p['numero'] . '|' . $p['rue'] . '|' . ($p['rue_bis'] ?? '') . '|' . $p['ville'];
+ if (!isset($passagesByAddress[$addressKey])) {
+ $passagesByAddress[$addressKey] = [];
}
- $dataKey = $p['numero'] . '|' . $p['rue'] . '|' . $p['rue_bis'] . '|' . $p['ville'];
- $passagesByData[$dataKey] = $p;
+ $passagesByAddress[$addressKey][] = $p;
}
- // Préparer les listes pour batch insert/update
+ // Traiter chaque adresse du secteur
$toInsert = [];
$toUpdate = [];
+ $toDelete = [];
foreach ($addresses as $address) {
- // Vérification en mémoire PHP (0 requête)
- if (!empty($address['fk_adresse']) && isset($passagesByAddress[$address['fk_adresse']])) {
- continue; // Déjà existant avec bon fk_adresse
- }
+ $addressKey = $address['numero'] . '|' . $address['rue'] . '|' . ($address['rue_bis'] ?? '') . '|' . $address['ville'];
+ $existingAtAddress = $passagesByAddress[$addressKey] ?? [];
+ $nbExisting = count($existingAtAddress);
- $dataKey = $address['numero'] . '|' . $address['rue'] . '|' . $address['rue_bis'] . '|' . $address['ville'];
- if (isset($passagesByData[$dataKey])) {
- // Passage existant mais sans fk_adresse ou avec fk_adresse différent
- if (!empty($address['fk_adresse']) && $passagesByData[$dataKey]['fk_adresse'] != $address['fk_adresse']) {
- $toUpdate[] = [
- 'id' => $passagesByData[$dataKey]['id'],
- 'fk_adresse' => $address['fk_adresse']
+ $fkHabitat = $address['fk_habitat'] ?? 1;
+ $nbLog = ($fkHabitat == 2 && isset($address['nb_log'])) ? (int)$address['nb_log'] : 1;
+ $residence = $address['residence'] ?? '';
+
+ // IMPORTANT : Uniformisation GPS pour les immeubles
+ // Tous les passages d'une même adresse doivent partager les mêmes coordonnées GPS
+ // Issues de sectors_adresses (gps_lat, gps_lng)
+ $gpsLat = $address['gps_lat'];
+ $gpsLng = $address['gps_lng'];
+
+ // CAS 1 : Maison individuelle (fk_habitat=1)
+ if ($fkHabitat == 1) {
+ if ($nbExisting == 0) {
+ // INSERT 1 passage
+ $toInsert[] = [
+ 'address' => $address,
+ 'residence' => '',
+ 'appt' => '',
+ 'fk_habitat' => 1
];
+ } else {
+ // UPDATE le premier passage avec fk_habitat=1
+ $toUpdate[] = [
+ 'id' => $existingAtAddress[0]['id'],
+ 'fk_habitat' => 1,
+ 'residence' => '',
+ 'gps_lat' => $gpsLat,
+ 'gps_lng' => $gpsLng
+ ];
+ // Les autres passages (si >1) ne sont PAS touchés
+ }
+ }
+ // CAS 2 : Immeuble (fk_habitat=2)
+ else if ($fkHabitat == 2) {
+ // UPDATE TOUS les passages existants avec fk_habitat=2, residence et GPS
+ foreach ($existingAtAddress as $existing) {
+ $updates = [
+ 'id' => $existing['id'],
+ 'fk_habitat' => 2,
+ 'gps_lat' => $gpsLat,
+ 'gps_lng' => $gpsLng
+ ];
+ // Update residence seulement si non vide
+ if (!empty($residence)) {
+ $updates['residence'] = $residence;
+ }
+ $toUpdate[] = $updates;
+ }
+
+ // Si moins de nb_log passages : INSERT les manquants
+ if ($nbExisting < $nbLog) {
+ $nbToInsert = $nbLog - $nbExisting;
+ for ($i = 0; $i < $nbToInsert; $i++) {
+ $toInsert[] = [
+ 'address' => $address,
+ 'residence' => $residence,
+ 'appt' => '', // Pas de numéro d'appt prédéfini
+ 'fk_habitat' => 2
+ ];
+ }
+ }
+ // Si plus de nb_log passages : DELETE les non visités en trop
+ else if ($nbExisting > $nbLog) {
+ $nbToDelete = $nbExisting - $nbLog;
+ // Trier les passages par created_at ASC (les plus anciens d'abord)
+ usort($existingAtAddress, function($a, $b) {
+ return strtotime($a['created_at']) - strtotime($b['created_at']);
+ });
+
+ $deleted = 0;
+ foreach ($existingAtAddress as $existing) {
+ if ($deleted >= $nbToDelete) break;
+ // Supprimer seulement si fk_type=2 ET encrypted_name vide
+ if ($existing['fk_type'] == 2 && ($existing['encrypted_name'] === '' || $existing['encrypted_name'] === null)) {
+ $toDelete[] = $existing['id'];
+ $deleted++;
+ }
+ }
}
- } else {
- // Nouveau passage à créer
- $toInsert[] = $address;
}
}
@@ -1364,19 +1559,24 @@ class SectorController
$insertParams = [];
$paramIndex = 0;
- foreach ($toInsert as $addr) {
+ foreach ($toInsert as $item) {
+ $addr = $item['address'];
$values[] = "(:op$paramIndex, :sect$paramIndex, :usr$paramIndex, :addr$paramIndex,
:num$paramIndex, :rue$paramIndex, :bis$paramIndex, :ville$paramIndex,
- :lat$paramIndex, :lng$paramIndex, 2, '', NOW(), :creat$paramIndex, 1)";
+ :res$paramIndex, :appt$paramIndex, :habitat$paramIndex,
+ :lat$paramIndex, :lng$paramIndex, 2, 0, '', NOW(), :creat$paramIndex, 1)";
$insertParams["op$paramIndex"] = $operationId;
$insertParams["sect$paramIndex"] = $sectorId;
$insertParams["usr$paramIndex"] = $firstUserId;
- $insertParams["addr$paramIndex"] = $addr['fk_adresse'];
+ $insertParams["addr$paramIndex"] = $addr['fk_adresse'] ?? null;
$insertParams["num$paramIndex"] = $addr['numero'];
$insertParams["rue$paramIndex"] = $addr['rue'];
- $insertParams["bis$paramIndex"] = $addr['rue_bis'];
+ $insertParams["bis$paramIndex"] = $addr['rue_bis'] ?? '';
$insertParams["ville$paramIndex"] = $addr['ville'];
+ $insertParams["res$paramIndex"] = $item['residence'];
+ $insertParams["appt$paramIndex"] = $item['appt'];
+ $insertParams["habitat$paramIndex"] = $item['fk_habitat'];
$insertParams["lat$paramIndex"] = $addr['gps_lat'];
$insertParams["lng$paramIndex"] = $addr['gps_lng'];
$insertParams["creat$paramIndex"] = $_SESSION['user_id'] ?? null;
@@ -1386,7 +1586,7 @@ class SectorController
$insertQuery = "INSERT INTO ope_pass
(fk_operation, fk_sector, fk_user, fk_adresse, numero, rue, rue_bis,
- ville, gps_lat, gps_lng, fk_type, encrypted_name, created_at, fk_user_creat, chk_active)
+ ville, residence, appt, fk_habitat, gps_lat, gps_lng, fk_type, nb_passages, encrypted_name, created_at, fk_user_creat, chk_active)
VALUES " . implode(',', $values);
try {
@@ -1401,28 +1601,67 @@ class SectorController
}
}
- // UPDATE MULTIPLE avec CASE WHEN
+ // UPDATE MULTIPLE avec CASE WHEN (inclut GPS pour uniformisation)
if (!empty($toUpdate)) {
$updateIds = array_column($toUpdate, 'id');
$placeholders = str_repeat('?,', count($updateIds) - 1) . '?';
- $caseWhen = [];
+ $caseWhenHabitat = [];
+ $caseWhenResidence = [];
+ $caseWhenGpsLat = [];
+ $caseWhenGpsLng = [];
$updateParams = [];
foreach ($toUpdate as $upd) {
- $caseWhen[] = "WHEN id = ? THEN ?";
+ // fk_habitat est toujours présent
+ $caseWhenHabitat[] = "WHEN id = ? THEN ?";
$updateParams[] = $upd['id'];
- $updateParams[] = $upd['fk_adresse'];
+ $updateParams[] = $upd['fk_habitat'];
+
+ // GPS : toujours présent maintenant (uniformisation)
+ if (isset($upd['gps_lat']) && isset($upd['gps_lng'])) {
+ $caseWhenGpsLat[] = "WHEN id = ? THEN ?";
+ $updateParams[] = $upd['id'];
+ $updateParams[] = $upd['gps_lat'];
+
+ $caseWhenGpsLng[] = "WHEN id = ? THEN ?";
+ $updateParams[] = $upd['id'];
+ $updateParams[] = $upd['gps_lng'];
+ }
+
+ // residence est optionnel
+ if (isset($upd['residence'])) {
+ $caseWhenResidence[] = "WHEN id = ? THEN ?";
+ $updateParams[] = $upd['id'];
+ $updateParams[] = $upd['residence'];
+ }
+ }
+
+ $setClause = ["fk_habitat = CASE " . implode(' ', $caseWhenHabitat) . " ELSE fk_habitat END"];
+ if (!empty($caseWhenGpsLat)) {
+ $setClause[] = "gps_lat = CASE " . implode(' ', $caseWhenGpsLat) . " ELSE gps_lat END";
+ }
+ if (!empty($caseWhenGpsLng)) {
+ $setClause[] = "gps_lng = CASE " . implode(' ', $caseWhenGpsLng) . " ELSE gps_lng END";
+ }
+ if (!empty($caseWhenResidence)) {
+ $setClause[] = "residence = CASE " . implode(' ', $caseWhenResidence) . " ELSE residence END";
}
$updateQuery = "UPDATE ope_pass
- SET fk_adresse = CASE " . implode(' ', $caseWhen) . " END
+ SET " . implode(', ', $setClause) . "
WHERE id IN ($placeholders)";
try {
$updateStmt = $this->db->prepare($updateQuery);
$updateStmt->execute(array_merge($updateParams, $updateIds));
$counters['passages_updated'] = count($toUpdate);
+
+ // Log pour vérifier l'uniformisation GPS (surtout pour immeubles)
+ $this->logService->info('[updatePassagesForSector] Passages mis à jour avec GPS uniformisés', [
+ 'nb_updated' => count($toUpdate),
+ 'sector_id' => $sectorId
+ ]);
} catch (\Exception $e) {
$this->logService->error('Erreur lors de la mise à jour multiple des passages', [
'sector_id' => $sectorId,
@@ -1431,6 +1670,23 @@ class SectorController
}
}
+ // DELETE MULTIPLE en une seule requête
+ if (!empty($toDelete)) {
+ $placeholders = str_repeat('?,', count($toDelete) - 1) . '?';
+ $deleteQuery = "DELETE FROM ope_pass WHERE id IN ($placeholders)";
+
+ try {
+ $deleteStmt = $this->db->prepare($deleteQuery);
+ $deleteStmt->execute($toDelete);
+ $counters['passages_deleted'] += count($toDelete);
+ } catch (\Exception $e) {
+ $this->logService->error('Erreur lors de la suppression multiple des passages', [
+ 'sector_id' => $sectorId,
+ 'error' => $e->getMessage()
+ ]);
+ }
+ }
+
} else {
$this->logService->warning('[updatePassagesForSector] Pas de création de passages', [
'reason' => !$firstUserId ? 'Pas d\'utilisateur affecté' : 'Pas d\'adresses',
diff --git a/api/src/Controllers/StripeController.php b/api/src/Controllers/StripeController.php
index a22a0c91..c2a1a8c8 100644
--- a/api/src/Controllers/StripeController.php
+++ b/api/src/Controllers/StripeController.php
@@ -6,6 +6,9 @@ namespace App\Controllers;
use App\Core\Controller;
use App\Services\StripeService;
+use App\Services\LogService;
+use App\Services\FileService;
+use App\Services\ApiService;
use Session;
use Exception;
@@ -77,7 +80,7 @@ class StripeController extends Controller {
$this->requireAuth();
// Log du début de la requête
- \LogService::log('Début createOnboardingLink', [
+ LogService::log('Début createOnboardingLink', [
'account_id' => $accountId,
'user_id' => Session::getUserId()
]);
@@ -98,7 +101,7 @@ class StripeController extends Controller {
$returnUrl = $data['return_url'] ?? '';
$refreshUrl = $data['refresh_url'] ?? '';
- \LogService::log('URLs reçues', [
+ LogService::log('URLs reçues', [
'return_url' => $returnUrl,
'refresh_url' => $refreshUrl
]);
@@ -110,7 +113,7 @@ class StripeController extends Controller {
$result = $this->stripeService->createOnboardingLink($accountId, $returnUrl, $refreshUrl);
- \LogService::log('Résultat createOnboardingLink', [
+ LogService::log('Résultat createOnboardingLink', [
'success' => $result['success'] ?? false,
'has_url' => isset($result['url'])
]);
@@ -127,7 +130,7 @@ class StripeController extends Controller {
}
} catch (Exception $e) {
- \LogService::log('Erreur createOnboardingLink', [
+ LogService::log('Erreur createOnboardingLink', [
'level' => 'error',
'message' => $e->getMessage(),
'trace' => $e->getTraceAsString()
@@ -190,7 +193,7 @@ class StripeController extends Controller {
// Vérifier que le passage existe et appartient à l'utilisateur
$stmt = $this->db->prepare('
- SELECT p.*, o.fk_entite
+ SELECT p.*, o.fk_entite, o.id as operation_id
FROM ope_pass p
JOIN operations o ON p.fk_operation = o.id
WHERE p.id = ? AND p.fk_user = ?
@@ -210,13 +213,15 @@ class StripeController extends Controller {
}
// Vérifier que le montant correspond (passage.montant est en euros, amount en centimes)
- $expectedAmount = (int)($passage['montant'] * 100);
+ $expectedAmount = (int)round($passage['montant'] * 100);
if ($amount !== $expectedAmount) {
$this->sendError("Le montant ne correspond pas au passage (attendu: {$expectedAmount} centimes, reçu: {$amount} centimes)", 400);
return;
}
$entiteId = $passage['fk_entite'];
+ $operationId = $passage['operation_id'];
+ $fkUser = $passage['fk_user']; // ope_users.id
// Déterminer le type de paiement (Tap to Pay ou Web)
$paymentMethodTypes = $data['payment_method_types'] ?? ['card_present'];
@@ -230,14 +235,16 @@ class StripeController extends Controller {
'payment_method_types' => $paymentMethodTypes,
'capture_method' => $data['capture_method'] ?? 'automatic',
'passage_id' => $passageId,
- 'amicale_id' => $data['amicale_id'] ?? $entiteId,
- 'member_id' => $data['member_id'] ?? Session::getUserId(),
+ 'fk_entite' => $data['amicale_id'] ?? $entiteId,
+ 'fk_user' => $data['member_id'] ?? $fkUser,
'stripe_account' => $data['stripe_account'] ?? null,
'metadata' => array_merge(
[
'passage_id' => (string)$passageId,
+ 'operation_id' => (string)$operationId,
'amicale_id' => (string)($data['amicale_id'] ?? $entiteId),
- 'member_id' => (string)($data['member_id'] ?? Session::getUserId()),
+ 'fk_user' => (string)$fkUser,
+ 'created_at' => (string)time(),
'type' => $isTapToPay ? 'tap_to_pay' : 'web'
],
$data['metadata'] ?? []
@@ -291,11 +298,12 @@ class StripeController extends Controller {
$stmt = $this->db->prepare("
SELECT p.*, o.fk_entite,
e.encrypted_name as entite_nom,
- u.first_name as user_prenom, u.sect_name as user_nom
+ ou.first_name as user_prenom, u.sect_name as user_nom
FROM ope_pass p
JOIN operations o ON p.fk_operation = o.id
LEFT JOIN entites e ON o.fk_entite = e.id
- LEFT JOIN users u ON p.fk_user = u.id
+ LEFT JOIN ope_users ou ON p.fk_user = ou.id
+ LEFT JOIN users u ON ou.fk_user = u.id
WHERE p.stripe_payment_id = :pi_id
");
$stmt->execute(['pi_id' => $paymentIntentId]);
@@ -330,7 +338,7 @@ class StripeController extends Controller {
$entiteNom = '';
if (!empty($passage['entite_nom'])) {
try {
- $entiteNom = \ApiService::decryptData($passage['entite_nom']);
+ $entiteNom = ApiService::decryptData($passage['entite_nom']);
} catch (Exception $e) {
$entiteNom = 'Entité inconnue';
}
@@ -400,6 +408,7 @@ class StripeController extends Controller {
$this->sendSuccess([
'has_account' => false,
'account_id' => null,
+ 'location_id' => null,
'charges_enabled' => false,
'payouts_enabled' => false,
'onboarding_completed' => false
@@ -415,6 +424,7 @@ class StripeController extends Controller {
$this->sendSuccess([
'has_account' => true,
'account_id' => $account['stripe_account_id'],
+ 'location_id' => $account['stripe_location_id'] ?? null,
'charges_enabled' => false,
'payouts_enabled' => false,
'onboarding_completed' => false,
@@ -440,6 +450,7 @@ class StripeController extends Controller {
$this->sendSuccess([
'has_account' => true,
'account_id' => $account['stripe_account_id'],
+ 'location_id' => $account['stripe_location_id'] ?? null,
'charges_enabled' => $stripeAccount->charges_enabled,
'payouts_enabled' => $stripeAccount->payouts_enabled,
'onboarding_completed' => $stripeAccount->details_submitted,
@@ -529,17 +540,17 @@ class StripeController extends Controller {
public function getPublicConfig(): void {
try {
$this->requireAuth();
-
+
$this->sendSuccess([
'public_key' => $this->stripeService->getPublicKey(),
'test_mode' => $this->stripeService->isTestMode()
]);
-
+
} catch (Exception $e) {
$this->sendError('Erreur: ' . $e->getMessage());
}
}
-
+
/**
* GET /api/stripe/stats
* Récupérer les statistiques de paiement
@@ -613,9 +624,164 @@ class StripeController extends Controller {
'to' => $dateTo
]
]);
-
+
} catch (Exception $e) {
$this->sendError('Erreur: ' . $e->getMessage());
}
}
+
+ /**
+ * POST /api/stripe/payment-links
+ * Créer un Payment Link Stripe pour paiement par QR Code
+ *
+ * Payload:
+ * {
+ * "amount": 2500,
+ * "currency": "eur",
+ * "description": "Calendrier pompiers",
+ * "passage_id": 789,
+ * "metadata": {...}
+ * }
+ */
+ public function createPaymentLink(): void {
+ try {
+ $this->requireAuth();
+
+ $data = $this->getJsonInput();
+
+ // Validation
+ if (!isset($data['amount']) || !isset($data['passage_id'])) {
+ $this->sendError('Montant et passage_id requis', 400);
+ return;
+ }
+
+ $amount = (int)$data['amount'];
+ $passageId = (int)$data['passage_id'];
+
+ // Validation du montant (doit être > 0)
+ if ($amount <= 0) {
+ $this->sendError('Le montant doit être supérieur à 0', 400);
+ return;
+ }
+
+ // Vérifier que le passage appartient à l'utilisateur ou à son entité
+ $userId = Session::getUserId();
+ $stmt = $this->db->prepare('
+ SELECT p.*, o.fk_entite, ou.fk_user as ope_user_id
+ FROM ope_pass p
+ JOIN operations o ON p.fk_operation = o.id
+ JOIN ope_users ou ON p.fk_user = ou.id
+ WHERE p.id = ?
+ ');
+ $stmt->execute([$passageId]);
+ $passage = $stmt->fetch();
+
+ if (!$passage) {
+ $this->sendError('Passage non trouvé', 404);
+ return;
+ }
+
+ // Vérifier les droits : soit l'utilisateur est le créateur du passage, soit il appartient à la même entité
+ $userEntityId = Session::getEntityId();
+ if ($passage['ope_user_id'] != $userId && $passage['fk_entite'] != $userEntityId) {
+ $this->sendError('Passage non autorisé', 403);
+ return;
+ }
+
+ // Vérifier qu'il n'y a pas déjà un paiement ou un payment link pour ce passage
+ if (!empty($passage['stripe_payment_id'])) {
+ $this->sendError('Un paiement Stripe existe déjà pour ce passage', 400);
+ return;
+ }
+
+ if (!empty($passage['stripe_payment_link_id'])) {
+ $this->sendError('Un Payment Link existe déjà pour ce passage', 400);
+ return;
+ }
+
+ // Vérifier que le montant correspond (passage.montant est en euros, amount en centimes)
+ $expectedAmount = (int)round($passage['montant'] * 100);
+ if ($amount !== $expectedAmount) {
+ $this->sendError("Le montant ne correspond pas au passage (attendu: {$expectedAmount} centimes, reçu: {$amount} centimes)", 400);
+ return;
+ }
+
+ // Préparer les paramètres
+ $params = [
+ 'amount' => $amount,
+ 'currency' => $data['currency'] ?? 'eur',
+ 'description' => $data['description'] ?? 'Calendrier pompiers',
+ 'passage_id' => $passageId,
+ 'metadata' => $data['metadata'] ?? []
+ ];
+
+ // Créer le Payment Link
+ $result = $this->stripeService->createPaymentLink($params);
+
+ if ($result['success']) {
+ $this->sendSuccess([
+ 'payment_link_id' => $result['payment_link_id'],
+ 'url' => $result['url'],
+ 'amount' => $result['amount'],
+ 'passage_id' => $passageId,
+ 'type' => 'qr_code'
+ ]);
+ } else {
+ $this->sendError($result['message'], 400);
+ }
+
+ } catch (Exception $e) {
+ $this->sendError('Erreur: ' . $e->getMessage());
+ }
+ }
+
+ /**
+ * POST /api/stripe/locations
+ * Créer une Location Stripe Terminal pour une entité (nécessaire pour Tap to Pay)
+ */
+ public function createLocation(): void {
+ try {
+ $this->requireAuth();
+
+ // Vérifier le rôle de l'utilisateur
+ $userId = Session::getUserId();
+ $stmt = $this->db->prepare('SELECT fk_role FROM users WHERE id = ?');
+ $stmt->execute([$userId]);
+ $result = $stmt->fetch();
+ $userRole = $result ? (int)$result['fk_role'] : 0;
+
+ if ($userRole < 2) {
+ $this->sendError('Droits insuffisants - Admin amicale minimum requis', 403);
+ return;
+ }
+
+ $data = $this->getJsonInput();
+ $entiteId = $data['fk_entite'] ?? Session::getEntityId();
+
+ if (!$entiteId) {
+ $this->sendError('ID entité requis', 400);
+ return;
+ }
+
+ // Vérifier les droits sur cette entité
+ if (Session::getEntityId() != $entiteId && $userRole < 3) {
+ $this->sendError('Non autorisé pour cette entité', 403);
+ return;
+ }
+
+ $result = $this->stripeService->createLocation($entiteId);
+
+ if ($result['success']) {
+ $this->sendSuccess([
+ 'location_id' => $result['location_id'],
+ 'message' => $result['message']
+ ]);
+ } else {
+ $this->sendError($result['message'], 400);
+ }
+
+ } catch (Exception $e) {
+ $this->sendError('Erreur lors de la création de la location: ' . $e->getMessage());
+ }
+ }
}
\ No newline at end of file
diff --git a/api/src/Controllers/StripeWebhookController.php b/api/src/Controllers/StripeWebhookController.php
index 121a55ea..0e977bbe 100644
--- a/api/src/Controllers/StripeWebhookController.php
+++ b/api/src/Controllers/StripeWebhookController.php
@@ -43,8 +43,8 @@ class StripeWebhookController extends Controller {
}
// Récupérer le secret webhook selon le mode
- $stripeConfig = $this->config->get('stripe');
- $webhookSecret = $this->stripeService->isTestMode()
+ $stripeConfig = $this->config->getStripeConfig();
+ $webhookSecret = $this->stripeService->isTestMode()
? $stripeConfig['webhook_secret_test']
: $stripeConfig['webhook_secret_live'];
@@ -95,31 +95,35 @@ class StripeWebhookController extends Controller {
case 'account.updated':
$this->handleAccountUpdated($event->data->object);
break;
-
+
case 'account.application.authorized':
$this->handleAccountAuthorized($event->data->object);
break;
-
+
case 'payment_intent.succeeded':
$this->handlePaymentIntentSucceeded($event->data->object);
break;
-
+
case 'payment_intent.payment_failed':
$this->handlePaymentIntentFailed($event->data->object);
break;
-
+
+ case 'checkout.session.completed':
+ $this->handleCheckoutSessionCompleted($event->data->object);
+ break;
+
case 'charge.dispute.created':
$this->handleChargeDisputeCreated($event->data->object);
break;
-
+
case 'terminal.reader.action_succeeded':
$this->handleTerminalReaderActionSucceeded($event->data->object);
break;
-
+
case 'terminal.reader.action_failed':
$this->handleTerminalReaderActionFailed($event->data->object);
break;
-
+
default:
// Événement non géré mais valide
error_log("Unhandled Stripe event type: {$event->type}");
@@ -278,7 +282,60 @@ class StripeWebhookController extends Controller {
error_log("Payment failed: {$paymentIntent->id}, reason: " . json_encode($paymentIntent->last_payment_error));
}
-
+
+ /**
+ * Gérer la complétion d'une session de paiement (Payment Link / Checkout)
+ */
+ private function handleCheckoutSessionCompleted($session): void {
+ $metadata = $session->metadata;
+
+ // Logger l'événement
+ error_log("Checkout session completed: {$session->id}, payment_intent: {$session->payment_intent}");
+
+ // Vérifier si un passage_id est présent dans les metadata
+ if (isset($metadata->passage_id) && !empty($metadata->passage_id)) {
+ $passageId = (int)$metadata->passage_id;
+
+ // Mettre à jour le passage avec le stripe_payment_id
+ $stmt = $this->db->prepare("
+ UPDATE ope_pass
+ SET stripe_payment_id = :payment_intent_id,
+ updated_at = NOW()
+ WHERE id = :passage_id
+ ");
+ $stmt->execute([
+ 'payment_intent_id' => $session->payment_intent,
+ 'passage_id' => $passageId
+ ]);
+
+ // Vérifier si la mise à jour a réussi
+ if ($stmt->rowCount() > 0) {
+ error_log("Passage {$passageId} updated with payment_intent {$session->payment_intent}");
+
+ // TODO: Envoyer un email de confirmation avec le reçu fiscal
+ // TODO: Mettre à jour les statistiques en temps réel
+ } else {
+ error_log("Warning: Passage {$passageId} not found or already updated");
+ }
+ } else {
+ error_log("Warning: checkout.session.completed without passage_id in metadata");
+ }
+
+ // Enregistrer l'historique de la session dans stripe_payment_history si nécessaire
+ if (isset($metadata->passage_id)) {
+ $stmt = $this->db->prepare("
+ SELECT id FROM ope_pass WHERE id = :passage_id
+ ");
+ $stmt->execute(['passage_id' => $metadata->passage_id]);
+ $passage = $stmt->fetch();
+
+ if ($passage) {
+ // Log dans l'historique
+ error_log("Checkout session completed for passage {$metadata->passage_id}: amount={$session->amount_total}, currency={$session->currency}");
+ }
+ }
+ }
+
/**
* Gérer un litige (chargeback)
*/
diff --git a/api/src/Controllers/UserController.php b/api/src/Controllers/UserController.php
index 08e90cfb..95091cd1 100755
--- a/api/src/Controllers/UserController.php
+++ b/api/src/Controllers/UserController.php
@@ -5,6 +5,7 @@ declare(strict_types=1);
namespace App\Controllers;
require_once __DIR__ . '/../Services/LogService.php';
+require_once __DIR__ . '/../Services/EventLogService.php';
require_once __DIR__ . '/../Services/ApiService.php';
require_once __DIR__ . '/../Services/PasswordSecurityService.php';
@@ -15,8 +16,9 @@ use AppConfig;
use Request;
use Response;
use Session;
-use LogService;
-use ApiService;
+use App\Services\LogService;
+use App\Services\EventLogService;
+use App\Services\ApiService;
use App\Services\PasswordSecurityService;
class UserController {
@@ -529,16 +531,13 @@ class UserController {
]);
}
- LogService::log('Utilisateur GeoSector créé', [
- 'level' => 'info',
- 'createdBy' => $currentUserId,
- 'newUserId' => $userId,
- 'email' => !empty($email) ? $email : 'non fourni',
- 'username' => $username,
- 'usernameManual' => $chkUsernameManuel === 1 ? 'oui' : 'non',
- 'passwordManual' => $chkMdpManuel === 1 ? 'oui' : 'non',
- 'emailsSent' => !empty($email) ? '2 emails (username + password)' : 'aucun (pas d\'email)'
- ]);
+ // Log de création utilisateur
+ EventLogService::logUserCreated(
+ (int)$userId,
+ (int)$entiteId,
+ (int)$role,
+ $username
+ );
// Préparer la réponse avec les informations de connexion si générées automatiquement
$responseData = [
@@ -762,12 +761,23 @@ class UserController {
return;
}
- LogService::log('Utilisateur GeoSector mis à jour', [
- 'level' => 'info',
- 'modifiedBy' => $currentUserId,
- 'userId' => $id,
- 'fields' => array_keys($data),
- ]);
+ // Log de mise à jour utilisateur
+ $changes = [];
+ $encryptedFields = ['name', 'email', 'phone', 'mobile', 'username', 'password'];
+
+ foreach ($data as $key => $value) {
+ if (in_array($key, $encryptedFields)) {
+ // Champs sensibles : booléen uniquement
+ $changes['encrypted_' . $key] = true;
+ } else {
+ // Champs non sensibles : valeur
+ $changes[$key] = ['new' => $value];
+ }
+ }
+
+ if (!empty($changes)) {
+ EventLogService::logUserUpdated((int)$id, $changes);
+ }
Response::json([
'status' => 'success',
@@ -858,24 +868,72 @@ class UserController {
if ($transferTo) {
try {
- // Transférer TOUS les passages de l'utilisateur vers l'utilisateur désigné
- $stmt3 = $this->db->prepare('
- UPDATE ope_pass
- SET fk_user = :new_user_id
- WHERE fk_user = :delete_user_id
+ // Transférer les passages opération par opération
+ // (car fk_user dans ope_pass pointe vers ope_users.id, pas users.id)
+
+ // Récupérer toutes les opérations où l'utilisateur à supprimer a des entrées dans ope_users
+ $stmtOps = $this->db->prepare('
+ SELECT DISTINCT fk_operation
+ FROM ope_users
+ WHERE fk_user = ?
');
- $stmt3->execute([
- 'new_user_id' => $transferTo,
- 'delete_user_id' => $id
- ]);
-
- $transferredCount = $stmt3->rowCount();
-
+ $stmtOps->execute([$id]);
+ $operations = $stmtOps->fetchAll(PDO::FETCH_COLUMN);
+
+ $totalTransferred = 0;
+
+ foreach ($operations as $operationId) {
+ // Trouver ope_users.id de l'utilisateur à supprimer dans cette opération
+ $stmtOldOpeUser = $this->db->prepare('
+ SELECT id FROM ope_users
+ WHERE fk_user = ? AND fk_operation = ?
+ ');
+ $stmtOldOpeUser->execute([$id, $operationId]);
+ $oldOpeUserId = $stmtOldOpeUser->fetchColumn();
+
+ if (!$oldOpeUserId) {
+ continue;
+ }
+
+ // Trouver ope_users.id de l'utilisateur de destination dans cette opération
+ $stmtNewOpeUser = $this->db->prepare('
+ SELECT id FROM ope_users
+ WHERE fk_user = ? AND fk_operation = ?
+ ');
+ $stmtNewOpeUser->execute([$transferTo, $operationId]);
+ $newOpeUserId = $stmtNewOpeUser->fetchColumn();
+
+ if (!$newOpeUserId) {
+ LogService::log('Impossible de transférer passages - utilisateur destination absent', [
+ 'level' => 'warning',
+ 'operation_id' => $operationId,
+ 'from_user' => $id,
+ 'to_user' => $transferTo
+ ]);
+ continue;
+ }
+
+ // Transférer les passages
+ $stmtTransfer = $this->db->prepare('
+ UPDATE ope_pass
+ SET fk_user = :new_ope_user_id
+ WHERE fk_user = :old_ope_user_id AND fk_operation = :operation_id
+ ');
+ $stmtTransfer->execute([
+ 'new_ope_user_id' => $newOpeUserId,
+ 'old_ope_user_id' => $oldOpeUserId,
+ 'operation_id' => $operationId
+ ]);
+
+ $totalTransferred += $stmtTransfer->rowCount();
+ }
+
LogService::log('Passages transférés avant suppression utilisateur', [
'level' => 'info',
'from_user' => $id,
'to_user' => $transferTo,
- 'passages_transferred' => $transferredCount
+ 'operations_count' => count($operations),
+ 'passages_transferred' => $totalTransferred
]);
} catch (PDOException $e) {
Response::json([
@@ -890,13 +948,10 @@ class UserController {
// —— Suppression réelle de l'utilisateur ——
try {
// Supprimer les enregistrements dépendants dans ope_users
+ // (CASCADE supprime automatiquement ope_users_sectors et ope_pass)
$stmtOpeUsers = $this->db->prepare('DELETE FROM ope_users WHERE fk_user = ?');
$stmtOpeUsers->execute([$id]);
- // Supprimer les enregistrements dépendants dans ope_users_sectors
- $stmtOpeUsersSectors = $this->db->prepare('DELETE FROM ope_users_sectors WHERE fk_user = ?');
- $stmtOpeUsersSectors->execute([$id]);
-
$stmt = $this->db->prepare('DELETE FROM users WHERE id = ?');
$stmt->execute([$id]);
@@ -908,12 +963,8 @@ class UserController {
return;
}
- LogService::log('Utilisateur GeoSector supprimé', [
- 'level' => 'info',
- 'deletedBy' => $currentUserId,
- 'userId' => $id,
- 'passage_transfer' => $transferTo ? "Tous les passages transférés vers utilisateur $transferTo" : 'Aucun transfert'
- ]);
+ // Log de suppression utilisateur (suppression physique = false pour soft_delete)
+ EventLogService::logUserDeleted((int)$id, false);
Response::json([
'status' => 'success',
diff --git a/api/src/Controllers/VilleController.php b/api/src/Controllers/VilleController.php
index 107f8e6e..e6ec88e8 100755
--- a/api/src/Controllers/VilleController.php
+++ b/api/src/Controllers/VilleController.php
@@ -14,8 +14,8 @@ use AppConfig;
use Request;
use Response;
use Session;
-use LogService;
-use ApiService;
+use App\Services\LogService;
+use App\Services\ApiService;
use Exception;
class VilleController {
diff --git a/api/src/Core/Router.php b/api/src/Core/Router.php
index b9c29611..c8984ca3 100755
--- a/api/src/Core/Router.php
+++ b/api/src/Core/Router.php
@@ -11,10 +11,11 @@ class Router {
'register',
'lostpassword',
'log',
+ 'health', // Health check endpoint pour monitoring et déploiement
'villes', // Ajout de la route villes comme endpoint public pour l'autocomplétion du code postal
'password/check', // Vérification de la force des mots de passe (public pour l'inscription)
'password/compromised', // Vérification si un mot de passe est compromis
- 'stripe/webhook', // Webhook Stripe (doit être public pour recevoir les événements)
+ 'stripe/webhooks', // Webhook Stripe (doit être public pour recevoir les événements)
];
public function __construct() {
@@ -34,6 +35,9 @@ class Router {
// Route pour les logs
$this->post('log', ['LogController', 'index']);
+ // Route health check (monitoring et déploiement)
+ $this->get('health', ['HealthController', 'check']);
+
// Routes privées utilisateurs
// IMPORTANT: Les routes spécifiques doivent être déclarées AVANT les routes avec paramètres
$this->post('users/check-username', ['UserController', 'checkUsername']); // Déplacé avant les routes avec :id
@@ -131,12 +135,14 @@ class Router {
$this->get('stripe/accounts/:entityId/status', ['StripeController', 'getAccountStatus']);
$this->post('stripe/accounts/:accountId/onboarding-link', ['StripeController', 'createOnboardingLink']);
- // Tap to Pay - Vérification compatibilité
+ // Tap to Pay - Vérification compatibilité et configuration
$this->post('stripe/devices/check-tap-to-pay', ['StripeController', 'checkTapToPayCapability']);
$this->get('stripe/devices/certified-android', ['StripeController', 'getCertifiedAndroidDevices']);
+ $this->post('stripe/locations', ['StripeController', 'createLocation']);
// Paiements
$this->post('stripe/payments/create-intent', ['StripeController', 'createPaymentIntent']);
+ $this->post('stripe/payment-links', ['StripeController', 'createPaymentLink']);
$this->get('stripe/payments/:paymentIntentId', ['StripeController', 'getPaymentStatus']);
// Statistiques et configuration
@@ -144,7 +150,21 @@ class Router {
$this->get('stripe/config', ['StripeController', 'getPublicConfig']);
// Webhook (IMPORTANT: pas d'authentification requise pour les webhooks Stripe)
- $this->post('stripe/webhook', ['StripeWebhookController', 'handleWebhook']);
+ $this->post('stripe/webhooks', ['StripeWebhookController', 'handleWebhook']);
+
+ // Routes Migration (Admin uniquement)
+ $this->get('migrations/test-connections', ['MigrationController', 'testConnections']);
+ $this->get('migrations/entities/available', ['MigrationController', 'getAvailableEntities']);
+ $this->get('migrations/entities/:id', ['MigrationController', 'getEntityDetails']);
+ $this->post('migrations/entity', ['MigrationController', 'migrateEntity']);
+ $this->post('migrations/entity/step', ['MigrationController', 'migrateEntityStep']);
+ $this->get('migrations/entity/:id/status', ['MigrationController', 'getMigrationStatus']);
+ $this->get('migrations/entity/:id/logs', ['MigrationController', 'getMigrationLogs']);
+ $this->get('migrations/entity/:id/report', ['MigrationController', 'getMigrationReport']);
+ $this->get('migrations/entity/:id/compare', ['MigrationController', 'compareEntityData']);
+ $this->get('migrations/entity/:id/verify', ['MigrationController', 'verifyMigration']);
+ $this->delete('migrations/entity/:id', ['MigrationController', 'rollbackEntity']);
+ $this->delete('migrations/entity/:id/step/:step', ['MigrationController', 'rollbackStep']);
}
public function handle(): void {
@@ -180,7 +200,6 @@ class Router {
// Check if endpoint is public
if ($this->isPublicEndpoint($endpoint)) {
- error_log("Public endpoint found: $endpoint");
$route = $this->findRoute($method, $endpoint);
if ($route) {
$this->executeRoute($route);
diff --git a/api/src/Core/Session.php b/api/src/Core/Session.php
index 76aa020e..609d0e20 100755
--- a/api/src/Core/Session.php
+++ b/api/src/Core/Session.php
@@ -5,22 +5,55 @@ declare(strict_types=1);
class Session {
public static function start(): void {
if (session_status() === PHP_SESSION_NONE) {
+ // Configuration d'un répertoire de sessions dédié et persistant
+ $sessionPath = __DIR__ . '/../../sessions';
+ if (!is_dir($sessionPath)) {
+ mkdir($sessionPath, 0700, true);
+ }
+ ini_set('session.save_path', $sessionPath);
+
// Configuration des sessions adaptée pour les applications mobiles
ini_set('session.use_strict_mode', '1');
ini_set('session.cookie_httponly', '1');
-
+
// Permettre les connexions non-HTTPS en développement
$isProduction = (getenv('APP_ENV') === 'production');
ini_set('session.cookie_secure', $isProduction ? '1' : '0');
-
+
// SameSite None pour permettre les requêtes cross-origin (applications mobiles)
ini_set('session.cookie_samesite', 'None');
- ini_set('session.gc_maxlifetime', '86400'); // 24 heures
+
+ // Configuration de la durée de vie des sessions : 24 heures
+ $sessionLifetime = 86400; // 24 heures
+ ini_set('session.gc_maxlifetime', (string)$sessionLifetime);
+ ini_set('session.cookie_lifetime', (string)$sessionLifetime);
+
+ // Configuration du garbage collector pour qu'il ne supprime pas trop tôt
+ // gc_probability / gc_divisor = probabilité d'exécution (1/100 = 1%)
+ ini_set('session.gc_probability', '1');
+ ini_set('session.gc_divisor', '100');
// Récupérer le session_id du Bearer token si présent
self::getSessionFromBearer();
session_start();
+
+ // Log détaillé après le démarrage de la session (DEBUG)
+ $logFile = __DIR__ . '/../../logs/session_' . date('Y-m-d') . '.log';
+ $sessionId = session_id();
+ $sessionExists = isset($_SESSION) && !empty($_SESSION);
+ $sessionData = $sessionExists ? json_encode($_SESSION) : 'empty';
+ $sessionFile = $sessionPath . '/sess_' . $sessionId;
+ $sessionFileExists = file_exists($sessionFile);
+
+ $logMessage = date('Y-m-d H:i:s') . " - Session started\n";
+ $logMessage .= " Session ID: $sessionId\n";
+ $logMessage .= " Session path: $sessionPath\n";
+ $logMessage .= " Session file exists: " . ($sessionFileExists ? 'YES' : 'NO') . "\n";
+ $logMessage .= " Session data exists: " . ($sessionExists ? 'YES' : 'NO') . "\n";
+ $logMessage .= " Session data: $sessionData\n";
+
+ file_put_contents($logFile, $logMessage, FILE_APPEND);
}
}
diff --git a/api/src/Services/AddressService.php b/api/src/Services/AddressService.php
index a2e20108..535be5c5 100644
--- a/api/src/Services/AddressService.php
+++ b/api/src/Services/AddressService.php
@@ -1,17 +1,35 @@
logService = new LogService();
+
try {
- $this->addressesDb = AddressesDatabase::getInstance();
+ $this->addressesDb = \AddressesDatabase::getInstance();
$this->logService->info('[AddressService] Connexion à la base d\'adresses réussie');
} catch (\Exception $e) {
// Si la connexion échoue, on continue sans la base d'adresses
@@ -21,53 +39,59 @@ class AddressService {
]);
$this->addressesDb = null;
}
- $this->mainDb = Database::getInstance();
+
+ $this->mainDb = \Database::getInstance();
+ $this->buildingService = new BuildingService();
}
-
+
/**
* Vérifie si la connexion à la base d'adresses est active
+ *
* @return bool
*/
- public function isConnected(): bool {
+ public function isConnected(): bool
+ {
return $this->addressesDb !== null;
}
-
+
/**
* Détermine le département de l'entité courante
- *
+ *
* @param int|null $entityId ID de l'entité
* @return string|null Code département (ex: "22", "23")
*/
- private function getDepartmentForEntity(?int $entityId = null): ?string {
+ private function getDepartmentForEntity(?int $entityId = null): ?string
+ {
if (!$entityId) {
$entityId = $_SESSION['entity_id'] ?? null;
}
-
+
if (!$entityId) {
return null;
}
-
+
try {
$query = "SELECT departement FROM entites WHERE id = :entity_id";
$stmt = $this->mainDb->prepare($query);
$stmt->execute(['entity_id' => $entityId]);
$result = $stmt->fetch();
-
+
return $result ? $result['departement'] : null;
} catch (\Exception $e) {
return null;
}
}
-
+
/**
* Récupère toutes les adresses contenues dans un polygone défini par des coordonnées
* Gère automatiquement les secteurs multi-départements
- *
+ *
* @param array $coordinates Array de coordonnées [[lat, lng], [lat, lng], ...]
* @param int|null $entityId ID de l'entité (pour déterminer le département principal)
* @return array Array des adresses trouvées
*/
- public function getAddressesInPolygon(array $coordinates, ?int $entityId = null): array {
+ public function getAddressesInPolygon(array $coordinates, ?int $entityId = null): array
+ {
// Si pas de connexion à la base d'adresses, retourner un tableau vide
if (!$this->addressesDb) {
$this->logService->error('[AddressService] Pas de connexion à la base d\'adresses externe', [
@@ -75,21 +99,20 @@ class AddressService {
]);
return [];
}
-
+
$this->logService->info('[AddressService] Début recherche adresses', [
'entity_id' => $entityId,
'nb_coordinates' => count($coordinates)
]);
-
+
if (count($coordinates) < 3) {
throw new InvalidArgumentException("Un polygone doit avoir au moins 3 points");
}
-
+
// D'abord, déterminer tous les départements touchés par ce secteur
- require_once __DIR__ . '/DepartmentBoundaryService.php';
- $boundaryService = new \DepartmentBoundaryService();
+ $boundaryService = new DepartmentBoundaryService();
$departmentsTouched = $boundaryService->getDepartmentsForSector($coordinates);
-
+
if (empty($departmentsTouched)) {
// Si aucun département n'est trouvé par analyse spatiale,
// chercher d'abord dans le département de l'entité et ses limitrophes
@@ -103,22 +126,22 @@ class AddressService {
]);
throw new RuntimeException("Impossible de déterminer le département");
}
-
+
// Obtenir les départements prioritaires (entité + limitrophes)
$priorityDepts = $boundaryService->getPriorityDepartments($entityDept);
-
+
// Log pour debug
$this->logService->warning('[AddressService] Aucun département trouvé par analyse spatiale', [
'departements_prioritaires' => implode(', ', $priorityDepts)
]);
-
+
// Utiliser les départements prioritaires pour la recherche
$departmentsTouched = [];
foreach ($priorityDepts as $deptCode) {
$departmentsTouched[] = ['code_dept' => $deptCode];
}
}
-
+
// Créer le polygone SQL à partir des coordonnées
$polygonPoints = [];
foreach ($coordinates as $coord) {
@@ -127,22 +150,22 @@ class AddressService {
}
$polygonPoints[] = $coord[1] . ' ' . $coord[0]; // MySQL attend longitude latitude
}
-
+
// Fermer le polygone
$polygonPoints[] = $polygonPoints[0];
-
+
$polygonString = 'POLYGON((' . implode(',', $polygonPoints) . '))';
-
+
// Collecter les adresses de tous les départements touchés
$allAddresses = [];
-
+
foreach ($departmentsTouched as $dept) {
$deptCode = $dept['code_dept'];
$tableName = "cp" . $deptCode;
-
+
try {
// Requête pour récupérer les adresses dans le polygone pour ce département
- $sql = "SELECT
+ $sql = "SELECT
id,
numero,
rue as voie,
@@ -161,32 +184,32 @@ class AddressService {
:dept_code as departement
FROM `$tableName`
WHERE ST_Contains(
- ST_GeomFromText(:polygon, 4326),
+ ST_GeomFromText(:polygon, 4326),
POINT(CAST(gps_lng AS DECIMAL(10,8)), CAST(gps_lat AS DECIMAL(10,8)))
)
AND gps_lat != ''
AND gps_lng != ''";
-
+
$stmt = $this->addressesDb->prepare($sql);
$stmt->execute([
'polygon' => $polygonString,
'dept_code' => $deptCode
]);
-
+
$addresses = $stmt->fetchAll();
-
+
// Ajouter les adresses à la collection globale
foreach ($addresses as $address) {
$allAddresses[] = $address;
}
-
+
// Log pour debug
$this->logService->info('[AddressService] Recherche dans table', [
'table' => $tableName,
'departement' => $deptCode,
'nb_adresses' => count($addresses)
]);
-
+
} catch (PDOException $e) {
// Log l'erreur mais continue avec les autres départements
$this->logService->error('[AddressService] Erreur SQL', [
@@ -197,35 +220,90 @@ class AddressService {
]);
}
}
-
+
$this->logService->info('[AddressService] Fin recherche adresses', [
'total_adresses' => count($allAddresses)
]);
return $allAddresses;
}
-
+
+ /**
+ * Enrichit les adresses avec les données bâtiments depuis la base 'batiments'
+ *
+ * Pour chaque adresse trouvée, cette méthode cherche si un bâtiment existe
+ * et ajoute les métadonnées (nb_log, residence, fk_habitat, etc.)
+ *
+ * @param array $addresses Liste d'adresses depuis getAddressesInPolygon()
+ * @param int|null $entityId ID de l'entité (pour logs)
+ * @return array Adresses enrichies avec données bâtiment
+ */
+ public function enrichAddressesWithBuildings(array $addresses, ?int $entityId = null): array
+ {
+ if (empty($addresses)) {
+ return [];
+ }
+
+ $this->logService->info('[AddressService] Début enrichissement avec bâtiments', [
+ 'entity_id' => $entityId,
+ 'nb_addresses' => count($addresses)
+ ]);
+
+ try {
+ $enrichedAddresses = $this->buildingService->enrichAddresses($addresses);
+
+ // Compter les immeubles vs maisons
+ $nbImmeubles = 0;
+ $nbMaisons = 0;
+ foreach ($enrichedAddresses as $addr) {
+ if (isset($addr['fk_habitat']) && $addr['fk_habitat'] == 2) {
+ $nbImmeubles++;
+ } else {
+ $nbMaisons++;
+ }
+ }
+
+ $this->logService->info('[AddressService] Fin enrichissement avec bâtiments', [
+ 'total_adresses' => count($enrichedAddresses),
+ 'nb_immeubles' => $nbImmeubles,
+ 'nb_maisons' => $nbMaisons
+ ]);
+
+ return $enrichedAddresses;
+
+ } catch (\Exception $e) {
+ $this->logService->error('[AddressService] Erreur lors de l\'enrichissement', [
+ 'error' => $e->getMessage(),
+ 'trace' => $e->getTraceAsString()
+ ]);
+
+ // En cas d'erreur, retourner les adresses sans enrichissement
+ return $addresses;
+ }
+ }
+
/**
* Récupère les adresses dans un rayon autour d'un point
- *
+ *
* @param float $latitude Latitude du centre
* @param float $longitude Longitude du centre
* @param float $radiusMeters Rayon en mètres
* @param int|null $entityId ID de l'entité (pour déterminer le département)
* @return array Array des adresses trouvées
*/
- public function getAddressesInRadius(float $latitude, float $longitude, float $radiusMeters, ?int $entityId = null): array {
+ public function getAddressesInRadius(float $latitude, float $longitude, float $radiusMeters, ?int $entityId = null): array
+ {
// Déterminer le département
$dept = $this->getDepartmentForEntity($entityId);
if (!$dept) {
throw new RuntimeException("Impossible de déterminer le département de l'entité");
}
-
+
// Nom de la table selon le département
$tableName = "cp" . $dept;
-
+
try {
// Utiliser ST_Distance_Sphere pour calculer la distance en mètres
- $sql = "SELECT
+ $sql = "SELECT
id,
numero,
rue as voie,
@@ -245,45 +323,44 @@ class AddressService {
AND gps_lat != ''
AND gps_lng != ''
ORDER BY distance";
-
+
$point = "POINT($longitude $latitude)";
-
+
$stmt = $this->addressesDb->prepare($sql);
$stmt->execute([
'point' => $point,
'radius' => $radiusMeters
]);
-
+
return $stmt->fetchAll();
} catch (PDOException $e) {
throw new RuntimeException("Erreur lors de la récupération des adresses dans la table $tableName : " . $e->getMessage());
}
}
-
+
/**
* Compte le nombre d'adresses dans un polygone
* Gère automatiquement les secteurs multi-départements
- *
+ *
* @param array $coordinates Array de coordonnées [[lat, lng], [lat, lng], ...]
* @param int|null $entityId ID de l'entité (pour déterminer le département principal)
* @return int Nombre d'adresses
*/
- public function countAddressesInPolygon(array $coordinates, ?int $entityId = null): int {
+ public function countAddressesInPolygon(array $coordinates, ?int $entityId = null): int
+ {
// Si pas de connexion à la base d'adresses, retourner 0
if (!$this->addressesDb) {
- error_log("AddressService: Pas de connexion à la base d'adresses, retour de 0 adresses");
return 0;
}
-
+
if (count($coordinates) < 3) {
throw new InvalidArgumentException("Un polygone doit avoir au moins 3 points");
}
-
+
// D'abord, déterminer tous les départements touchés par ce secteur
- require_once __DIR__ . '/DepartmentBoundaryService.php';
- $boundaryService = new \DepartmentBoundaryService();
+ $boundaryService = new DepartmentBoundaryService();
$departmentsTouched = $boundaryService->getDepartmentsForSector($coordinates);
-
+
if (empty($departmentsTouched)) {
// Si aucun département n'est trouvé, utiliser le département de l'entité
$dept = $this->getDepartmentForEntity($entityId);
@@ -292,7 +369,7 @@ class AddressService {
}
$departmentsTouched = [['code_dept' => $dept]];
}
-
+
// Créer le polygone SQL à partir des coordonnées
$polygonPoints = [];
foreach ($coordinates as $coord) {
@@ -301,19 +378,19 @@ class AddressService {
}
$polygonPoints[] = $coord[1] . ' ' . $coord[0]; // MySQL attend longitude latitude
}
-
+
// Fermer le polygone
$polygonPoints[] = $polygonPoints[0];
-
+
$polygonString = 'POLYGON((' . implode(',', $polygonPoints) . '))';
-
+
// Compter les adresses dans tous les départements touchés
$totalCount = 0;
-
+
foreach ($departmentsTouched as $dept) {
$deptCode = $dept['code_dept'];
$tableName = "cp" . $deptCode;
-
+
try {
$sql = "SELECT COUNT(*) as count
FROM `$tableName`
@@ -323,23 +400,20 @@ class AddressService {
)
AND gps_lat != ''
AND gps_lng != ''";
-
+
$stmt = $this->addressesDb->prepare($sql);
$stmt->execute(['polygon' => $polygonString]);
-
+
$result = $stmt->fetch();
$deptCount = (int)$result['count'];
$totalCount += $deptCount;
-
- // Log pour debug
- error_log("Département $deptCode : $deptCount adresses comptées");
-
+
} catch (PDOException $e) {
// Log l'erreur mais continue avec les autres départements
error_log("Erreur de comptage pour le département $deptCode : " . $e->getMessage());
}
}
-
+
return $totalCount;
}
-}
\ No newline at end of file
+}
diff --git a/api/src/Services/ApiService.php b/api/src/Services/ApiService.php
index 8e89da3f..03ebdf33 100755
--- a/api/src/Services/ApiService.php
+++ b/api/src/Services/ApiService.php
@@ -2,10 +2,13 @@
declare(strict_types=1);
+namespace App\Services;
+
use PHPMailer\PHPMailer\PHPMailer;
use PHPMailer\PHPMailer\SMTP;
use PHPMailer\PHPMailer\Exception;
-use App\Services\PasswordSecurityService;
+use AppConfig;
+use App\Services\LogService;
require_once __DIR__ . '/EmailTemplates.php';
require_once __DIR__ . '/PasswordSecurityService.php';
@@ -70,11 +73,21 @@ class ApiService {
$mail->Body = EmailTemplates::getLostPasswordTemplate($name, $data['username'] ?? '', $data['password']);
break;
+ case 'password_reset':
+ $mail->Subject = 'Réinitialisation de votre mot de passe GEOSECTOR';
+ $mail->Body = EmailTemplates::getLostPasswordTemplate($name, $data['username'] ?? '', $data['password'] ?? '');
+ break;
+
case 'alert':
$mail->Subject = $data['subject'] ?? 'Alerte GEOSECTOR';
$mail->Body = EmailTemplates::getAlertTemplate($data['subject'] ?? 'Alerte', $data['message'] ?? '');
break;
+ case 'security_alert':
+ $mail->Subject = $data['subject'] ?? 'Alerte de Sécurité GEOSECTOR';
+ $mail->Body = $data['body'] ?? EmailTemplates::getAlertTemplate($data['subject'] ?? 'Alerte de Sécurité', $data['message'] ?? '');
+ break;
+
case 'receipt':
$mail->Subject = 'Reçu de passage GEOSECTOR';
$mail->Body = EmailTemplates::getReceiptTemplate(
diff --git a/api/src/Services/BackupEncryptionService.php b/api/src/Services/BackupEncryptionService.php
index 3600a842..6df1e6f9 100755
--- a/api/src/Services/BackupEncryptionService.php
+++ b/api/src/Services/BackupEncryptionService.php
@@ -2,14 +2,18 @@
declare(strict_types=1);
+namespace App\Services;
+
require_once __DIR__ . '/../Config/AppConfig.php';
require_once __DIR__ . '/LogService.php';
use Exception;
+use AppConfig;
+use App\Services\LogService;
/**
* Service de chiffrement et compression des sauvegardes
- *
+ *
* Ce service gère le processus complet de sécurisation des backups JSON :
* 1. Compression GZIP pour réduire la taille
* 2. Chiffrement AES-256-CBC pour la sécurité
diff --git a/api/src/Services/BuildingService.php b/api/src/Services/BuildingService.php
new file mode 100644
index 00000000..20a309ba
--- /dev/null
+++ b/api/src/Services/BuildingService.php
@@ -0,0 +1,319 @@
+initConnection();
+ }
+
+ /**
+ * Initialise la connexion à la base de données batiments
+ */
+ private function initConnection(): void
+ {
+ try {
+ $config = AppConfig::getInstance()->getBuildingsDatabaseConfig();
+
+ $dsn = sprintf(
+ 'mysql:host=%s;dbname=%s;charset=utf8mb4',
+ $config['host'],
+ $config['name']
+ );
+
+ $this->dbBuildings = new PDO(
+ $dsn,
+ $config['username'],
+ $config['password'],
+ [
+ PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION,
+ PDO::ATTR_DEFAULT_FETCH_MODE => PDO::FETCH_ASSOC,
+ PDO::ATTR_EMULATE_PREPARES => false,
+ ]
+ );
+
+ $this->connected = true;
+
+ } catch (PDOException $e) {
+ error_log("Erreur de connexion à la base batiments : " . $e->getMessage());
+ $this->connected = false;
+ }
+ }
+
+ /**
+ * Vérifie si la connexion à la base batiments est active
+ *
+ * @return bool True si connecté
+ */
+ public function isConnected(): bool
+ {
+ return $this->connected;
+ }
+
+ /**
+ * Enrichit une liste d'adresses avec les données bâtiment
+ *
+ * Pour chaque adresse trouvée par AddressService, cette méthode cherche
+ * si un bâtiment existe dans bat{dept} avec le lien cle_interop_adr.
+ *
+ * @param array $addresses Liste d'adresses depuis AddressService
+ * Format attendu : ['id' => 'cp22.12345', 'departement' => '22', ...]
+ * @return array Adresses enrichies avec :
+ * - fk_batiment : batiment_groupe_id (ou null)
+ * - fk_habitat : 1=individuel, 2=collectif
+ * - nb_niveau : Nombre d'étages (ou null)
+ * - nb_log : Nombre de logements (ou null)
+ * - residence : Nom de la copropriété (ou '')
+ * - alt_sol : Altitude sol (ou null)
+ */
+ public function enrichAddresses(array $addresses): array
+ {
+ // Si pas de connexion, retourner les adresses sans enrichissement
+ if (!$this->isConnected() || empty($addresses)) {
+ // Ajouter fk_habitat=1 par défaut (maison individuelle)
+ foreach ($addresses as &$address) {
+ $address['fk_batiment'] = null;
+ $address['fk_habitat'] = 1;
+ $address['nb_niveau'] = null;
+ $address['nb_log'] = null;
+ $address['residence'] = '';
+ $address['alt_sol'] = null;
+ }
+ return $addresses;
+ }
+
+ try {
+ // Grouper les adresses par département
+ $addressesByDept = [];
+ foreach ($addresses as $address) {
+ $dept = $this->extractDepartmentFromAddress($address);
+ if ($dept) {
+ $addressesByDept[$dept][] = $address;
+ }
+ }
+
+ // Enrichir les adresses département par département
+ $enrichedAddresses = [];
+
+ foreach ($addressesByDept as $dept => $deptAddresses) {
+ $enrichedDept = $this->enrichAddressesByDepartment((string)$dept, $deptAddresses);
+ $enrichedAddresses = array_merge($enrichedAddresses, $enrichedDept);
+ }
+
+ return $enrichedAddresses;
+
+ } catch (PDOException $e) {
+ error_log("Erreur lors de l'enrichissement des adresses avec batiments : " . $e->getMessage());
+
+ // En cas d'erreur, retourner les adresses avec valeurs par défaut
+ foreach ($addresses as &$address) {
+ $address['fk_batiment'] = null;
+ $address['fk_habitat'] = 1;
+ $address['nb_niveau'] = null;
+ $address['nb_log'] = null;
+ $address['residence'] = '';
+ $address['alt_sol'] = null;
+ }
+ return $addresses;
+ }
+ }
+
+ /**
+ * Enrichit les adresses d'un département spécifique
+ *
+ * @param string $dept Code du département (ex: '22', '35')
+ * @param array $addresses Liste des adresses du département
+ * @return array Adresses enrichies
+ */
+ private function enrichAddressesByDepartment(string $dept, array $addresses): array
+ {
+ // Vérifier que la table bat{dept} existe
+ if (!$this->tableExists("bat{$dept}")) {
+ // Table inexistante, retourner avec valeurs par défaut
+ foreach ($addresses as &$address) {
+ $address['fk_batiment'] = null;
+ $address['fk_habitat'] = 1;
+ $address['nb_niveau'] = null;
+ $address['nb_log'] = null;
+ $address['residence'] = '';
+ $address['alt_sol'] = null;
+ }
+ return $addresses;
+ }
+
+ // Créer un mapping address_id => address pour retrouver rapidement
+ $addressMap = [];
+ $addressIds = [];
+
+ foreach ($addresses as $address) {
+ // Extraire l'ID BAN (partie après le point)
+ $addressId = $this->extractAddressId($address['id']);
+ if ($addressId) {
+ $addressIds[] = $addressId;
+ $addressMap[$addressId] = $address;
+ }
+ }
+
+ if (empty($addressIds)) {
+ // Pas d'IDs valides, retourner avec valeurs par défaut
+ foreach ($addresses as &$address) {
+ $address['fk_batiment'] = null;
+ $address['fk_habitat'] = 1;
+ $address['nb_niveau'] = null;
+ $address['nb_log'] = null;
+ $address['residence'] = '';
+ $address['alt_sol'] = null;
+ }
+ return $addresses;
+ }
+
+ // Requête pour récupérer les bâtiments
+ $placeholders = str_repeat('?,', count($addressIds) - 1) . '?';
+ $query = "
+ SELECT
+ b.cle_interop_adr,
+ b.batiment_groupe_id,
+ b.nb_niveau,
+ b.nb_log,
+ b.residence,
+ b.altitude_sol_mean
+ FROM bat{$dept} b
+ WHERE b.cle_interop_adr IN ($placeholders)
+ ";
+
+ $stmt = $this->dbBuildings->prepare($query);
+ $stmt->execute($addressIds);
+ $buildings = $stmt->fetchAll();
+
+ // Créer un mapping cle_interop_adr => building
+ $buildingMap = [];
+ foreach ($buildings as $building) {
+ $buildingMap[$building['cle_interop_adr']] = $building;
+ }
+
+ // Enrichir les adresses
+ $enrichedAddresses = [];
+ foreach ($addresses as $address) {
+ $addressId = $this->extractAddressId($address['id']);
+
+ if ($addressId && isset($buildingMap[$addressId])) {
+ // Bâtiment trouvé : enrichir avec ses données
+ $building = $buildingMap[$addressId];
+ $address['fk_batiment'] = $building['batiment_groupe_id'];
+ $address['fk_habitat'] = 2; // Collectif
+ $address['nb_niveau'] = $building['nb_niveau'];
+ $address['nb_log'] = $building['nb_log'];
+ $address['residence'] = $building['residence'] ?? '';
+ $address['alt_sol'] = $building['altitude_sol_mean'];
+ } else {
+ // Pas de bâtiment : maison individuelle
+ $address['fk_batiment'] = null;
+ $address['fk_habitat'] = 1; // Individuel
+ $address['nb_niveau'] = null;
+ $address['nb_log'] = null;
+ $address['residence'] = '';
+ $address['alt_sol'] = null;
+ }
+
+ $enrichedAddresses[] = $address;
+ }
+
+ return $enrichedAddresses;
+ }
+
+ /**
+ * Extrait le code département depuis une adresse
+ *
+ * @param array $address Adresse depuis AddressService
+ * @return string|null Code département (ex: '22', '35')
+ */
+ private function extractDepartmentFromAddress(array $address): ?string
+ {
+ // Méthode 1 : Clé 'departement' directement
+ if (isset($address['departement'])) {
+ return $address['departement'];
+ }
+
+ // Méthode 2 : Depuis code_postal
+ if (isset($address['code_postal'])) {
+ $cp = $address['code_postal'];
+ if (strlen($cp) === 4) {
+ return '0' . substr($cp, 0, 1);
+ }
+ return substr($cp, 0, 2);
+ }
+
+ // Méthode 3 : Depuis l'ID (format cp22.12345)
+ if (isset($address['id']) && strpos($address['id'], 'cp') === 0) {
+ preg_match('/^cp(\d{2})\./', $address['id'], $matches);
+ if (isset($matches[1])) {
+ return $matches[1];
+ }
+ }
+
+ return null;
+ }
+
+ /**
+ * Extrait l'ID BAN depuis l'ID complet (ex: cp22.12345 → 12345)
+ *
+ * @param string $fullId ID complet de l'adresse
+ * @return string|null ID BAN
+ */
+ private function extractAddressId(string $fullId): ?string
+ {
+ if (strpos($fullId, '.') !== false) {
+ $parts = explode('.', $fullId);
+ return $parts[1] ?? null;
+ }
+ return $fullId;
+ }
+
+ /**
+ * Vérifie si une table existe dans la base batiments
+ *
+ * @param string $tableName Nom de la table (ex: 'bat22')
+ * @return bool True si la table existe
+ */
+ private function tableExists(string $tableName): bool
+ {
+ try {
+ // SHOW TABLES LIKE ne supporte pas les placeholders PDO
+ // On échappe manuellement le nom de table (alphanumérique uniquement)
+ if (!preg_match('/^[a-zA-Z0-9_]+$/', $tableName)) {
+ error_log("Nom de table invalide : {$tableName}");
+ return false;
+ }
+
+ $stmt = $this->dbBuildings->query("SHOW TABLES LIKE '{$tableName}'");
+ return $stmt && $stmt->rowCount() > 0;
+ } catch (PDOException $e) {
+ error_log("Erreur lors de la vérification de la table {$tableName} : " . $e->getMessage());
+ return false;
+ }
+ }
+}
diff --git a/api/src/Services/DepartmentBoundaryService.php b/api/src/Services/DepartmentBoundaryService.php
index 830790a4..05105070 100644
--- a/api/src/Services/DepartmentBoundaryService.php
+++ b/api/src/Services/DepartmentBoundaryService.php
@@ -1,11 +1,20 @@
db = Database::getInstance();
+ $this->db = \Database::getInstance();
}
/**
diff --git a/api/src/Services/EmailTemplates.php b/api/src/Services/EmailTemplates.php
index 013a3a31..59f00716 100755
--- a/api/src/Services/EmailTemplates.php
+++ b/api/src/Services/EmailTemplates.php
@@ -2,6 +2,8 @@
declare(strict_types=1);
+namespace App\Services;
+
class EmailTemplates {
/**
* Template d'email de bienvenue
@@ -12,7 +14,7 @@ class EmailTemplates {
Votre compte a été créé avec succès sur GeoSector.
Identifiant : $username
Mot de passe : $password
- Vous pouvez vous connecter dès maintenant sur app.geosector.fr
+ Vous pouvez vous connecter dès maintenant sur app3.geosector.fr
À très bientôt,
L'équipe GeoSector";
}
@@ -36,7 +38,7 @@ class EmailTemplates {
Une fois que vous aurez reçu votre mot de passe, vous pourrez vous connecter sur
- app.geosector.fr
+ app3.geosector.fr
À très bientôt,
@@ -59,7 +61,7 @@ class EmailTemplates {
Vous pouvez maintenant vous connecter avec votre identifiant (reçu dans un email précédent)
- et ce mot de passe sur app.geosector.fr
+ et ce mot de passe sur app3.geosector.fr
Rappel : Ne communiquez jamais votre mot de passe à un tiers. L'équipe GeoSector
@@ -78,7 +80,7 @@ class EmailTemplates {
Bonjour $name,
Vous avez demandé la réinitialisation de votre mot de passe sur GeoSector.
Nouveau mot de passe : $password
- Vous pouvez vous connecter avec ce nouveau mot de passe sur app.geosector.fr
+ Vous pouvez vous connecter avec ce nouveau mot de passe sur app3.geosector.fr
À très bientôt,
L'équipe GeoSector";
}
@@ -95,7 +97,8 @@ class EmailTemplates {
}
/**
- * Template de reçu de passage
+ * Template de reçu de passage (ancien format simple)
+ * @deprecated Utiliser getReceiptDonationTemplate() pour les reçus de don
*/
public static function getReceiptTemplate(string $name, string $date, string $address, string $amount, string $paymentMethod): string {
return "
@@ -126,4 +129,69 @@ class EmailTemplates {
À bientôt,
L'équipe GeoSector
";
}
+
+ /**
+ * Template d'email pour reçu de don avec pièce jointe PDF
+ * Utilisé lors de l'envoi automatique des reçus pour les dons (fk_type=1)
+ *
+ * @param array $data Données du reçu (passage_id, entite_name, donor_name, amount, donation_date, payment_method, entite_address, entite_email)
+ * @return string HTML de l'email
+ */
+ public static function getReceiptDonationTemplate(array $data): string {
+ // Extraction des données avec valeurs par défaut
+ $passageId = $data['passage_id'] ?? '';
+ $entiteName = htmlspecialchars($data['entite_name'] ?? 'Amicale des Sapeurs-Pompiers');
+ $donorName = htmlspecialchars($data['donor_name'] ?? '');
+ $amount = htmlspecialchars($data['amount'] ?? '0,00');
+ $donationDate = htmlspecialchars($data['donation_date'] ?? date('d/m/Y'));
+ $paymentMethod = htmlspecialchars($data['payment_method'] ?? 'Espèces');
+ $entiteAddress = htmlspecialchars($data['entite_address'] ?? '');
+ $entiteEmail = htmlspecialchars($data['entite_email'] ?? '');
+
+ return "
+
+
+
+
+
+
+
+
+
$passageId
+
+
+
+
+
Bonjour Mme/M. $donorName,
+
+
Nous vous remercions chaleureusement pour votre don de $amount € effectué le $donationDate.
+
+
Mode de paiement : $paymentMethod
+
+
Vous trouverez ci-joint votre reçu.
+
+
Votre soutien est précieux pour nous permettre de poursuivre nos actions.
+
+
Cordialement,
L'équipe de $entiteName
+
+
+
+
+
+";
+ }
}
diff --git a/api/src/Services/EventLogService.php b/api/src/Services/EventLogService.php
new file mode 100644
index 00000000..fdce06a6
--- /dev/null
+++ b/api/src/Services/EventLogService.php
@@ -0,0 +1,533 @@
+ $userId,
+ 'entity_id' => $entityId,
+ 'username' => $username
+ ]);
+ }
+
+ /**
+ * Log une tentative de connexion échouée
+ *
+ * @param string $username Nom d'utilisateur tenté
+ * @param string $reason Raison (invalid_password, user_not_found, account_inactive, blocked_ip)
+ * @param int $attempt Numéro de tentative
+ */
+ public static function logLoginFailed(string $username, string $reason, int $attempt = 1): void
+ {
+ self::writeEvent('login_failed', [
+ 'username' => $username,
+ 'reason' => $reason,
+ 'attempt' => $attempt
+ ]);
+ }
+
+ /**
+ * Log une déconnexion
+ *
+ * @param int $userId ID utilisateur
+ * @param int|null $entityId ID entité
+ * @param int $sessionDuration Durée session en secondes
+ */
+ public static function logLogout(int $userId, ?int $entityId, int $sessionDuration = 0): void
+ {
+ self::writeEvent('logout', [
+ 'user_id' => $userId,
+ 'entity_id' => $entityId,
+ 'session_duration' => $sessionDuration
+ ]);
+ }
+
+ // ==================== MÉTHODES PASSAGES ====================
+
+ /**
+ * Log la création d'un passage
+ *
+ * @param int $passageId ID du passage créé
+ * @param int $operationId ID opération
+ * @param int $sectorId ID secteur
+ * @param float $amount Montant
+ * @param string $paymentType Type paiement (cash, stripe, check, etc.)
+ */
+ public static function logPassageCreated(
+ int $passageId,
+ int $operationId,
+ int $sectorId,
+ float $amount,
+ string $paymentType
+ ): void {
+ self::writeEvent('passage_created', [
+ 'passage_id' => $passageId,
+ 'operation_id' => $operationId,
+ 'sector_id' => $sectorId,
+ 'amount' => $amount,
+ 'payment_type' => $paymentType
+ ]);
+ }
+
+ /**
+ * Log la modification d'un passage
+ *
+ * @param int $passageId ID du passage
+ * @param array $changes Tableau des changements ['field' => ['old' => val, 'new' => val]]
+ */
+ public static function logPassageUpdated(int $passageId, array $changes): void
+ {
+ self::writeEvent('passage_updated', [
+ 'passage_id' => $passageId,
+ 'changes' => $changes
+ ]);
+ }
+
+ /**
+ * Log la suppression d'un passage
+ *
+ * @param int $passageId ID du passage
+ * @param int $operationId ID opération
+ * @param bool $softDelete Suppression logique ou physique
+ */
+ public static function logPassageDeleted(int $passageId, int $operationId, bool $softDelete = true): void
+ {
+ $userId = Session::getUserId();
+ self::writeEvent('passage_deleted', [
+ 'passage_id' => $passageId,
+ 'operation_id' => $operationId,
+ 'deleted_by' => $userId,
+ 'soft_delete' => $softDelete
+ ]);
+ }
+
+ // ==================== MÉTHODES SECTEURS ====================
+
+ /**
+ * Log la création d'un secteur
+ *
+ * @param int $sectorId ID du secteur créé
+ * @param int $operationId ID opération
+ * @param string $sectorName Nom du secteur
+ */
+ public static function logSectorCreated(int $sectorId, int $operationId, string $sectorName): void
+ {
+ self::writeEvent('sector_created', [
+ 'sector_id' => $sectorId,
+ 'operation_id' => $operationId,
+ 'sector_name' => $sectorName
+ ]);
+ }
+
+ /**
+ * Log la modification d'un secteur
+ *
+ * @param int $sectorId ID du secteur
+ * @param int $operationId ID opération
+ * @param array $changes Tableau des changements
+ */
+ public static function logSectorUpdated(int $sectorId, int $operationId, array $changes): void
+ {
+ self::writeEvent('sector_updated', [
+ 'sector_id' => $sectorId,
+ 'operation_id' => $operationId,
+ 'changes' => $changes
+ ]);
+ }
+
+ /**
+ * Log la suppression d'un secteur
+ *
+ * @param int $sectorId ID du secteur
+ * @param int $operationId ID opération
+ * @param bool $softDelete Suppression logique ou physique
+ */
+ public static function logSectorDeleted(int $sectorId, int $operationId, bool $softDelete = true): void
+ {
+ $userId = Session::getUserId();
+ self::writeEvent('sector_deleted', [
+ 'sector_id' => $sectorId,
+ 'operation_id' => $operationId,
+ 'deleted_by' => $userId,
+ 'soft_delete' => $softDelete
+ ]);
+ }
+
+ // ==================== MÉTHODES USERS ====================
+
+ /**
+ * Log la création d'un utilisateur
+ *
+ * @param int $newUserId ID utilisateur créé
+ * @param int $entityId ID entité
+ * @param int $roleId ID rôle
+ * @param string $username Nom d'utilisateur
+ */
+ public static function logUserCreated(int $newUserId, int $entityId, int $roleId, string $username): void
+ {
+ $createdBy = Session::getUserId();
+ self::writeEvent('user_created', [
+ 'new_user_id' => $newUserId,
+ 'entity_id' => $entityId,
+ 'created_by' => $createdBy,
+ 'role_id' => $roleId,
+ 'username' => $username
+ ]);
+ }
+
+ /**
+ * Log la modification d'un utilisateur
+ *
+ * @param int $userId ID utilisateur modifié
+ * @param array $changes Tableau des changements (booléen pour champs chiffrés)
+ */
+ public static function logUserUpdated(int $userId, array $changes): void
+ {
+ $updatedBy = Session::getUserId();
+ $entityId = Session::getEntityId();
+
+ self::writeEvent('user_updated', [
+ 'user_id' => $userId,
+ 'entity_id' => $entityId,
+ 'updated_by' => $updatedBy,
+ 'changes' => $changes
+ ]);
+ }
+
+ /**
+ * Log la suppression d'un utilisateur
+ *
+ * @param int $userId ID utilisateur supprimé
+ * @param bool $softDelete Suppression logique ou physique
+ */
+ public static function logUserDeleted(int $userId, bool $softDelete = true): void
+ {
+ $deletedBy = Session::getUserId();
+ $entityId = Session::getEntityId();
+
+ self::writeEvent('user_deleted', [
+ 'user_id' => $userId,
+ 'entity_id' => $entityId,
+ 'deleted_by' => $deletedBy,
+ 'soft_delete' => $softDelete
+ ]);
+ }
+
+ // ==================== MÉTHODES ENTITÉS ====================
+
+ /**
+ * Log la création d'une entité
+ *
+ * @param int $entityId ID entité créée
+ * @param int $entityTypeId Type d'entité
+ * @param string $postalCode Code postal
+ */
+ public static function logEntityCreated(int $entityId, int $entityTypeId, string $postalCode): void
+ {
+ $createdBy = Session::getUserId() ?? 1; // Super-admin par défaut
+ self::writeEvent('entity_created', [
+ 'entity_id' => $entityId,
+ 'created_by' => $createdBy,
+ 'entity_type_id' => $entityTypeId,
+ 'postal_code' => $postalCode
+ ]);
+ }
+
+ /**
+ * Log la modification d'une entité
+ *
+ * @param int $entityId ID entité
+ * @param array $changes Tableau des changements (booléen pour champs chiffrés)
+ */
+ public static function logEntityUpdated(int $entityId, array $changes): void
+ {
+ $updatedBy = Session::getUserId();
+ self::writeEvent('entity_updated', [
+ 'entity_id' => $entityId,
+ 'user_id' => $updatedBy,
+ 'updated_by' => $updatedBy,
+ 'changes' => $changes
+ ]);
+ }
+
+ /**
+ * Log la suppression d'une entité
+ *
+ * @param int $entityId ID entité
+ * @param string $reason Raison de la suppression
+ */
+ public static function logEntityDeleted(int $entityId, string $reason = ''): void
+ {
+ $deletedBy = Session::getUserId() ?? 1;
+ self::writeEvent('entity_deleted', [
+ 'entity_id' => $entityId,
+ 'deleted_by' => $deletedBy,
+ 'soft_delete' => true,
+ 'reason' => $reason
+ ]);
+ }
+
+ // ==================== MÉTHODES OPÉRATIONS ====================
+
+ /**
+ * Log la création d'une opération
+ *
+ * @param int $operationId ID opération créée
+ * @param string $dateStart Date début (YYYY-MM-DD)
+ * @param string $dateEnd Date fin (YYYY-MM-DD)
+ */
+ public static function logOperationCreated(int $operationId, string $dateStart, string $dateEnd): void
+ {
+ $entityId = Session::getEntityId();
+ $createdBy = Session::getUserId();
+
+ self::writeEvent('operation_created', [
+ 'operation_id' => $operationId,
+ 'entity_id' => $entityId,
+ 'created_by' => $createdBy,
+ 'date_start' => $dateStart,
+ 'date_end' => $dateEnd
+ ]);
+ }
+
+ /**
+ * Log la modification d'une opération
+ *
+ * @param int $operationId ID opération
+ * @param array $changes Tableau des changements
+ */
+ public static function logOperationUpdated(int $operationId, array $changes): void
+ {
+ $entityId = Session::getEntityId();
+ $updatedBy = Session::getUserId();
+
+ self::writeEvent('operation_updated', [
+ 'operation_id' => $operationId,
+ 'entity_id' => $entityId,
+ 'updated_by' => $updatedBy,
+ 'changes' => $changes
+ ]);
+ }
+
+ /**
+ * Log la suppression d'une opération
+ *
+ * @param int $operationId ID opération
+ * @param bool $softDelete Suppression logique ou physique
+ */
+ public static function logOperationDeleted(int $operationId, bool $softDelete = true): void
+ {
+ $entityId = Session::getEntityId();
+ $deletedBy = Session::getUserId();
+
+ self::writeEvent('operation_deleted', [
+ 'operation_id' => $operationId,
+ 'entity_id' => $entityId,
+ 'deleted_by' => $deletedBy,
+ 'soft_delete' => $softDelete
+ ]);
+ }
+
+ // ==================== MÉTHODES PRIVÉES ====================
+
+ /**
+ * Méthode centrale d'écriture d'un événement
+ *
+ * @param string $eventName Nom de l'événement
+ * @param array $data Données spécifiques à l'événement
+ */
+ private static function writeEvent(string $eventName, array $data): void
+ {
+ try {
+ // Enrichir avec timestamp, user_id, entity_id, IP, platform, app_version
+ $event = self::enrichEvent($eventName, $data);
+
+ // Générer le chemin du fichier quotidien
+ $filename = self::LOG_DIR . '/' . date('Y-m-d') . '.jsonl';
+
+ // Créer le dossier si nécessaire
+ self::ensureLogDirectoryExists();
+
+ // Encoder en JSON compact (une ligne)
+ $jsonLine = json_encode($event, JSON_UNESCAPED_SLASHES | JSON_UNESCAPED_UNICODE) . "\n";
+
+ // Écrire en mode append
+ if (file_put_contents($filename, $jsonLine, FILE_APPEND | LOCK_EX) === false) {
+ error_log("EventLogService: Impossible d'écrire dans {$filename}");
+ return;
+ }
+
+ // Appliquer les permissions au fichier
+ if (file_exists($filename)) {
+ @chmod($filename, self::FILE_PERMISSIONS);
+ }
+ } catch (\Throwable $e) {
+ // Ne jamais bloquer l'application si le logging échoue
+ error_log("EventLogService: Erreur lors de l'écriture de l'événement {$eventName}: " . $e->getMessage());
+ }
+ }
+
+ /**
+ * Enrichit un événement avec les métadonnées communes
+ *
+ * @param string $eventName Nom de l'événement
+ * @param array $data Données de l'événement
+ * @return array Événement enrichi
+ */
+ private static function enrichEvent(string $eventName, array $data): array
+ {
+ // Récupérer les informations client
+ $clientInfo = ClientDetector::getClientInfo();
+
+ // Structure de base
+ $event = [
+ 'timestamp' => gmdate('Y-m-d\TH:i:s\Z'), // ISO 8601 UTC
+ 'event' => $eventName,
+ ];
+
+ // Ajouter user_id si disponible et pas déjà dans $data
+ if (!isset($data['user_id'])) {
+ $userId = Session::getUserId();
+ if ($userId !== null) {
+ $event['user_id'] = $userId;
+ }
+ }
+
+ // Ajouter entity_id si disponible et pas déjà dans $data
+ if (!isset($data['entity_id'])) {
+ $entityId = Session::getEntityId();
+ if ($entityId !== null) {
+ $event['entity_id'] = $entityId;
+ }
+ }
+
+ // Fusionner avec les données spécifiques
+ $event = array_merge($event, $data);
+
+ // Ajouter IP
+ $event['ip'] = $clientInfo['ip'];
+
+ // Ajouter platform
+ $event['platform'] = self::getPlatform($clientInfo);
+
+ // Ajouter app_version si mobile
+ if ($event['platform'] === 'ios' || $event['platform'] === 'android') {
+ $appVersion = self::getAppVersion($clientInfo);
+ if ($appVersion !== null) {
+ $event['app_version'] = $appVersion;
+ }
+ }
+
+ return $event;
+ }
+
+ /**
+ * Détermine la plateforme (ios, android, web)
+ *
+ * @param array $clientInfo Informations client de ClientDetector
+ * @return string Platform (ios|android|web)
+ */
+ private static function getPlatform(array $clientInfo): string
+ {
+ if ($clientInfo['type'] !== 'mobile') {
+ return 'web';
+ }
+
+ $userAgent = $clientInfo['userAgent'];
+
+ // Détection iOS
+ if (stripos($userAgent, 'iOS') !== false ||
+ stripos($userAgent, 'iPhone') !== false ||
+ stripos($userAgent, 'iPad') !== false) {
+ return 'ios';
+ }
+
+ // Détection Android
+ if (stripos($userAgent, 'Android') !== false) {
+ return 'android';
+ }
+
+ // Par défaut mobile générique = web
+ return 'web';
+ }
+
+ /**
+ * Extrait la version de l'application depuis le User-Agent
+ * Format attendu: AppName/VersionNumber ou Platform/Version AppName/Version
+ *
+ * @param array $clientInfo Informations client
+ * @return string|null Version de l'app ou null
+ */
+ private static function getAppVersion(array $clientInfo): ?string
+ {
+ $userAgent = $clientInfo['userAgent'];
+
+ // Tentative extraction format: GeoSector/3.3.6
+ if (preg_match('/GeoSector\/([0-9\.]+)/', $userAgent, $matches)) {
+ return $matches[1];
+ }
+
+ // Format alternatif: AppName/Version
+ if (preg_match('/([A-Za-z0-9_]+)\/([0-9\.]+)/', $userAgent, $matches)) {
+ return $matches[2];
+ }
+
+ return null;
+ }
+
+ /**
+ * Crée le dossier de logs si nécessaire avec les bonnes permissions
+ */
+ private static function ensureLogDirectoryExists(): void
+ {
+ if (!is_dir(self::LOG_DIR)) {
+ if (!@mkdir(self::LOG_DIR, self::DIR_PERMISSIONS, true)) {
+ error_log("EventLogService: Impossible de créer le dossier " . self::LOG_DIR);
+ return;
+ }
+ }
+
+ // Vérifier les permissions
+ if (!is_writable(self::LOG_DIR)) {
+ @chmod(self::LOG_DIR, self::DIR_PERMISSIONS);
+ }
+ }
+}
diff --git a/api/src/Services/ExportService.php b/api/src/Services/ExportService.php
index 2ac7cca1..f035e5a7 100755
--- a/api/src/Services/ExportService.php
+++ b/api/src/Services/ExportService.php
@@ -2,17 +2,24 @@
declare(strict_types=1);
+namespace App\Services;
+
require_once __DIR__ . '/../../vendor/autoload.php';
use PhpOffice\PhpSpreadsheet\Spreadsheet;
use PhpOffice\PhpSpreadsheet\Writer\Xls;
use PhpOffice\PhpSpreadsheet\Writer\Csv;
+use PDO;
+use Database;
+use Session;
+use App\Services\LogService;
+use App\Services\ApiService;
require_once __DIR__ . '/../Services/FileService.php';
class ExportService {
- private \PDO $db;
+ private PDO $db;
private FileService $fileService;
public function __construct() {
@@ -249,10 +256,11 @@ class ExportService {
p.fk_habitat, p.appt, p.niveau, p.encrypted_name, p.encrypted_email,
p.encrypted_phone, p.montant, p.fk_type_reglement, p.remarque,
p.fk_user, p.fk_sector, p.fk_operation,
- u.encrypted_name as user_name, u.first_name as user_first_name, u.sect_name,
+ u.encrypted_name as user_name, ou.first_name as user_first_name, u.sect_name,
xtr.libelle as reglement_libelle
FROM ope_pass p
- LEFT JOIN users u ON u.id = p.fk_user
+ LEFT JOIN ope_users ou ON ou.id = p.fk_user
+ LEFT JOIN users u ON u.id = ou.fk_user
LEFT JOIN x_types_reglements xtr ON xtr.id = p.fk_type_reglement
WHERE p.fk_operation = ? AND p.chk_active = 1
';
@@ -457,10 +465,11 @@ class ExportService {
SELECT
ous.id, ous.fk_sector, ous.fk_user, ous.created_at, ous.fk_operation,
s.libelle as sector_name,
- u.encrypted_name as user_name, u.first_name
+ u.encrypted_name as user_name, ou.first_name
FROM ope_users_sectors ous
INNER JOIN ope_sectors s ON s.id = ous.fk_sector
- INNER JOIN users u ON u.id = ous.fk_user
+ INNER JOIN ope_users ou ON ou.id = ous.fk_user
+ INNER JOIN users u ON u.id = ou.fk_user
WHERE ous.fk_operation = ? AND ous.chk_active = 1
ORDER BY s.libelle, u.encrypted_name
';
@@ -619,10 +628,11 @@ class ExportService {
p.passed_at, p.fk_type, p.numero, p.rue_bis, p.rue, p.ville,
p.fk_habitat, p.appt, p.niveau, p.encrypted_name, p.encrypted_email,
p.encrypted_phone, p.montant, p.fk_type_reglement, p.remarque,
- u.encrypted_name as user_name, u.first_name as user_first_name, u.sect_name,
+ u.encrypted_name as user_name, ou.first_name as user_first_name, u.sect_name,
xtr.libelle as reglement_libelle
FROM ope_pass p
- LEFT JOIN users u ON u.id = p.fk_user
+ LEFT JOIN ope_users ou ON ou.id = p.fk_user
+ LEFT JOIN users u ON u.id = ou.fk_user
LEFT JOIN x_types_reglements xtr ON xtr.id = p.fk_type_reglement
WHERE p.fk_operation = ? AND p.chk_active = 1
';
diff --git a/api/src/Services/FileService.php b/api/src/Services/FileService.php
index 01cb3d95..181c4bdb 100755
--- a/api/src/Services/FileService.php
+++ b/api/src/Services/FileService.php
@@ -2,8 +2,15 @@
declare(strict_types=1);
+namespace App\Services;
+
require_once __DIR__ . '/../../vendor/autoload.php';
+use PDO;
+use Database;
+use Session;
+use App\Services\LogService;
+
class FileService {
private const BASE_UPLOADS_DIR = '/var/www/geosector/api/uploads';
diff --git a/api/src/Services/LogService.php b/api/src/Services/LogService.php
index 1ba53a81..f953e5fe 100755
--- a/api/src/Services/LogService.php
+++ b/api/src/Services/LogService.php
@@ -2,6 +2,10 @@
declare(strict_types=1);
+namespace App\Services;
+
+use AppConfig;
+use ClientDetector;
class LogService {
public static function log(string $message, array $metadata = []): void {
diff --git a/api/src/Services/MigrationService.php b/api/src/Services/MigrationService.php
new file mode 100644
index 00000000..0b256733
--- /dev/null
+++ b/api/src/Services/MigrationService.php
@@ -0,0 +1,791 @@
+targetDb = Database::getInstance();
+ $this->appConfig = AppConfig::getInstance();
+ }
+
+ public function __destruct() {
+ $this->closeSshTunnel();
+ }
+
+ /**
+ * Crée un tunnel SSH vers le serveur distant
+ */
+ private function createSshTunnel(): bool {
+ if ($this->sshTunnelCreated) {
+ return true;
+ }
+
+ // Vérifier si un tunnel est déjà en cours d'exécution
+ $checkCommand = "ps aux | grep 'ssh -[vf]* -N -L " . self::SOURCE_DB_PORT . ":' | grep -v grep";
+ exec($checkCommand, $output, $return_var);
+
+ if (empty($output)) {
+ // Créer le tunnel SSH
+ $command = sprintf(
+ "ssh -f -N -o StrictHostKeyChecking=no -L %d:%s:%d -p %d %s@%s -i %s 2>&1",
+ self::SOURCE_DB_PORT,
+ self::REMOTE_DB_HOST,
+ self::REMOTE_DB_PORT,
+ self::SSH_PORT,
+ self::SSH_USER,
+ self::SSH_HOST,
+ self::SSH_KEY_FILE
+ );
+
+ exec($command, $output, $return_var);
+
+ if ($return_var !== 0) {
+ LogService::log('Erreur lors de la création du tunnel SSH', [
+ 'level' => 'error',
+ 'output' => implode("\n", $output)
+ ]);
+ return false;
+ }
+
+ // Attendre que le tunnel soit établi
+ sleep(2);
+
+ // Vérification du tunnel
+ $checkTunnel = "netstat -an | grep " . self::SOURCE_DB_PORT . " | grep LISTEN";
+ exec($checkTunnel, $tunnelOutput);
+
+ if (empty($tunnelOutput)) {
+ LogService::log('Le tunnel SSH semble créé mais le port n\'est pas en écoute', [
+ 'level' => 'warning',
+ 'port' => self::SOURCE_DB_PORT
+ ]);
+ }
+
+ LogService::log('Tunnel SSH établi', [
+ 'level' => 'info',
+ 'port' => self::SOURCE_DB_PORT
+ ]);
+ } else {
+ LogService::log('Un tunnel SSH est déjà en cours d\'exécution', [
+ 'level' => 'info'
+ ]);
+ }
+
+ $this->sshTunnelCreated = true;
+ return true;
+ }
+
+ /**
+ * Ferme le tunnel SSH
+ */
+ private function closeSshTunnel(): void {
+ if (!$this->sshTunnelCreated) {
+ return;
+ }
+
+ $command = "ps aux | grep 'ssh -[vf]* -N -L " . self::SOURCE_DB_PORT . ":' | grep -v grep | awk '{print $2}' | xargs -r kill 2>/dev/null";
+ exec($command);
+
+ $this->sshTunnelCreated = false;
+ $this->sourceDb = null;
+
+ LogService::log('Tunnel SSH fermé', ['level' => 'info']);
+ }
+
+ /**
+ * Récupère la connexion à la base source
+ */
+ private function getSourceConnection(): PDO {
+ if ($this->sourceDb !== null) {
+ return $this->sourceDb;
+ }
+
+ // Établir le tunnel SSH
+ if (!$this->createSshTunnel()) {
+ throw new Exception("Impossible d'établir le tunnel SSH");
+ }
+
+ // Options de connexion PDO
+ $options = [
+ PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION,
+ PDO::ATTR_DEFAULT_FETCH_MODE => PDO::FETCH_ASSOC,
+ PDO::MYSQL_ATTR_INIT_COMMAND => 'SET NAMES utf8mb4',
+ PDO::ATTR_TIMEOUT => 600
+ ];
+
+ if (defined('PDO::MYSQL_ATTR_SSL_VERIFY_SERVER_CERT')) {
+ $options[PDO::MYSQL_ATTR_SSL_VERIFY_SERVER_CERT] = false;
+ }
+
+ try {
+ // Se connecter à la base spécifique
+ $dsn = sprintf(
+ 'mysql:host=%s;dbname=%s;port=%d',
+ self::SOURCE_DB_HOST,
+ self::SOURCE_DB_NAME,
+ self::SOURCE_DB_PORT
+ );
+
+ $this->sourceDb = new PDO($dsn, self::SOURCE_DB_USER, self::SOURCE_DB_PASS, $options);
+
+ LogService::log('Connexion établie à la base source', [
+ 'level' => 'info',
+ 'database' => self::SOURCE_DB_NAME
+ ]);
+
+ return $this->sourceDb;
+ } catch (PDOException $e) {
+ $this->closeSshTunnel();
+ throw new Exception("Erreur de connexion à la base source: " . $e->getMessage());
+ }
+ }
+
+ /**
+ * Teste les connexions aux deux bases de données
+ */
+ public function testConnections(): array {
+ $result = [
+ 'source' => ['status' => 'error', 'message' => ''],
+ 'target' => ['status' => 'error', 'message' => '']
+ ];
+
+ // Test connexion source
+ try {
+ $sourceDb = $this->getSourceConnection();
+ $stmt = $sourceDb->query('SELECT DATABASE() as db, VERSION() as version');
+ $info = $stmt->fetch();
+
+ $result['source'] = [
+ 'status' => 'success',
+ 'database' => $info['db'],
+ 'version' => $info['version'],
+ 'message' => 'Connexion réussie'
+ ];
+ } catch (Exception $e) {
+ $result['source']['message'] = $e->getMessage();
+ }
+
+ // Test connexion cible
+ try {
+ $stmt = $this->targetDb->query('SELECT DATABASE() as db, VERSION() as version');
+ $info = $stmt->fetch();
+
+ $result['target'] = [
+ 'status' => 'success',
+ 'database' => $info['db'],
+ 'version' => $info['version'],
+ 'message' => 'Connexion réussie'
+ ];
+ } catch (Exception $e) {
+ $result['target']['message'] = $e->getMessage();
+ }
+
+ return $result;
+ }
+
+ /**
+ * Liste les entités disponibles à migrer depuis la base source
+ */
+ public function getAvailableEntities(): array {
+ $sourceDb = $this->getSourceConnection();
+
+ $stmt = $sourceDb->query("
+ SELECT
+ rowid as id,
+ libelle as name,
+ cp as postal_code,
+ ville as city,
+ active,
+ date_creat as created_at
+ FROM users_entites
+ WHERE active = 1
+ ORDER BY libelle ASC
+ ");
+
+ return $stmt->fetchAll();
+ }
+
+ /**
+ * Récupère les détails d'une entité source
+ */
+ public function getEntityDetails(int $id): ?array {
+ $sourceDb = $this->getSourceConnection();
+
+ $stmt = $sourceDb->prepare("
+ SELECT
+ e.rowid as id,
+ e.libelle as name,
+ e.cp as postal_code,
+ e.ville as city,
+ e.active,
+ e.date_creat as created_at,
+ (SELECT COUNT(*) FROM users WHERE fk_entite = e.rowid) as users_count,
+ (SELECT COUNT(*) FROM operations WHERE fk_entite = e.rowid) as operations_count,
+ (SELECT COUNT(*) FROM ope_pass p
+ JOIN operations o ON p.fk_operation = o.rowid
+ WHERE o.fk_entite = e.rowid) as passages_count
+ FROM users_entites e
+ WHERE e.rowid = ?
+ ");
+
+ $stmt->execute([$id]);
+ $entity = $stmt->fetch();
+
+ return $entity ?: null;
+ }
+
+ /**
+ * Migre une entité complète
+ */
+ public function migrateEntity(
+ int $entityId,
+ ?array $steps = null,
+ bool $dryRun = false,
+ bool $truncate = false
+ ): array {
+ $startTime = microtime(true);
+ $migrationId = 'mig_' . time() . '_' . $entityId;
+
+ // Si aucune étape spécifiée, migrer toutes les étapes
+ if ($steps === null) {
+ $steps = self::MIGRATION_STEPS;
+ }
+
+ // Récupérer le nom de l'entité
+ $entityDetails = $this->getEntityDetails($entityId);
+ if (!$entityDetails) {
+ throw new Exception("Entité $entityId non trouvée");
+ }
+
+ $stepsCompleted = [];
+ $totalRecords = 0;
+ $totalErrors = 0;
+ $totalWarnings = 0;
+
+ LogService::log('Début de migration entité', [
+ 'level' => 'info',
+ 'migration_id' => $migrationId,
+ 'entity_id' => $entityId,
+ 'entity_name' => $entityDetails['name'],
+ 'steps' => $steps,
+ 'dry_run' => $dryRun
+ ]);
+
+ foreach ($steps as $step) {
+ try {
+ $stepResult = $this->migrateStep($entityId, $step, $dryRun, []);
+ $stepsCompleted[] = [
+ 'step' => $step,
+ 'status' => 'success',
+ 'records_migrated' => $stepResult['records_migrated'],
+ 'duration_ms' => $stepResult['duration_ms']
+ ];
+
+ $totalRecords += $stepResult['records_migrated'];
+ $totalWarnings += count($stepResult['warnings'] ?? []);
+ } catch (Exception $e) {
+ $stepsCompleted[] = [
+ 'step' => $step,
+ 'status' => 'error',
+ 'error' => $e->getMessage(),
+ 'duration_ms' => 0
+ ];
+ $totalErrors++;
+
+ LogService::log('Erreur lors de la migration d\'une étape', [
+ 'level' => 'error',
+ 'migration_id' => $migrationId,
+ 'step' => $step,
+ 'error' => $e->getMessage()
+ ]);
+
+ // Arrêter la migration en cas d'erreur critique
+ break;
+ }
+ }
+
+ $totalDuration = (microtime(true) - $startTime) * 1000;
+
+ LogService::log('Fin de migration entité', [
+ 'level' => 'info',
+ 'migration_id' => $migrationId,
+ 'entity_id' => $entityId,
+ 'total_records' => $totalRecords,
+ 'total_errors' => $totalErrors,
+ 'duration_ms' => $totalDuration
+ ]);
+
+ return [
+ 'entity_name' => $entityDetails['name'],
+ 'migration_id' => $migrationId,
+ 'steps_completed' => $stepsCompleted,
+ 'total_duration_ms' => round($totalDuration, 2),
+ 'summary' => [
+ 'total_records' => $totalRecords,
+ 'total_errors' => $totalErrors,
+ 'total_warnings' => $totalWarnings
+ ]
+ ];
+ }
+
+ /**
+ * Migre une étape spécifique pour une entité
+ */
+ public function migrateStep(
+ int $entityId,
+ string $step,
+ bool $dryRun = false,
+ array $options = []
+ ): array {
+ $startTime = microtime(true);
+
+ LogService::log("Début migration étape: $step", [
+ 'level' => 'info',
+ 'entity_id' => $entityId,
+ 'step' => $step,
+ 'dry_run' => $dryRun
+ ]);
+
+ // Appeler la méthode spécifique pour chaque étape
+ $methodName = 'migrate' . $this->snakeToPascal($step);
+
+ if (!method_exists($this, $methodName)) {
+ throw new Exception("Méthode de migration non trouvée pour l'étape: $step");
+ }
+
+ $result = $this->$methodName($entityId, $dryRun, $options);
+
+ $duration = (microtime(true) - $startTime) * 1000;
+
+ LogService::log("Fin migration étape: $step", [
+ 'level' => 'info',
+ 'entity_id' => $entityId,
+ 'records_migrated' => $result['records_migrated'],
+ 'duration_ms' => $duration
+ ]);
+
+ return [
+ 'records_migrated' => $result['records_migrated'],
+ 'duration_ms' => round($duration, 2),
+ 'warnings' => $result['warnings'] ?? [],
+ 'details' => $result['details'] ?? []
+ ];
+ }
+
+ /**
+ * Convertit snake_case en PascalCase
+ */
+ private function snakeToPascal(string $string): string {
+ return str_replace('_', '', ucwords($string, '_'));
+ }
+
+ /**
+ * Migration des devises (x_devises)
+ */
+ private function migrateXDevises(int $entityId, bool $dryRun, array $options): array {
+ // Les tables x_* sont globales, pas liées à une entité
+ $sourceDb = $this->getSourceConnection();
+
+ $stmt = $sourceDb->query("SELECT * FROM x_devises WHERE active = 1");
+ $records = $stmt->fetchAll();
+
+ if ($dryRun) {
+ return ['records_migrated' => count($records), 'warnings' => []];
+ }
+
+ $inserted = 0;
+ foreach ($records as $record) {
+ $stmt = $this->targetDb->prepare("
+ INSERT INTO x_devises (id, code, libelle, symbole, chk_active, created_at, updated_at)
+ VALUES (?, ?, ?, ?, ?, NOW(), NOW())
+ ON DUPLICATE KEY UPDATE
+ code = VALUES(code),
+ libelle = VALUES(libelle),
+ symbole = VALUES(symbole),
+ chk_active = VALUES(chk_active),
+ updated_at = NOW()
+ ");
+
+ $stmt->execute([
+ $record['rowid'],
+ $record['code'],
+ $record['libelle'],
+ $record['symbole'],
+ $record['active']
+ ]);
+
+ $inserted++;
+ }
+
+ return ['records_migrated' => $inserted, 'warnings' => []];
+ }
+
+ /**
+ * Migration des types d'entités (x_entites_types)
+ */
+ private function migrateXEntitesTypes(int $entityId, bool $dryRun, array $options): array {
+ $sourceDb = $this->getSourceConnection();
+
+ $stmt = $sourceDb->query("SELECT * FROM x_entites_types WHERE active = 1");
+ $records = $stmt->fetchAll();
+
+ if ($dryRun) {
+ return ['records_migrated' => count($records), 'warnings' => []];
+ }
+
+ $inserted = 0;
+ foreach ($records as $record) {
+ $stmt = $this->targetDb->prepare("
+ INSERT INTO x_entites_types (id, libelle, chk_active, created_at, updated_at)
+ VALUES (?, ?, ?, NOW(), NOW())
+ ON DUPLICATE KEY UPDATE
+ libelle = VALUES(libelle),
+ chk_active = VALUES(chk_active),
+ updated_at = NOW()
+ ");
+
+ $stmt->execute([
+ $record['rowid'],
+ $record['libelle'],
+ $record['active']
+ ]);
+
+ $inserted++;
+ }
+
+ return ['records_migrated' => $inserted, 'warnings' => []];
+ }
+
+ /**
+ * Méthodes de migration pour les autres tables x_* (structure similaire)
+ * TODO: Implémenter les autres méthodes
+ */
+ private function migrateXTypesPassages(int $entityId, bool $dryRun, array $options): array {
+ // TODO: À implémenter
+ return ['records_migrated' => 0, 'warnings' => ['Non implémenté']];
+ }
+
+ private function migrateXTypesReglements(int $entityId, bool $dryRun, array $options): array {
+ // TODO: À implémenter
+ return ['records_migrated' => 0, 'warnings' => ['Non implémenté']];
+ }
+
+ private function migrateXUsersRoles(int $entityId, bool $dryRun, array $options): array {
+ // TODO: À implémenter
+ return ['records_migrated' => 0, 'warnings' => ['Non implémenté']];
+ }
+
+ private function migrateXPays(int $entityId, bool $dryRun, array $options): array {
+ // TODO: À implémenter
+ return ['records_migrated' => 0, 'warnings' => ['Non implémenté']];
+ }
+
+ private function migrateXRegions(int $entityId, bool $dryRun, array $options): array {
+ // TODO: À implémenter
+ return ['records_migrated' => 0, 'warnings' => ['Non implémenté']];
+ }
+
+ private function migrateXDepartements(int $entityId, bool $dryRun, array $options): array {
+ // TODO: À implémenter
+ return ['records_migrated' => 0, 'warnings' => ['Non implémenté']];
+ }
+
+ private function migrateXVilles(int $entityId, bool $dryRun, array $options): array {
+ // TODO: À implémenter
+ return ['records_migrated' => 0, 'warnings' => ['Non implémenté']];
+ }
+
+ /**
+ * Migration de l'entité (users_entites -> entites)
+ */
+ private function migrateEntites(int $entityId, bool $dryRun, array $options): array {
+ $sourceDb = $this->getSourceConnection();
+
+ $stmt = $sourceDb->prepare("SELECT * FROM users_entites WHERE rowid = ?");
+ $stmt->execute([$entityId]);
+ $entity = $stmt->fetch();
+
+ if (!$entity) {
+ throw new Exception("Entité $entityId non trouvée dans la base source");
+ }
+
+ if ($dryRun) {
+ return ['records_migrated' => 1, 'warnings' => []];
+ }
+
+ // Chiffrement des données sensibles
+ $encryptedName = ApiService::encryptData($entity['libelle']);
+ $encryptedEmail = !empty($entity['email']) ? ApiService::encryptSearchableData($entity['email']) : null;
+
+ // Gestion des téléphones
+ $phone = $entity['tel1'] ?? '';
+ $mobile = $entity['tel2'] ?? '';
+
+ // Détection mobile (commence par 06 ou 07)
+ if (preg_match('/^0[67]/', $phone)) {
+ $mobile = $phone;
+ $phone = $entity['tel2'] ?? '';
+ }
+
+ $encryptedPhone = !empty($phone) ? ApiService::encryptData($phone) : null;
+ $encryptedMobile = !empty($mobile) ? ApiService::encryptData($mobile) : null;
+ $encryptedIban = !empty($entity['iban']) ? ApiService::encryptData($entity['iban']) : null;
+ $encryptedBic = !empty($entity['bic']) ? ApiService::encryptData($entity['bic']) : null;
+
+ $stmt = $this->targetDb->prepare("
+ INSERT INTO entites (
+ id, fk_type, encrypted_name, encrypted_email, encrypted_phone, encrypted_mobile,
+ code_postal, ville, encrypted_iban, encrypted_bic, chk_active, chk_demo,
+ created_at, updated_at
+ ) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, 0, NOW(), NOW())
+ ON DUPLICATE KEY UPDATE
+ fk_type = VALUES(fk_type),
+ encrypted_name = VALUES(encrypted_name),
+ encrypted_email = VALUES(encrypted_email),
+ encrypted_phone = VALUES(encrypted_phone),
+ encrypted_mobile = VALUES(encrypted_mobile),
+ code_postal = VALUES(code_postal),
+ ville = VALUES(ville),
+ encrypted_iban = VALUES(encrypted_iban),
+ encrypted_bic = VALUES(encrypted_bic),
+ chk_active = VALUES(chk_active),
+ updated_at = NOW()
+ ");
+
+ $stmt->execute([
+ $entity['rowid'],
+ $entity['fk_type'] ?? 1,
+ $encryptedName,
+ $encryptedEmail,
+ $encryptedPhone,
+ $encryptedMobile,
+ $entity['cp'] ?? null,
+ $entity['ville'] ?? null,
+ $encryptedIban,
+ $encryptedBic,
+ $entity['active']
+ ]);
+
+ return ['records_migrated' => 1, 'warnings' => []];
+ }
+
+ /**
+ * Migration des utilisateurs
+ */
+ private function migrateUsers(int $entityId, bool $dryRun, array $options): array {
+ // TODO: À implémenter (voir migrate_users.php pour la logique)
+ return ['records_migrated' => 0, 'warnings' => ['Non implémenté']];
+ }
+
+ /**
+ * Migration des opérations
+ */
+ private function migrateOperations(int $entityId, bool $dryRun, array $options): array {
+ // TODO: À implémenter
+ return ['records_migrated' => 0, 'warnings' => ['Non implémenté']];
+ }
+
+ /**
+ * Autres méthodes de migration
+ * TODO: Implémenter toutes les méthodes pour chaque étape
+ */
+ private function migrateOpeSectors(int $entityId, bool $dryRun, array $options): array {
+ return ['records_migrated' => 0, 'warnings' => ['Non implémenté']];
+ }
+
+ private function migrateSectorsAdresses(int $entityId, bool $dryRun, array $options): array {
+ return ['records_migrated' => 0, 'warnings' => ['Non implémenté']];
+ }
+
+ private function migrateOpeUsers(int $entityId, bool $dryRun, array $options): array {
+ return ['records_migrated' => 0, 'warnings' => ['Non implémenté']];
+ }
+
+ private function migrateOpeUsersSectors(int $entityId, bool $dryRun, array $options): array {
+ return ['records_migrated' => 0, 'warnings' => ['Non implémenté']];
+ }
+
+ private function migrateOpePass(int $entityId, bool $dryRun, array $options): array {
+ return ['records_migrated' => 0, 'warnings' => ['Non implémenté']];
+ }
+
+ private function migrateOpePassHisto(int $entityId, bool $dryRun, array $options): array {
+ return ['records_migrated' => 0, 'warnings' => ['Non implémenté']];
+ }
+
+ private function migrateMedias(int $entityId, bool $dryRun, array $options): array {
+ return ['records_migrated' => 0, 'warnings' => ['Non implémenté']];
+ }
+
+ /**
+ * Récupère le statut de migration d'une entité
+ */
+ public function getMigrationStatus(int $entityId): array {
+ // TODO: Implémenter la vérification du statut
+ return [
+ 'entity_id' => $entityId,
+ 'migrated' => false,
+ 'steps_completed' => [],
+ 'last_migration_date' => null
+ ];
+ }
+
+ /**
+ * Récupère les logs de migration d'une entité
+ */
+ public function getMigrationLogs(int $entityId): array {
+ // TODO: Lire les logs depuis LogService
+ return [];
+ }
+
+ /**
+ * Génère un rapport de migration
+ */
+ public function generateMigrationReport(int $entityId): array {
+ // TODO: Implémenter la génération de rapport
+ return [
+ 'entity_id' => $entityId,
+ 'generated_at' => date('Y-m-d H:i:s'),
+ 'summary' => []
+ ];
+ }
+
+ /**
+ * Compare les données source vs cible
+ */
+ public function compareEntityData(int $entityId): array {
+ $sourceDb = $this->getSourceConnection();
+
+ $comparison = [];
+
+ // Comparer les counts pour chaque table
+ $tables = [
+ 'users' => 'fk_entite',
+ 'operations' => 'fk_entite'
+ ];
+
+ foreach ($tables as $table => $fkColumn) {
+ $sourceTable = $table;
+ $targetTable = $table;
+
+ // Cas spéciaux
+ if ($table === 'users_entites') {
+ $sourceTable = 'users_entites';
+ $targetTable = 'entites';
+ }
+
+ // Count source
+ $stmt = $sourceDb->prepare("SELECT COUNT(*) as count FROM $sourceTable WHERE $fkColumn = ?");
+ $stmt->execute([$entityId]);
+ $sourceCount = $stmt->fetch()['count'];
+
+ // Count cible
+ $stmt = $this->targetDb->prepare("SELECT COUNT(*) as count FROM $targetTable WHERE $fkColumn = ?");
+ $stmt->execute([$entityId]);
+ $targetCount = $stmt->fetch()['count'];
+
+ $comparison[$table] = [
+ 'source_count' => $sourceCount,
+ 'target_count' => $targetCount,
+ 'difference' => $targetCount - $sourceCount,
+ 'status' => $sourceCount === $targetCount ? 'ok' : 'warning'
+ ];
+ }
+
+ return $comparison;
+ }
+
+ /**
+ * Vérifie l'intégrité des données migrées
+ */
+ public function verifyMigration(int $entityId): array {
+ // TODO: Implémenter les vérifications d'intégrité
+ return [
+ 'entity_id' => $entityId,
+ 'verified_at' => date('Y-m-d H:i:s'),
+ 'checks' => [],
+ 'errors' => [],
+ 'warnings' => []
+ ];
+ }
+
+ /**
+ * Annule la migration d'une entité (rollback)
+ */
+ public function rollbackEntity(int $entityId): array {
+ // TODO: Implémenter le rollback complet
+ $deletedRecords = [];
+
+ // Supprimer dans l'ordre inverse des dépendances
+ $tables = array_reverse(self::MIGRATION_STEPS);
+
+ foreach ($tables as $table) {
+ // Logique de suppression selon la table
+ }
+
+ return ['deleted_records' => $deletedRecords];
+ }
+
+ /**
+ * Annule une étape spécifique de migration
+ */
+ public function rollbackStep(int $entityId, string $step): array {
+ // TODO: Implémenter le rollback d'une étape
+ return ['deleted_records' => []];
+ }
+}
diff --git a/api/src/Services/OperationDataService.php b/api/src/Services/OperationDataService.php
index 8b3e3d15..c5a0896a 100755
--- a/api/src/Services/OperationDataService.php
+++ b/api/src/Services/OperationDataService.php
@@ -2,12 +2,12 @@
declare(strict_types=1);
+namespace App\Services;
+
require_once __DIR__ . '/ApiService.php';
require_once __DIR__ . '/LogService.php';
use PDO;
-use ApiService;
-use LogService;
class OperationDataService {
@@ -221,13 +221,13 @@ class OperationDataService {
if (!empty($sectorIdsString)) {
// Utiliser ope_users au lieu de users pour avoir les données historiques
$usersSectorsStmt = $db->prepare(
- "SELECT DISTINCT ou.fk_user as id, ou.first_name, ou.encrypted_name, ou.sect_name, us.fk_sector
+ "SELECT DISTINCT ou.fk_user as user_id, ou.id as ope_user_id, ou.first_name, ou.encrypted_name, ou.sect_name, us.fk_sector
FROM ope_users ou
- JOIN ope_users_sectors us ON ou.fk_user = us.fk_user AND ou.fk_operation = us.fk_operation
- WHERE us.fk_sector IN ($sectorIdsString)
- AND us.fk_operation = ?
- AND us.chk_active = 1
- AND ou.chk_active = 1
+ JOIN ope_users_sectors us ON ou.id = us.fk_user AND ou.fk_operation = us.fk_operation
+ WHERE us.fk_sector IN ($sectorIdsString)
+ AND us.fk_operation = ?
+ AND us.chk_active = 1
+ AND ou.chk_active = 1
AND ou.fk_user != ?" // Exclure l'utilisateur connecté
);
$usersSectorsStmt->execute([$activeOperationId, $userId]);
diff --git a/api/src/Services/PasswordSecurityService.php b/api/src/Services/PasswordSecurityService.php
index 1baf8038..a4639f49 100644
--- a/api/src/Services/PasswordSecurityService.php
+++ b/api/src/Services/PasswordSecurityService.php
@@ -4,7 +4,7 @@ declare(strict_types=1);
namespace App\Services;
-use LogService;
+use App\Services\LogService;
require_once __DIR__ . '/LogService.php';
diff --git a/api/src/Services/ReceiptPDFGenerator.php b/api/src/Services/ReceiptPDFGenerator.php
index 926b6ffd..cb587339 100644
--- a/api/src/Services/ReceiptPDFGenerator.php
+++ b/api/src/Services/ReceiptPDFGenerator.php
@@ -6,147 +6,99 @@ namespace App\Services;
require_once __DIR__ . '/../../vendor/autoload.php';
-use FPDF;
+use setasign\Fpdi\Fpdi;
/**
- * Générateur de reçus PDF avec FPDF
- * Supporte les logos PNG/JPG
+ * Générateur de reçus PDF avec FPDI (utilise un template PDF)
+ * Génère des PDF légers et valides en format paysage (A4 Landscape)
*/
-class ReceiptPDFGenerator extends FPDF {
-
+class ReceiptPDFGenerator extends Fpdi {
+
+ private const TEMPLATE_PATH = __DIR__ . '/../../docs/_recu_template.pdf';
private const DEFAULT_LOGO_PATH = __DIR__ . '/../../docs/_logo_recu.png';
- private const LOGO_WIDTH = 40; // Largeur du logo en mm
- private const LOGO_HEIGHT = 40; // Hauteur du logo en mm
-
+ private const LOGO_WIDTH = 45; // Largeur du logo en mm
+
/**
- * Génère un reçu fiscal PDF
+ * Génère un reçu fiscal PDF et l'enregistre directement dans un fichier
+ *
+ * @param array $data Données du reçu
+ * @param string $outputPath Chemin complet du fichier PDF à créer
+ * @param string|null $logoPath Chemin vers le logo (optionnel)
+ * @return bool True si la génération a réussi
*/
- public function generateReceipt(array $data, ?string $logoPath = null): string {
- $this->AddPage();
- $this->SetFont('Arial', '', 12);
-
- // Déterminer quel logo utiliser
- $logoToUse = null;
- if ($logoPath && file_exists($logoPath)) {
- $logoToUse = $logoPath;
- } elseif (file_exists(self::DEFAULT_LOGO_PATH)) {
- $logoToUse = self::DEFAULT_LOGO_PATH;
- }
-
- // Ajouter le logo (PNG ou JPG)
- if ($logoToUse) {
- try {
- // Déterminer le type d'image
- $imageInfo = getimagesize($logoToUse);
- if ($imageInfo !== false) {
- $type = '';
- switch ($imageInfo[2]) {
- case IMAGETYPE_JPEG:
- $type = 'JPG';
- break;
- case IMAGETYPE_PNG:
- $type = 'PNG';
- break;
- }
-
- if ($type) {
- // Position du logo : x=10, y=10, largeur=40mm, hauteur=40mm
- $this->Image($logoToUse, 10, 10, self::LOGO_WIDTH, self::LOGO_HEIGHT, $type);
+ public function generateReceipt(array $data, string $outputPath, ?string $logoPath = null): bool {
+ try {
+ // Créer la page en orientation paysage (Landscape)
+ $this->AddPage('L');
+
+ // Importer le template PDF
+ if (file_exists(self::TEMPLATE_PATH)) {
+ $this->setSourceFile(self::TEMPLATE_PATH);
+ $tplIdx = $this->importPage(1);
+ $this->useTemplate($tplIdx);
+ }
+
+ // Configuration de base
+ $this->SetFont('Arial');
+ $this->SetFontSize(16);
+ $this->SetTextColor(50, 50, 50);
+
+ // Nom de l'amicale (en haut à droite du template)
+ $this->SetXY(116, 26);
+ $this->Write(0, $this->cleanText($data['entite_city'] ?? ''));
+
+ // Nom du donateur
+ $this->SetXY(35, 41);
+ $this->Write(0, $this->cleanText($data['donor_name'] ?? ''));
+
+ // Adresse du donateur
+ $this->SetXY(35, 55);
+ $this->Write(0, $this->cleanText($data['donor_address'] ?? ''));
+
+ // Montant et mode de règlement
+ $this->SetXY(48, 68);
+ $amount = $data['amount'] ?? '0';
+ $paymentMethod = !empty($data['payment_method'])
+ ? ' en ' . mb_strtolower($data['payment_method'], 'UTF-8')
+ : '';
+ $this->Write(0, $this->cleanText($amount . ' euros' . $paymentMethod));
+
+ // Date du don
+ $this->SetXY(20, 82);
+ $this->Write(0, $this->cleanText($data['donation_date'] ?? date('d/m/Y')));
+
+ // Logo de l'entité (en haut à droite)
+ $logoToUse = null;
+ if ($logoPath && file_exists($logoPath)) {
+ $logoToUse = $logoPath;
+ } elseif (file_exists(self::DEFAULT_LOGO_PATH)) {
+ $logoToUse = self::DEFAULT_LOGO_PATH;
+ }
+
+ if ($logoToUse) {
+ try {
+ $this->Image($logoToUse, 245, 8, self::LOGO_WIDTH);
+ } catch (\Exception $e) {
+ // Si erreur avec le logo custom, utiliser le logo par défaut
+ if ($logoToUse !== self::DEFAULT_LOGO_PATH && file_exists(self::DEFAULT_LOGO_PATH)) {
+ try {
+ $this->Image(self::DEFAULT_LOGO_PATH, 245, 8, self::LOGO_WIDTH);
+ } catch (\Exception $e2) {
+ // Continuer sans logo
+ }
}
}
- } catch (\Exception $e) {
- // Si erreur avec le logo, continuer sans
}
+
+ // Écrire directement dans le fichier (mode 'F')
+ $this->Output($outputPath, 'F');
+
+ return true;
+
+ } catch (\Exception $e) {
+ error_log('Erreur génération PDF: ' . $e->getMessage());
+ return false;
}
-
- // En-tête à droite du logo
- $this->SetXY(60, 20);
- $this->SetFont('Arial', 'B', 14);
- $this->Cell(130, 6, $this->cleanText(strtoupper($data['entite_name'] ?? '')), 0, 1, 'C');
-
- if (!empty($data['entite_city'])) {
- $this->SetX(60);
- $this->SetFont('Arial', '', 11);
- $this->Cell(130, 5, $this->cleanText(strtoupper($data['entite_city'])), 0, 1, 'C');
- }
-
- if (!empty($data['entite_address'])) {
- $this->SetX(60);
- $this->SetFont('Arial', '', 10);
- $this->Cell(130, 5, $this->cleanText($data['entite_address']), 0, 1, 'C');
- }
-
- // Titre du reçu
- $this->SetY(65);
- $this->SetFont('Arial', 'B', 16);
- $this->Cell(0, 10, $this->cleanText('REÇU DE DON'), 0, 1, 'C');
-
- $this->SetFont('Arial', 'B', 14);
- $this->Cell(0, 8, $this->cleanText('N° ' . ($data['receipt_number'] ?? '')), 0, 1, 'C');
-
- // Ligne de séparation
- $this->Ln(5);
- $this->Line(20, $this->GetY(), 190, $this->GetY());
- $this->Ln(8);
-
- // Informations du donateur
- $this->SetFont('Arial', 'B', 12);
- $this->Cell(0, 6, $this->cleanText('DONATEUR :'), 0, 1, 'L');
-
- $this->SetFont('Arial', '', 11);
- $this->Cell(0, 6, $this->cleanText($data['donor_name'] ?? ''), 0, 1, 'L');
-
- if (!empty($data['donor_address'])) {
- $this->SetFont('Arial', '', 10);
- $this->Cell(0, 5, $this->cleanText($data['donor_address']), 0, 1, 'L');
- }
-
- $this->Ln(8);
-
- // Cadre pour le montant
- $this->SetFillColor(240, 240, 240);
- $this->Rect(20, $this->GetY(), 170, 25, 'F');
-
- // Montant en gros et centré
- $this->Ln(5);
- $this->SetFont('Arial', 'B', 18);
- $this->Cell(0, 8, $this->cleanText($data['amount'] ?? '0') . $this->cleanText(' euros'), 0, 1, 'C');
-
- // Date centrée
- $this->SetFont('Arial', '', 12);
- $this->Cell(0, 6, $this->cleanText('Don du ' . ($data['donation_date'] ?? date('d/m/Y'))), 0, 1, 'C');
-
- $this->Ln(10);
-
- if (!empty($data['payment_method'])) {
- $this->SetFont('Arial', '', 10);
- $this->Cell(0, 5, $this->cleanText('Mode de règlement : ') . $this->cleanText($data['payment_method']), 0, 1, 'L');
- }
-
- if (!empty($data['operation_name'])) {
- $this->SetFont('Arial', 'I', 10);
- $this->Cell(0, 5, $this->cleanText('Campagne : ') . $this->cleanText($data['operation_name']), 0, 1, 'L');
- }
-
- // Mention de remerciement
- $this->Ln(15);
- $this->SetFont('Arial', '', 10);
- $this->MultiCell(0, 5, $this->cleanText(
- "Nous vous remercions pour votre généreux soutien à notre amicale.\n" .
- "Votre don contribue au financement de nos activités et équipements."
- ), 0, 'C');
-
- // Signature
- $this->SetY(-60);
- $this->SetFont('Arial', '', 10);
- $this->Cell(0, 5, $this->cleanText('Fait à ' . ($data['entite_city'] ?? '') . ', le ' . ($data['signature_date'] ?? date('d/m/Y'))), 0, 1, 'R');
- $this->Ln(5);
- $this->Cell(0, 5, $this->cleanText('Le Président'), 0, 1, 'R');
- $this->Ln(15);
- $this->Cell(0, 5, $this->cleanText('(Signature et cachet)'), 0, 1, 'R');
-
- // Retourner le PDF en string
- return $this->Output('S');
}
/**
diff --git a/api/src/Services/ReceiptService.php b/api/src/Services/ReceiptService.php
index 2bf4bdfe..4d370f5f 100644
--- a/api/src/Services/ReceiptService.php
+++ b/api/src/Services/ReceiptService.php
@@ -8,12 +8,13 @@ require_once __DIR__ . '/LogService.php';
require_once __DIR__ . '/ApiService.php';
require_once __DIR__ . '/FileService.php';
require_once __DIR__ . '/ReceiptPDFGenerator.php';
+require_once __DIR__ . '/EmailTemplates.php';
use PDO;
use Database;
-use LogService;
-use ApiService;
-use FileService;
+use App\Services\LogService;
+use App\Services\FileService;
+use App\Services\ApiService;
use Exception;
use DateTime;
@@ -88,26 +89,28 @@ class ReceiptService {
// Préparer les données pour la génération du PDF
$receiptData = $this->prepareReceiptData($passageData, $operationData, $entiteData, $email);
-
- // Générer le PDF optimisé
- $pdfContent = $this->generateOptimizedPDF($receiptData, $logoPath);
-
+
// Créer le répertoire de stockage
$uploadPath = "/{$operationData['fk_entite']}/recus/{$operationData['id']}";
$fullPath = $this->fileService->createDirectory($operationData['fk_entite'], $uploadPath);
-
+
// Nom du fichier
$fileName = 'recu_' . $passageId . '.pdf';
$filePath = $fullPath . '/' . $fileName;
-
- // Sauvegarder le fichier
- if (file_put_contents($filePath, $pdfContent) === false) {
- throw new Exception('Impossible de sauvegarder le fichier PDF');
+
+ // Générer le PDF directement dans le fichier
+ $pdfGenerated = $this->generateOptimizedPDF($receiptData, $filePath, $logoPath);
+
+ if (!$pdfGenerated || !file_exists($filePath)) {
+ throw new Exception('Impossible de générer le fichier PDF');
}
-
+
// Appliquer les permissions
$this->fileService->setFilePermissions($filePath);
-
+
+ // Récupérer la taille du fichier généré
+ $fileSize = filesize($filePath);
+
// Enregistrer dans la table medias
$mediaId = $this->saveToMedias(
$operationData['fk_entite'],
@@ -115,21 +118,21 @@ class ReceiptService {
$passageId,
$fileName,
$filePath,
- strlen($pdfContent)
+ $fileSize
);
-
+
// Mettre à jour le passage avec les infos du reçu
$this->updatePassageReceipt($passageId, $fileName);
-
- // Ajouter à la queue d'email
- $this->queueReceiptEmail($passageId, $email, $receiptData, $pdfContent);
+
+ // Ajouter à la queue d'email (le PDF sera lu depuis le fichier)
+ $this->queueReceiptEmail($passageId, $email, $receiptData, $filePath);
LogService::log('Reçu généré avec succès', [
'level' => 'info',
'passageId' => $passageId,
'mediaId' => $mediaId,
'fileName' => $fileName,
- 'fileSize' => strlen($pdfContent)
+ 'fileSize' => $fileSize
]);
return true;
@@ -146,10 +149,15 @@ class ReceiptService {
/**
* Génère un PDF optimisé avec logo et mise en page épurée
+ *
+ * @param array $data Données du reçu
+ * @param string $outputPath Chemin du fichier PDF à créer
+ * @param string|null $logoPath Chemin du logo
+ * @return bool True si la génération a réussi
*/
- private function generateOptimizedPDF(array $data, ?string $logoPath): string {
+ private function generateOptimizedPDF(array $data, string $outputPath, ?string $logoPath): bool {
$pdf = new ReceiptPDFGenerator();
- return $pdf->generateReceipt($data, $logoPath);
+ return $pdf->generateReceipt($data, $outputPath, $logoPath);
}
@@ -158,12 +166,13 @@ class ReceiptService {
*/
private function getPassageData(int $passageId): ?array {
$stmt = $this->db->prepare('
- SELECT p.*,
+ SELECT p.*,
u.encrypted_name as user_encrypted_name,
u.encrypted_email as user_encrypted_email,
u.encrypted_phone as user_encrypted_phone
FROM ope_pass p
- LEFT JOIN users u ON p.fk_user = u.id
+ LEFT JOIN ope_users ou ON p.fk_user = ou.id
+ LEFT JOIN users u ON ou.fk_user = u.id
WHERE p.id = ? AND p.chk_active = 1
');
$stmt->execute([$passageId]);
@@ -345,25 +354,52 @@ class ReceiptService {
/**
* Ajoute le reçu à la queue d'email
+ *
+ * @param int $passageId ID du passage
+ * @param string $email Email du destinataire
+ * @param array $receiptData Données du reçu
+ * @param string $pdfFilePath Chemin vers le fichier PDF
*/
- private function queueReceiptEmail(int $passageId, string $email, array $receiptData, string $pdfContent): void {
+ private function queueReceiptEmail(int $passageId, string $email, array $receiptData, string $pdfFilePath): void {
+ // Lire le contenu du PDF depuis le fichier
+ if (!file_exists($pdfFilePath)) {
+ throw new \Exception('Fichier PDF introuvable pour la mise en queue: ' . $pdfFilePath);
+ }
+
+ $pdfContent = file_get_contents($pdfFilePath);
+ if ($pdfContent === false) {
+ throw new \Exception('Impossible de lire le fichier PDF: ' . $pdfFilePath);
+ }
+
// Préparer le sujet
- $subject = "Votre reçu de don N°" . $receiptData['receipt_number'];
-
- // Préparer le corps de l'email
- $body = $this->generateEmailBody($receiptData);
-
+ $subject = "Votre reçu de don - " . $receiptData['entite_name'];
+
+ // Préparer les données pour le template
+ $templateData = [
+ 'passage_id' => $passageId,
+ 'entite_name' => $receiptData['entite_name'],
+ 'donor_name' => $receiptData['donor_name'],
+ 'amount' => $receiptData['amount'],
+ 'donation_date' => $receiptData['donation_date'],
+ 'payment_method' => $receiptData['payment_method'],
+ 'entite_address' => $receiptData['entite_address'],
+ 'entite_email' => $receiptData['entite_email']
+ ];
+
+ // Générer le corps de l'email via le template centralisé
+ $body = EmailTemplates::getReceiptDonationTemplate($templateData);
+
// Préparer les headers avec pièce jointe
$boundary = md5((string)time());
$headers = "MIME-Version: 1.0\r\n";
$headers .= "Content-Type: multipart/mixed; boundary=\"$boundary\"\r\n";
-
+
// Corps complet avec pièce jointe
$fullBody = "--$boundary\r\n";
$fullBody .= "Content-Type: text/html; charset=UTF-8\r\n";
$fullBody .= "Content-Transfer-Encoding: 7bit\r\n\r\n";
$fullBody .= $body . "\r\n\r\n";
-
+
// Pièce jointe PDF
$fullBody .= "--$boundary\r\n";
$fullBody .= "Content-Type: application/pdf; name=\"recu_" . $receiptData['receipt_number'] . ".pdf\"\r\n";
@@ -371,14 +407,14 @@ class ReceiptService {
$fullBody .= "Content-Disposition: attachment; filename=\"recu_" . $receiptData['receipt_number'] . ".pdf\"\r\n\r\n";
$fullBody .= chunk_split(base64_encode($pdfContent)) . "\r\n";
$fullBody .= "--$boundary--";
-
+
// Insérer dans la queue
$stmt = $this->db->prepare('
INSERT INTO email_queue (
fk_pass, to_email, subject, body, headers, created_at, status
) VALUES (?, ?, ?, ?, ?, NOW(), ?)
');
-
+
$stmt->execute([
$passageId,
$email,
@@ -388,62 +424,7 @@ class ReceiptService {
'pending'
]);
}
-
- /**
- * Génère le corps HTML de l'email
- */
- private function generateEmailBody(array $data): string {
- // Convertir toutes les valeurs en string pour htmlspecialchars
- $safeData = array_map(function($value) {
- return is_string($value) ? $value : (string)$value;
- }, $data);
-
- $html = '
-
-
-
-
-
-
-
-
-
-
Bonjour ' . htmlspecialchars($safeData['donor_name']) . ',
-
-
Nous vous remercions chaleureusement pour votre don de ' .
- htmlspecialchars($safeData['amount']) . ' € effectué le ' .
- htmlspecialchars($safeData['donation_date']) . '.
-
-
Vous trouverez ci-joint votre reçu fiscal N°' . htmlspecialchars($safeData['receipt_number']) .
- ' qui vous permettra de bénéficier d\'une réduction d\'impôt égale à 66% du montant de votre don.
-
-
Votre soutien est précieux pour nous permettre de poursuivre nos actions.
-
-
Cordialement,
- L\'équipe de ' . htmlspecialchars($safeData['entite_name']) . '
-
-
-
-
-';
-
- return $html;
- }
-
+
/**
* Met à jour la date d'envoi du reçu
*/
diff --git a/api/src/Services/Security/AlertService.php b/api/src/Services/Security/AlertService.php
index 3f1fe91a..6014b59b 100644
--- a/api/src/Services/Security/AlertService.php
+++ b/api/src/Services/Security/AlertService.php
@@ -9,8 +9,9 @@ require_once __DIR__ . '/EmailThrottler.php';
use PDO;
use Database;
-use ApiService;
+use App\Services\ApiService;
use AppConfig;
+use App\Services\LogService;
/**
* Service central de gestion des alertes de sécurité et monitoring
@@ -94,7 +95,7 @@ class AlertService {
$context['request'] = [
'uri' => $_SERVER['REQUEST_URI'],
'method' => $_SERVER['REQUEST_METHOD'] ?? 'unknown',
- 'ip' => $_SERVER['REMOTE_ADDR'] ?? 'unknown',
+ 'ip' => \AppConfig::getInstance()->getClientIp(),
'user_agent' => $_SERVER['HTTP_USER_AGENT'] ?? 'unknown'
];
}
diff --git a/api/src/Services/Security/PerformanceMonitor.php b/api/src/Services/Security/PerformanceMonitor.php
index ab7bd1af..3dd337ae 100644
--- a/api/src/Services/Security/PerformanceMonitor.php
+++ b/api/src/Services/Security/PerformanceMonitor.php
@@ -111,7 +111,7 @@ class PerformanceMonitor {
$memoryUsed = $memoryPeak - $memoryStart;
// Enrichir avec les infos de requête
- $ip = $_SERVER['REMOTE_ADDR'] ?? null;
+ $ip = \AppConfig::getInstance()->getClientIp();
$userAgent = $_SERVER['HTTP_USER_AGENT'] ?? null;
$requestSize = strlen(file_get_contents('php://input'));
diff --git a/api/src/Services/Security/SecurityMonitor.php b/api/src/Services/Security/SecurityMonitor.php
index a59dd2e3..7a7ead5b 100644
--- a/api/src/Services/Security/SecurityMonitor.php
+++ b/api/src/Services/Security/SecurityMonitor.php
@@ -74,13 +74,10 @@ class SecurityMonitor {
// Critères de détection
$isBruteForce = false;
$reason = '';
-
- if ($attempts >= 5) {
+
+ if ($attempts >= 8) {
$isBruteForce = true;
- $reason = "Plus de 5 tentatives en 5 minutes";
- } elseif ($uniqueUsers >= 3) {
- $isBruteForce = true;
- $reason = "Tentatives sur 3 usernames différents";
+ $reason = "Plus de 8 tentatives en 5 minutes";
}
if ($isBruteForce) {
@@ -114,15 +111,19 @@ class SecurityMonitor {
try {
// Chercher si le username existe (pour stocker la version chiffrée)
+ require_once __DIR__ . '/../ApiService.php';
$encryptedUsername = null;
if ($username) {
+ // Chiffrer le username pour la recherche
+ $searchUsername = \ApiService::encryptSearchableData($username);
+
$userStmt = $db->prepare('
- SELECT encrypted_user_name
- FROM users
- WHERE username = :username
+ SELECT encrypted_user_name
+ FROM users
+ WHERE encrypted_user_name = :encrypted_username
LIMIT 1
');
- $userStmt->execute(['username' => $username]);
+ $userStmt->execute(['encrypted_username' => $searchUsername]);
$user = $userStmt->fetch(PDO::FETCH_ASSOC);
if ($user) {
$encryptedUsername = $user['encrypted_user_name'];
@@ -178,7 +179,7 @@ class SecurityMonitor {
if (isset($_SERVER['REQUEST_URI'])) {
$context['endpoint'] = $_SERVER['REQUEST_URI'];
$context['method'] = $_SERVER['REQUEST_METHOD'] ?? 'unknown';
- $context['ip'] = $_SERVER['REMOTE_ADDR'] ?? 'unknown';
+ $context['ip'] = \AppConfig::getInstance()->getClientIp();
}
AlertService::trigger('SQL_INJECTION', $context, 'SECURITY');
diff --git a/api/src/Services/StripeService.php b/api/src/Services/StripeService.php
index 6ba18121..79f3ce2e 100644
--- a/api/src/Services/StripeService.php
+++ b/api/src/Services/StripeService.php
@@ -12,6 +12,8 @@ use AppConfig;
use Database;
use PDO;
use Exception;
+use App\Services\LogService;
+use App\Services\ApiService;
/**
* Service principal pour gérer l'intégration Stripe
@@ -85,10 +87,67 @@ class StripeService {
// Si le compte existe, vérifier s'il est complet
try {
$stripeAccount = $this->stripe->accounts->retrieve($existingAccount['stripe_account_id']);
-
+
+ // Si pas de location_id, créer la Location maintenant
+ $locationId = $existingAccount['stripe_location_id'];
+ if (empty($locationId)) {
+ try {
+ // Déchiffrer le nom pour la Location
+ $nom = !empty($entite['encrypted_name']) ? ApiService::decryptData($entite['encrypted_name']) : 'Amicale';
+
+ // Construire l'adresse
+ $adresse1 = !empty($entite['adresse1']) ? $entite['adresse1'] : 'Adresse non renseignée';
+ $adresse2 = !empty($entite['adresse2']) ? $entite['adresse2'] : '';
+ $ville = !empty($entite['ville']) ? $entite['ville'] : 'Ville';
+ $codePostal = !empty($entite['code_postal']) ? $entite['code_postal'] : '00000';
+
+ $location = $this->stripe->terminal->locations->create([
+ 'display_name' => $nom,
+ 'address' => [
+ 'line1' => $adresse1,
+ 'line2' => $adresse2,
+ 'city' => $ville,
+ 'postal_code' => $codePostal,
+ 'country' => 'FR',
+ ],
+ 'metadata' => [
+ 'entite_id' => $entiteId,
+ 'type' => 'tap_to_pay'
+ ]
+ ], [
+ 'stripe_account' => $existingAccount['stripe_account_id']
+ ]);
+
+ $locationId = $location->id;
+
+ // Mettre à jour en base
+ $stmt = $this->db->prepare(
+ "UPDATE stripe_accounts
+ SET stripe_location_id = :location_id, updated_at = NOW()
+ WHERE fk_entite = :fk_entite"
+ );
+ $stmt->execute([
+ 'location_id' => $locationId,
+ 'fk_entite' => $entiteId
+ ]);
+
+ LogService::log('Location créée pour compte existant', [
+ 'entite_id' => $entiteId,
+ 'location_id' => $locationId
+ ]);
+ } catch (Exception $e) {
+ LogService::log('Erreur création Location pour compte existant', [
+ 'level' => 'warning',
+ 'entite_id' => $entiteId,
+ 'error' => $e->getMessage()
+ ]);
+ }
+ }
+
return [
'success' => true,
'account_id' => $existingAccount['stripe_account_id'],
+ 'location_id' => $locationId,
'message' => 'Compte Stripe existant',
'existing' => true,
'charges_enabled' => $stripeAccount->charges_enabled,
@@ -106,8 +165,8 @@ class StripeService {
}
// Déchiffrer les données
- $nom = !empty($entite['encrypted_name']) ? \ApiService::decryptData($entite['encrypted_name']) : '';
- $email = !empty($entite['encrypted_email']) ? \ApiService::decryptSearchableData($entite['encrypted_email']) : null;
+ $nom = !empty($entite['encrypted_name']) ? ApiService::decryptData($entite['encrypted_name']) : '';
+ $email = !empty($entite['encrypted_email']) ? ApiService::decryptSearchableData($entite['encrypted_email']) : null;
// Créer le compte Stripe Connect Express
$accountData = [
@@ -147,21 +206,64 @@ class StripeService {
}
$account = $this->stripe->accounts->create($accountData);
-
- // Sauvegarder en base de données
+
+ // Créer automatiquement la Location Terminal pour Tap to Pay
+ $location = null;
+ $locationId = null;
+ try {
+ // Construire l'adresse complète
+ $adresse1 = !empty($entite['adresse1']) ? $entite['adresse1'] : 'Adresse non renseignée';
+ $adresse2 = !empty($entite['adresse2']) ? $entite['adresse2'] : '';
+ $ville = !empty($entite['ville']) ? $entite['ville'] : 'Ville';
+ $codePostal = !empty($entite['code_postal']) ? $entite['code_postal'] : '00000';
+
+ $location = $this->stripe->terminal->locations->create([
+ 'display_name' => $nom,
+ 'address' => [
+ 'line1' => $adresse1,
+ 'line2' => $adresse2,
+ 'city' => $ville,
+ 'postal_code' => $codePostal,
+ 'country' => 'FR',
+ ],
+ 'metadata' => [
+ 'entite_id' => $entiteId,
+ 'type' => 'tap_to_pay'
+ ]
+ ], [
+ 'stripe_account' => $account->id
+ ]);
+
+ $locationId = $location->id;
+ LogService::log('Location Stripe créée automatiquement', [
+ 'entite_id' => $entiteId,
+ 'location_id' => $locationId
+ ]);
+ } catch (Exception $e) {
+ // Si la création de la Location échoue, logger mais continuer
+ LogService::log('Erreur création Location', [
+ 'level' => 'warning',
+ 'entite_id' => $entiteId,
+ 'error' => $e->getMessage()
+ ]);
+ }
+
+ // Sauvegarder en base de données avec le location_id
$stmt = $this->db->prepare(
- "INSERT INTO stripe_accounts (fk_entite, stripe_account_id, created_at)
- VALUES (:fk_entite, :stripe_account_id, NOW())"
+ "INSERT INTO stripe_accounts (fk_entite, stripe_account_id, stripe_location_id, created_at)
+ VALUES (:fk_entite, :stripe_account_id, :stripe_location_id, NOW())"
);
$stmt->execute([
'fk_entite' => $entiteId,
- 'stripe_account_id' => $account->id
+ 'stripe_account_id' => $account->id,
+ 'stripe_location_id' => $locationId
]);
-
+
return [
'success' => true,
'account_id' => $account->id,
- 'message' => 'Compte Stripe créé avec succès'
+ 'location_id' => $locationId,
+ 'message' => 'Compte Stripe créé avec succès' . ($locationId ? ' (Location Terminal créée)' : '')
];
} catch (ApiErrorException $e) {
@@ -197,41 +299,41 @@ class StripeService {
*/
public function createOnboardingLink(string $accountId, string $returnUrl, string $refreshUrl): array {
try {
- \LogService::log('StripeService::createOnboardingLink début', [
+ LogService::log('StripeService::createOnboardingLink début', [
'account_id' => $accountId,
'return_url' => $returnUrl,
'refresh_url' => $refreshUrl
]);
-
+
$accountLink = $this->stripe->accountLinks->create([
'account' => $accountId,
'refresh_url' => $refreshUrl,
'return_url' => $returnUrl,
'type' => 'account_onboarding',
]);
-
- \LogService::log('StripeService::createOnboardingLink succès', [
+
+ LogService::log('StripeService::createOnboardingLink succès', [
'url' => $accountLink->url
]);
-
+
return [
'success' => true,
'url' => $accountLink->url
];
-
+
} catch (ApiErrorException $e) {
- \LogService::log('StripeService::createOnboardingLink erreur Stripe', [
+ LogService::log('StripeService::createOnboardingLink erreur Stripe', [
'level' => 'error',
'error' => $e->getMessage(),
'code' => $e->getCode()
]);
-
+
return [
'success' => false,
'message' => 'Erreur Stripe: ' . $e->getMessage()
];
} catch (\Exception $e) {
- \LogService::log('StripeService::createOnboardingLink erreur générale', [
+ LogService::log('StripeService::createOnboardingLink erreur générale', [
'level' => 'error',
'error' => $e->getMessage()
]);
@@ -263,7 +365,7 @@ class StripeService {
}
// Déchiffrer les données de l'entité
- $nom = !empty($data['encrypted_name']) ? \ApiService::decryptData($data['encrypted_name']) : 'Amicale';
+ $nom = !empty($data['encrypted_name']) ? ApiService::decryptData($data['encrypted_name']) : 'Amicale';
// Construire l'adresse complète
$adresse1 = !empty($data['adresse1']) ? $data['adresse1'] : '';
@@ -500,20 +602,165 @@ class StripeService {
}
}
+ /**
+ * Créer un Payment Link Stripe pour paiement par QR Code
+ *
+ * @param array $params [
+ * 'amount' => int (en centimes),
+ * 'currency' => string (défaut: 'eur'),
+ * 'description' => string,
+ * 'passage_id' => int,
+ * 'metadata' => array
+ * ]
+ * @return array ['success' => bool, 'payment_link_id' => string, 'url' => string, 'amount' => int]
+ */
+ public function createPaymentLink(array $params): array {
+ try {
+ $amount = $params['amount'] ?? 0;
+ $passageId = $params['passage_id'] ?? 0;
+
+ if ($amount <= 0) {
+ throw new Exception("Le montant doit être supérieur à 0");
+ }
+
+ // Récupérer les infos du passage avec opération et entité
+ $stmt = $this->db->prepare("
+ SELECT p.*, o.fk_entite, o.id as operation_id, sa.stripe_account_id
+ FROM ope_pass p
+ JOIN operations o ON p.fk_operation = o.id
+ JOIN stripe_accounts sa ON o.fk_entite = sa.fk_entite
+ WHERE p.id = :passage_id
+ ");
+ $stmt->execute(['passage_id' => $passageId]);
+ $passage = $stmt->fetch(PDO::FETCH_ASSOC);
+
+ if (!$passage) {
+ throw new Exception("Passage non trouvé");
+ }
+
+ if (!$passage['stripe_account_id']) {
+ throw new Exception("Stripe non activé pour cette amicale");
+ }
+
+ // Préparer les metadata
+ $metadata = array_merge($params['metadata'] ?? [], [
+ 'passage_id' => (string)$passageId,
+ 'operation_id' => (string)$passage['operation_id'],
+ 'amicale_id' => (string)$passage['fk_entite'],
+ 'fk_user' => (string)$passage['fk_user'], // ID du membre (ope_users.id)
+ 'created_at' => (string)time(), // Timestamp Unix de création du Payment Link
+ 'type' => 'qr_code_payment'
+ ]);
+
+ // Créer le Payment Link sur le compte Connect
+ $paymentLink = $this->stripe->paymentLinks->create([
+ 'line_items' => [
+ [
+ 'price_data' => [
+ 'currency' => $params['currency'] ?? 'eur',
+ 'product_data' => [
+ 'name' => $params['description'] ?? 'Calendrier pompiers',
+ ],
+ 'unit_amount' => $amount,
+ ],
+ 'quantity' => 1,
+ ],
+ ],
+ 'metadata' => $metadata,
+ 'after_completion' => [
+ 'type' => 'hosted_confirmation',
+ 'hosted_confirmation' => [
+ 'custom_message' => 'Merci pour votre paiement ! Votre reçu vous sera envoyé par email.',
+ ],
+ ],
+ 'payment_method_types' => ['card'],
+ 'billing_address_collection' => 'auto',
+ ], [
+ 'stripe_account' => $passage['stripe_account_id']
+ ]);
+
+ // Logger la création
+ LogService::log('Payment Link créé', [
+ 'payment_link_id' => $paymentLink->id,
+ 'passage_id' => $passageId,
+ 'amount' => $amount,
+ 'amicale_id' => $passage['fk_entite']
+ ]);
+
+ // Mettre à jour le passage avec le payment_link_id
+ $stmt = $this->db->prepare("
+ UPDATE ope_pass
+ SET stripe_payment_link_id = :link_id, updated_at = NOW()
+ WHERE id = :passage_id
+ ");
+ $stmt->execute([
+ 'link_id' => $paymentLink->id,
+ 'passage_id' => $passageId
+ ]);
+
+ return [
+ 'success' => true,
+ 'payment_link_id' => $paymentLink->id,
+ 'url' => $paymentLink->url,
+ 'amount' => $amount
+ ];
+
+ } catch (ApiErrorException $e) {
+ LogService::log('Erreur Stripe Payment Link', [
+ 'level' => 'error',
+ 'error' => $e->getMessage(),
+ 'passage_id' => $passageId ?? null
+ ]);
+
+ return [
+ 'success' => false,
+ 'message' => 'Erreur Stripe: ' . $e->getMessage()
+ ];
+ } catch (Exception $e) {
+ return [
+ 'success' => false,
+ 'message' => 'Erreur: ' . $e->getMessage()
+ ];
+ }
+ }
+
+ /**
+ * Récupérer le statut d'un PaymentIntent depuis Stripe
+ */
+ public function getPaymentIntentStatus(string $paymentIntentId): array {
+ try {
+ $paymentIntent = $this->stripe->paymentIntents->retrieve($paymentIntentId);
+
+ return [
+ 'success' => true,
+ 'status' => $paymentIntent->status,
+ 'amount' => $paymentIntent->amount,
+ 'currency' => $paymentIntent->currency,
+ 'payment_method' => $paymentIntent->payment_method,
+ 'created' => $paymentIntent->created
+ ];
+ } catch (Exception $e) {
+ return [
+ 'success' => false,
+ 'message' => 'Erreur: ' . $e->getMessage()
+ ];
+ }
+ }
+
/**
* Obtenir le mode actuel (test ou live)
*/
public function isTestMode(): bool {
return $this->testMode;
}
-
+
/**
* Obtenir la clé publique pour le frontend
*/
public function getPublicKey(): string {
$stripeConfig = $this->config->getStripeConfig();
- return $this->testMode
- ? $stripeConfig['public_key_test']
+ return $this->testMode
+ ? $stripeConfig['public_key_test']
: $stripeConfig['public_key_live'];
}
}
\ No newline at end of file
diff --git a/api/vendor/composer/autoload_classmap.php b/api/vendor/composer/autoload_classmap.php
index ff8d2196..a39e68e2 100644
--- a/api/vendor/composer/autoload_classmap.php
+++ b/api/vendor/composer/autoload_classmap.php
@@ -13,14 +13,19 @@ return array(
'App\\Controllers\\ChatController' => $baseDir . '/src/Controllers/ChatController.php',
'App\\Controllers\\EntiteController' => $baseDir . '/src/Controllers/EntiteController.php',
'App\\Controllers\\FileController' => $baseDir . '/src/Controllers/FileController.php',
+ 'App\\Controllers\\HealthController' => $baseDir . '/src/Controllers/HealthController.php',
'App\\Controllers\\LoginController' => $baseDir . '/src/Controllers/LoginController.php',
+ 'App\\Controllers\\MigrationController' => $baseDir . '/src/Controllers/MigrationController.php',
'App\\Controllers\\OperationController' => $baseDir . '/src/Controllers/OperationController.php',
'App\\Controllers\\PassageController' => $baseDir . '/src/Controllers/PassageController.php',
'App\\Controllers\\PasswordController' => $baseDir . '/src/Controllers/PasswordController.php',
'App\\Controllers\\SectorController' => $baseDir . '/src/Controllers/SectorController.php',
'App\\Controllers\\SecurityController' => $baseDir . '/src/Controllers/SecurityController.php',
+ 'App\\Controllers\\StripeController' => $baseDir . '/src/Controllers/StripeController.php',
+ 'App\\Controllers\\StripeWebhookController' => $baseDir . '/src/Controllers/StripeWebhookController.php',
'App\\Controllers\\UserController' => $baseDir . '/src/Controllers/UserController.php',
'App\\Controllers\\VilleController' => $baseDir . '/src/Controllers/VilleController.php',
+ 'App\\Core\\Controller' => $baseDir . '/src/Core/Controller.php',
'App\\Services\\PDFGenerator' => $baseDir . '/src/Services/PDFGenerator.php',
'App\\Services\\PasswordSecurityService' => $baseDir . '/src/Services/PasswordSecurityService.php',
'App\\Services\\ReceiptPDFGenerator' => $baseDir . '/src/Services/ReceiptPDFGenerator.php',
@@ -31,6 +36,7 @@ return array(
'App\\Services\\Security\\PerformanceMonitor' => $baseDir . '/src/Services/Security/PerformanceMonitor.php',
'App\\Services\\Security\\SecurityMonitor' => $baseDir . '/src/Services/Security/SecurityMonitor.php',
'App\\Services\\SimplePDF' => $baseDir . '/src/Services/SimplePDF.php',
+ 'App\\Services\\StripeService' => $baseDir . '/src/Services/StripeService.php',
'BackupEncryptionService' => $baseDir . '/src/Services/BackupEncryptionService.php',
'ClientDetector' => $baseDir . '/src/Utils/ClientDetector.php',
'Complex\\Complex' => $vendorDir . '/markbaker/complex/classes/src/Complex.php',
@@ -58,6 +64,7 @@ return array(
'Database' => $baseDir . '/src/Core/Database.php',
'DepartmentBoundaryService' => $baseDir . '/src/Services/DepartmentBoundaryService.php',
'EmailTemplates' => $baseDir . '/src/Services/EmailTemplates.php',
+ 'EventLogService' => $baseDir . '/src/Services/EventLogService.php',
'ExportService' => $baseDir . '/src/Services/ExportService.php',
'FPDF' => $vendorDir . '/setasign/fpdf/fpdf.php',
'FileService' => $baseDir . '/src/Services/FileService.php',
@@ -78,6 +85,7 @@ return array(
'Matrix\\Operators\\Multiplication' => $vendorDir . '/markbaker/matrix/classes/src/Operators/Multiplication.php',
'Matrix\\Operators\\Operator' => $vendorDir . '/markbaker/matrix/classes/src/Operators/Operator.php',
'Matrix\\Operators\\Subtraction' => $vendorDir . '/markbaker/matrix/classes/src/Operators/Subtraction.php',
+ 'MigrationService' => $baseDir . '/src/Services/MigrationService.php',
'MonitoredDatabase' => $baseDir . '/src/Core/MonitoredDatabase.php',
'MonitoredStatement' => $baseDir . '/src/Core/MonitoredDatabase.php',
'OperationDataService' => $baseDir . '/src/Services/OperationDataService.php',
@@ -91,6 +99,8 @@ return array(
'PhpOffice\\PhpSpreadsheet\\Calculation\\ArrayEnabled' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/ArrayEnabled.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\BinaryComparison' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/BinaryComparison.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\Calculation' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Calculation.php',
+ 'PhpOffice\\PhpSpreadsheet\\Calculation\\CalculationBase' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/CalculationBase.php',
+ 'PhpOffice\\PhpSpreadsheet\\Calculation\\CalculationLocale' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/CalculationLocale.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\Category' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Category.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\Database\\DAverage' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DAverage.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\Database\\DCount' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DCount.php',
@@ -176,21 +186,25 @@ return array(
'PhpOffice\\PhpSpreadsheet\\Calculation\\Financial\\TreasuryBill' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/TreasuryBill.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\FormulaParser' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/FormulaParser.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\FormulaToken' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/FormulaToken.php',
+ 'PhpOffice\\PhpSpreadsheet\\Calculation\\FunctionArray' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/FunctionArray.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\Functions' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Functions.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\Information\\ErrorValue' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Information/ErrorValue.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\Information\\ExcelError' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Information/ExcelError.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\Information\\Value' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Information/Value.php',
+ 'PhpOffice\\PhpSpreadsheet\\Calculation\\Internal\\ExcelArrayPseudoFunctions' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Internal/ExcelArrayPseudoFunctions.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\Internal\\MakeMatrix' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Internal/MakeMatrix.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\Internal\\WildcardMatch' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Internal/WildcardMatch.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\Logical\\Boolean' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Logical/Boolean.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\Logical\\Conditional' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Logical/Conditional.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\Logical\\Operations' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Logical/Operations.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\Address' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Address.php',
+ 'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\ChooseRowsEtc' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/ChooseRowsEtc.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\ExcelMatch' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/ExcelMatch.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\Filter' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Filter.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\Formula' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Formula.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\HLookup' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/HLookup.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\Helpers' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Helpers.php',
+ 'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\Hstack' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Hstack.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\Hyperlink' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Hyperlink.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\Indirect' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Indirect.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\Lookup' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Lookup.php',
@@ -201,8 +215,10 @@ return array(
'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\RowColumnInformation' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/RowColumnInformation.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\Selection' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Selection.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\Sort' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Sort.php',
+ 'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\TorowTocol' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/TorowTocol.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\Unique' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Unique.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\VLookup' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/VLookup.php',
+ 'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\Vstack' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Vstack.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\MathTrig\\Absolute' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/MathTrig/Absolute.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\MathTrig\\Angle' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/MathTrig/Angle.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\MathTrig\\Arabic' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/MathTrig/Arabic.php',
@@ -362,6 +378,9 @@ return array(
'PhpOffice\\PhpSpreadsheet\\Reader\\Security\\XmlScanner' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Security/XmlScanner.php',
'PhpOffice\\PhpSpreadsheet\\Reader\\Slk' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Slk.php',
'PhpOffice\\PhpSpreadsheet\\Reader\\Xls' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls.php',
+ 'PhpOffice\\PhpSpreadsheet\\Reader\\XlsBase' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/XlsBase.php',
+ 'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\Biff5' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/Biff5.php',
+ 'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\Biff8' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/Biff8.php',
'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\Color' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/Color.php',
'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\Color\\BIFF5' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/Color/BIFF5.php',
'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\Color\\BIFF8' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/Color/BIFF8.php',
@@ -370,7 +389,10 @@ return array(
'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\DataValidationHelper' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/DataValidationHelper.php',
'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\ErrorCode' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/ErrorCode.php',
'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\Escher' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/Escher.php',
+ 'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\ListFunctions' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/ListFunctions.php',
+ 'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\LoadSpreadsheet' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/LoadSpreadsheet.php',
'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\MD5' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/MD5.php',
+ 'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\Mappings' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/Mappings.php',
'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\RC4' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/RC4.php',
'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\Style\\Border' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/Style/Border.php',
'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\Style\\CellAlignment' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/Style/CellAlignment.php',
@@ -456,6 +478,8 @@ return array(
'PhpOffice\\PhpSpreadsheet\\Style\\ConditionalFormatting\\ConditionalDataBarExtension' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/ConditionalFormatting/ConditionalDataBarExtension.php',
'PhpOffice\\PhpSpreadsheet\\Style\\ConditionalFormatting\\ConditionalFormatValueObject' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/ConditionalFormatting/ConditionalFormatValueObject.php',
'PhpOffice\\PhpSpreadsheet\\Style\\ConditionalFormatting\\ConditionalFormattingRuleExtension' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/ConditionalFormatting/ConditionalFormattingRuleExtension.php',
+ 'PhpOffice\\PhpSpreadsheet\\Style\\ConditionalFormatting\\ConditionalIconSet' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/ConditionalFormatting/ConditionalIconSet.php',
+ 'PhpOffice\\PhpSpreadsheet\\Style\\ConditionalFormatting\\IconSetValues' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/ConditionalFormatting/IconSetValues.php',
'PhpOffice\\PhpSpreadsheet\\Style\\ConditionalFormatting\\StyleMerger' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/ConditionalFormatting/StyleMerger.php',
'PhpOffice\\PhpSpreadsheet\\Style\\ConditionalFormatting\\Wizard' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/ConditionalFormatting/Wizard.php',
'PhpOffice\\PhpSpreadsheet\\Style\\ConditionalFormatting\\Wizard\\Blanks' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/ConditionalFormatting/Wizard/Blanks.php',
@@ -478,6 +502,8 @@ return array(
'PhpOffice\\PhpSpreadsheet\\Style\\NumberFormat\\PercentageFormatter' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/NumberFormat/PercentageFormatter.php',
'PhpOffice\\PhpSpreadsheet\\Style\\NumberFormat\\Wizard\\Accounting' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/NumberFormat/Wizard/Accounting.php',
'PhpOffice\\PhpSpreadsheet\\Style\\NumberFormat\\Wizard\\Currency' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/NumberFormat/Wizard/Currency.php',
+ 'PhpOffice\\PhpSpreadsheet\\Style\\NumberFormat\\Wizard\\CurrencyBase' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/NumberFormat/Wizard/CurrencyBase.php',
+ 'PhpOffice\\PhpSpreadsheet\\Style\\NumberFormat\\Wizard\\CurrencyNegative' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/NumberFormat/Wizard/CurrencyNegative.php',
'PhpOffice\\PhpSpreadsheet\\Style\\NumberFormat\\Wizard\\Date' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/NumberFormat/Wizard/Date.php',
'PhpOffice\\PhpSpreadsheet\\Style\\NumberFormat\\Wizard\\DateTime' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/NumberFormat/Wizard/DateTime.php',
'PhpOffice\\PhpSpreadsheet\\Style\\NumberFormat\\Wizard\\DateTimeWizard' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/NumberFormat/Wizard/DateTimeWizard.php',
@@ -524,6 +550,7 @@ return array(
'PhpOffice\\PhpSpreadsheet\\Worksheet\\SheetView' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Worksheet/SheetView.php',
'PhpOffice\\PhpSpreadsheet\\Worksheet\\Table' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Worksheet/Table.php',
'PhpOffice\\PhpSpreadsheet\\Worksheet\\Table\\Column' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Worksheet/Table/Column.php',
+ 'PhpOffice\\PhpSpreadsheet\\Worksheet\\Table\\TableDxfsStyle' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Worksheet/Table/TableDxfsStyle.php',
'PhpOffice\\PhpSpreadsheet\\Worksheet\\Table\\TableStyle' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Worksheet/Table/TableStyle.php',
'PhpOffice\\PhpSpreadsheet\\Worksheet\\Validations' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Worksheet/Validations.php',
'PhpOffice\\PhpSpreadsheet\\Worksheet\\Worksheet' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Worksheet/Worksheet.php',
@@ -561,7 +588,6 @@ return array(
'PhpOffice\\PhpSpreadsheet\\Writer\\Xls\\Style\\CellAlignment' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Writer/Xls/Style/CellAlignment.php',
'PhpOffice\\PhpSpreadsheet\\Writer\\Xls\\Style\\CellBorder' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Writer/Xls/Style/CellBorder.php',
'PhpOffice\\PhpSpreadsheet\\Writer\\Xls\\Style\\CellFill' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Writer/Xls/Style/CellFill.php',
- 'PhpOffice\\PhpSpreadsheet\\Writer\\Xls\\Style\\ColorMap' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Writer/Xls/Style/ColorMap.php',
'PhpOffice\\PhpSpreadsheet\\Writer\\Xls\\Workbook' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Writer/Xls/Workbook.php',
'PhpOffice\\PhpSpreadsheet\\Writer\\Xls\\Worksheet' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Writer/Xls/Worksheet.php',
'PhpOffice\\PhpSpreadsheet\\Writer\\Xls\\Xf' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Writer/Xls/Xf.php',
@@ -574,6 +600,7 @@ return array(
'PhpOffice\\PhpSpreadsheet\\Writer\\Xlsx\\DocProps' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Writer/Xlsx/DocProps.php',
'PhpOffice\\PhpSpreadsheet\\Writer\\Xlsx\\Drawing' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Writer/Xlsx/Drawing.php',
'PhpOffice\\PhpSpreadsheet\\Writer\\Xlsx\\FunctionPrefix' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Writer/Xlsx/FunctionPrefix.php',
+ 'PhpOffice\\PhpSpreadsheet\\Writer\\Xlsx\\Metadata' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Writer/Xlsx/Metadata.php',
'PhpOffice\\PhpSpreadsheet\\Writer\\Xlsx\\Rels' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Writer/Xlsx/Rels.php',
'PhpOffice\\PhpSpreadsheet\\Writer\\Xlsx\\RelsRibbon' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Writer/Xlsx/RelsRibbon.php',
'PhpOffice\\PhpSpreadsheet\\Writer\\Xlsx\\RelsVBA' => $vendorDir . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Writer/Xlsx/RelsVBA.php',
@@ -1001,6 +1028,7 @@ return array(
'ZipStream\\LocalFileHeader' => $vendorDir . '/maennchen/zipstream-php/src/LocalFileHeader.php',
'ZipStream\\OperationMode' => $vendorDir . '/maennchen/zipstream-php/src/OperationMode.php',
'ZipStream\\PackField' => $vendorDir . '/maennchen/zipstream-php/src/PackField.php',
+ 'ZipStream\\Stream\\CallbackStreamWrapper' => $vendorDir . '/maennchen/zipstream-php/src/Stream/CallbackStreamWrapper.php',
'ZipStream\\Time' => $vendorDir . '/maennchen/zipstream-php/src/Time.php',
'ZipStream\\Version' => $vendorDir . '/maennchen/zipstream-php/src/Version.php',
'ZipStream\\Zip64\\DataDescriptor' => $vendorDir . '/maennchen/zipstream-php/src/Zip64/DataDescriptor.php',
@@ -1009,4 +1037,55 @@ return array(
'ZipStream\\Zip64\\ExtendedInformationExtraField' => $vendorDir . '/maennchen/zipstream-php/src/Zip64/ExtendedInformationExtraField.php',
'ZipStream\\ZipStream' => $vendorDir . '/maennchen/zipstream-php/src/ZipStream.php',
'ZipStream\\Zs\\ExtendedInformationExtraField' => $vendorDir . '/maennchen/zipstream-php/src/Zs/ExtendedInformationExtraField.php',
+ 'setasign\\Fpdi\\FpdfTpl' => $vendorDir . '/setasign/fpdi/src/FpdfTpl.php',
+ 'setasign\\Fpdi\\FpdfTplTrait' => $vendorDir . '/setasign/fpdi/src/FpdfTplTrait.php',
+ 'setasign\\Fpdi\\FpdfTrait' => $vendorDir . '/setasign/fpdi/src/FpdfTrait.php',
+ 'setasign\\Fpdi\\Fpdi' => $vendorDir . '/setasign/fpdi/src/Fpdi.php',
+ 'setasign\\Fpdi\\FpdiException' => $vendorDir . '/setasign/fpdi/src/FpdiException.php',
+ 'setasign\\Fpdi\\FpdiTrait' => $vendorDir . '/setasign/fpdi/src/FpdiTrait.php',
+ 'setasign\\Fpdi\\GraphicsState' => $vendorDir . '/setasign/fpdi/src/GraphicsState.php',
+ 'setasign\\Fpdi\\Math\\Matrix' => $vendorDir . '/setasign/fpdi/src/Math/Matrix.php',
+ 'setasign\\Fpdi\\Math\\Vector' => $vendorDir . '/setasign/fpdi/src/Math/Vector.php',
+ 'setasign\\Fpdi\\PdfParser\\CrossReference\\AbstractReader' => $vendorDir . '/setasign/fpdi/src/PdfParser/CrossReference/AbstractReader.php',
+ 'setasign\\Fpdi\\PdfParser\\CrossReference\\CrossReference' => $vendorDir . '/setasign/fpdi/src/PdfParser/CrossReference/CrossReference.php',
+ 'setasign\\Fpdi\\PdfParser\\CrossReference\\CrossReferenceException' => $vendorDir . '/setasign/fpdi/src/PdfParser/CrossReference/CrossReferenceException.php',
+ 'setasign\\Fpdi\\PdfParser\\CrossReference\\FixedReader' => $vendorDir . '/setasign/fpdi/src/PdfParser/CrossReference/FixedReader.php',
+ 'setasign\\Fpdi\\PdfParser\\CrossReference\\LineReader' => $vendorDir . '/setasign/fpdi/src/PdfParser/CrossReference/LineReader.php',
+ 'setasign\\Fpdi\\PdfParser\\CrossReference\\ReaderInterface' => $vendorDir . '/setasign/fpdi/src/PdfParser/CrossReference/ReaderInterface.php',
+ 'setasign\\Fpdi\\PdfParser\\Filter\\Ascii85' => $vendorDir . '/setasign/fpdi/src/PdfParser/Filter/Ascii85.php',
+ 'setasign\\Fpdi\\PdfParser\\Filter\\Ascii85Exception' => $vendorDir . '/setasign/fpdi/src/PdfParser/Filter/Ascii85Exception.php',
+ 'setasign\\Fpdi\\PdfParser\\Filter\\AsciiHex' => $vendorDir . '/setasign/fpdi/src/PdfParser/Filter/AsciiHex.php',
+ 'setasign\\Fpdi\\PdfParser\\Filter\\FilterException' => $vendorDir . '/setasign/fpdi/src/PdfParser/Filter/FilterException.php',
+ 'setasign\\Fpdi\\PdfParser\\Filter\\FilterInterface' => $vendorDir . '/setasign/fpdi/src/PdfParser/Filter/FilterInterface.php',
+ 'setasign\\Fpdi\\PdfParser\\Filter\\Flate' => $vendorDir . '/setasign/fpdi/src/PdfParser/Filter/Flate.php',
+ 'setasign\\Fpdi\\PdfParser\\Filter\\FlateException' => $vendorDir . '/setasign/fpdi/src/PdfParser/Filter/FlateException.php',
+ 'setasign\\Fpdi\\PdfParser\\Filter\\Lzw' => $vendorDir . '/setasign/fpdi/src/PdfParser/Filter/Lzw.php',
+ 'setasign\\Fpdi\\PdfParser\\Filter\\LzwException' => $vendorDir . '/setasign/fpdi/src/PdfParser/Filter/LzwException.php',
+ 'setasign\\Fpdi\\PdfParser\\PdfParser' => $vendorDir . '/setasign/fpdi/src/PdfParser/PdfParser.php',
+ 'setasign\\Fpdi\\PdfParser\\PdfParserException' => $vendorDir . '/setasign/fpdi/src/PdfParser/PdfParserException.php',
+ 'setasign\\Fpdi\\PdfParser\\StreamReader' => $vendorDir . '/setasign/fpdi/src/PdfParser/StreamReader.php',
+ 'setasign\\Fpdi\\PdfParser\\Tokenizer' => $vendorDir . '/setasign/fpdi/src/PdfParser/Tokenizer.php',
+ 'setasign\\Fpdi\\PdfParser\\Type\\PdfArray' => $vendorDir . '/setasign/fpdi/src/PdfParser/Type/PdfArray.php',
+ 'setasign\\Fpdi\\PdfParser\\Type\\PdfBoolean' => $vendorDir . '/setasign/fpdi/src/PdfParser/Type/PdfBoolean.php',
+ 'setasign\\Fpdi\\PdfParser\\Type\\PdfDictionary' => $vendorDir . '/setasign/fpdi/src/PdfParser/Type/PdfDictionary.php',
+ 'setasign\\Fpdi\\PdfParser\\Type\\PdfHexString' => $vendorDir . '/setasign/fpdi/src/PdfParser/Type/PdfHexString.php',
+ 'setasign\\Fpdi\\PdfParser\\Type\\PdfIndirectObject' => $vendorDir . '/setasign/fpdi/src/PdfParser/Type/PdfIndirectObject.php',
+ 'setasign\\Fpdi\\PdfParser\\Type\\PdfIndirectObjectReference' => $vendorDir . '/setasign/fpdi/src/PdfParser/Type/PdfIndirectObjectReference.php',
+ 'setasign\\Fpdi\\PdfParser\\Type\\PdfName' => $vendorDir . '/setasign/fpdi/src/PdfParser/Type/PdfName.php',
+ 'setasign\\Fpdi\\PdfParser\\Type\\PdfNull' => $vendorDir . '/setasign/fpdi/src/PdfParser/Type/PdfNull.php',
+ 'setasign\\Fpdi\\PdfParser\\Type\\PdfNumeric' => $vendorDir . '/setasign/fpdi/src/PdfParser/Type/PdfNumeric.php',
+ 'setasign\\Fpdi\\PdfParser\\Type\\PdfStream' => $vendorDir . '/setasign/fpdi/src/PdfParser/Type/PdfStream.php',
+ 'setasign\\Fpdi\\PdfParser\\Type\\PdfString' => $vendorDir . '/setasign/fpdi/src/PdfParser/Type/PdfString.php',
+ 'setasign\\Fpdi\\PdfParser\\Type\\PdfToken' => $vendorDir . '/setasign/fpdi/src/PdfParser/Type/PdfToken.php',
+ 'setasign\\Fpdi\\PdfParser\\Type\\PdfType' => $vendorDir . '/setasign/fpdi/src/PdfParser/Type/PdfType.php',
+ 'setasign\\Fpdi\\PdfParser\\Type\\PdfTypeException' => $vendorDir . '/setasign/fpdi/src/PdfParser/Type/PdfTypeException.php',
+ 'setasign\\Fpdi\\PdfReader\\DataStructure\\Rectangle' => $vendorDir . '/setasign/fpdi/src/PdfReader/DataStructure/Rectangle.php',
+ 'setasign\\Fpdi\\PdfReader\\Page' => $vendorDir . '/setasign/fpdi/src/PdfReader/Page.php',
+ 'setasign\\Fpdi\\PdfReader\\PageBoundaries' => $vendorDir . '/setasign/fpdi/src/PdfReader/PageBoundaries.php',
+ 'setasign\\Fpdi\\PdfReader\\PdfReader' => $vendorDir . '/setasign/fpdi/src/PdfReader/PdfReader.php',
+ 'setasign\\Fpdi\\PdfReader\\PdfReaderException' => $vendorDir . '/setasign/fpdi/src/PdfReader/PdfReaderException.php',
+ 'setasign\\Fpdi\\TcpdfFpdi' => $vendorDir . '/setasign/fpdi/src/TcpdfFpdi.php',
+ 'setasign\\Fpdi\\Tcpdf\\Fpdi' => $vendorDir . '/setasign/fpdi/src/Tcpdf/Fpdi.php',
+ 'setasign\\Fpdi\\Tfpdf\\FpdfTpl' => $vendorDir . '/setasign/fpdi/src/Tfpdf/FpdfTpl.php',
+ 'setasign\\Fpdi\\Tfpdf\\Fpdi' => $vendorDir . '/setasign/fpdi/src/Tfpdf/Fpdi.php',
);
diff --git a/api/vendor/composer/autoload_psr4.php b/api/vendor/composer/autoload_psr4.php
index 252cbb54..0b929495 100644
--- a/api/vendor/composer/autoload_psr4.php
+++ b/api/vendor/composer/autoload_psr4.php
@@ -6,6 +6,7 @@ $vendorDir = dirname(__DIR__);
$baseDir = dirname($vendorDir);
return array(
+ 'setasign\\Fpdi\\' => array($vendorDir . '/setasign/fpdi/src'),
'ZipStream\\' => array($vendorDir . '/maennchen/zipstream-php/src'),
'Stripe\\' => array($vendorDir . '/stripe/stripe-php/lib'),
'Psr\\SimpleCache\\' => array($vendorDir . '/psr/simple-cache/src'),
diff --git a/api/vendor/composer/autoload_static.php b/api/vendor/composer/autoload_static.php
index c3652825..d6ad4b10 100644
--- a/api/vendor/composer/autoload_static.php
+++ b/api/vendor/composer/autoload_static.php
@@ -7,6 +7,10 @@ namespace Composer\Autoload;
class ComposerStaticInit03e608fa83a14a82b3f9223977e9674e
{
public static $prefixLengthsPsr4 = array (
+ 's' =>
+ array (
+ 'setasign\\Fpdi\\' => 14,
+ ),
'Z' =>
array (
'ZipStream\\' => 10,
@@ -35,6 +39,10 @@ class ComposerStaticInit03e608fa83a14a82b3f9223977e9674e
);
public static $prefixDirsPsr4 = array (
+ 'setasign\\Fpdi\\' =>
+ array (
+ 0 => __DIR__ . '/..' . '/setasign/fpdi/src',
+ ),
'ZipStream\\' =>
array (
0 => __DIR__ . '/..' . '/maennchen/zipstream-php/src',
@@ -86,14 +94,19 @@ class ComposerStaticInit03e608fa83a14a82b3f9223977e9674e
'App\\Controllers\\ChatController' => __DIR__ . '/../..' . '/src/Controllers/ChatController.php',
'App\\Controllers\\EntiteController' => __DIR__ . '/../..' . '/src/Controllers/EntiteController.php',
'App\\Controllers\\FileController' => __DIR__ . '/../..' . '/src/Controllers/FileController.php',
+ 'App\\Controllers\\HealthController' => __DIR__ . '/../..' . '/src/Controllers/HealthController.php',
'App\\Controllers\\LoginController' => __DIR__ . '/../..' . '/src/Controllers/LoginController.php',
+ 'App\\Controllers\\MigrationController' => __DIR__ . '/../..' . '/src/Controllers/MigrationController.php',
'App\\Controllers\\OperationController' => __DIR__ . '/../..' . '/src/Controllers/OperationController.php',
'App\\Controllers\\PassageController' => __DIR__ . '/../..' . '/src/Controllers/PassageController.php',
'App\\Controllers\\PasswordController' => __DIR__ . '/../..' . '/src/Controllers/PasswordController.php',
'App\\Controllers\\SectorController' => __DIR__ . '/../..' . '/src/Controllers/SectorController.php',
'App\\Controllers\\SecurityController' => __DIR__ . '/../..' . '/src/Controllers/SecurityController.php',
+ 'App\\Controllers\\StripeController' => __DIR__ . '/../..' . '/src/Controllers/StripeController.php',
+ 'App\\Controllers\\StripeWebhookController' => __DIR__ . '/../..' . '/src/Controllers/StripeWebhookController.php',
'App\\Controllers\\UserController' => __DIR__ . '/../..' . '/src/Controllers/UserController.php',
'App\\Controllers\\VilleController' => __DIR__ . '/../..' . '/src/Controllers/VilleController.php',
+ 'App\\Core\\Controller' => __DIR__ . '/../..' . '/src/Core/Controller.php',
'App\\Services\\PDFGenerator' => __DIR__ . '/../..' . '/src/Services/PDFGenerator.php',
'App\\Services\\PasswordSecurityService' => __DIR__ . '/../..' . '/src/Services/PasswordSecurityService.php',
'App\\Services\\ReceiptPDFGenerator' => __DIR__ . '/../..' . '/src/Services/ReceiptPDFGenerator.php',
@@ -104,6 +117,7 @@ class ComposerStaticInit03e608fa83a14a82b3f9223977e9674e
'App\\Services\\Security\\PerformanceMonitor' => __DIR__ . '/../..' . '/src/Services/Security/PerformanceMonitor.php',
'App\\Services\\Security\\SecurityMonitor' => __DIR__ . '/../..' . '/src/Services/Security/SecurityMonitor.php',
'App\\Services\\SimplePDF' => __DIR__ . '/../..' . '/src/Services/SimplePDF.php',
+ 'App\\Services\\StripeService' => __DIR__ . '/../..' . '/src/Services/StripeService.php',
'BackupEncryptionService' => __DIR__ . '/../..' . '/src/Services/BackupEncryptionService.php',
'ClientDetector' => __DIR__ . '/../..' . '/src/Utils/ClientDetector.php',
'Complex\\Complex' => __DIR__ . '/..' . '/markbaker/complex/classes/src/Complex.php',
@@ -131,6 +145,7 @@ class ComposerStaticInit03e608fa83a14a82b3f9223977e9674e
'Database' => __DIR__ . '/../..' . '/src/Core/Database.php',
'DepartmentBoundaryService' => __DIR__ . '/../..' . '/src/Services/DepartmentBoundaryService.php',
'EmailTemplates' => __DIR__ . '/../..' . '/src/Services/EmailTemplates.php',
+ 'EventLogService' => __DIR__ . '/../..' . '/src/Services/EventLogService.php',
'ExportService' => __DIR__ . '/../..' . '/src/Services/ExportService.php',
'FPDF' => __DIR__ . '/..' . '/setasign/fpdf/fpdf.php',
'FileService' => __DIR__ . '/../..' . '/src/Services/FileService.php',
@@ -151,6 +166,7 @@ class ComposerStaticInit03e608fa83a14a82b3f9223977e9674e
'Matrix\\Operators\\Multiplication' => __DIR__ . '/..' . '/markbaker/matrix/classes/src/Operators/Multiplication.php',
'Matrix\\Operators\\Operator' => __DIR__ . '/..' . '/markbaker/matrix/classes/src/Operators/Operator.php',
'Matrix\\Operators\\Subtraction' => __DIR__ . '/..' . '/markbaker/matrix/classes/src/Operators/Subtraction.php',
+ 'MigrationService' => __DIR__ . '/../..' . '/src/Services/MigrationService.php',
'MonitoredDatabase' => __DIR__ . '/../..' . '/src/Core/MonitoredDatabase.php',
'MonitoredStatement' => __DIR__ . '/../..' . '/src/Core/MonitoredDatabase.php',
'OperationDataService' => __DIR__ . '/../..' . '/src/Services/OperationDataService.php',
@@ -164,6 +180,8 @@ class ComposerStaticInit03e608fa83a14a82b3f9223977e9674e
'PhpOffice\\PhpSpreadsheet\\Calculation\\ArrayEnabled' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/ArrayEnabled.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\BinaryComparison' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/BinaryComparison.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\Calculation' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Calculation.php',
+ 'PhpOffice\\PhpSpreadsheet\\Calculation\\CalculationBase' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/CalculationBase.php',
+ 'PhpOffice\\PhpSpreadsheet\\Calculation\\CalculationLocale' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/CalculationLocale.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\Category' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Category.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\Database\\DAverage' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DAverage.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\Database\\DCount' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DCount.php',
@@ -249,21 +267,25 @@ class ComposerStaticInit03e608fa83a14a82b3f9223977e9674e
'PhpOffice\\PhpSpreadsheet\\Calculation\\Financial\\TreasuryBill' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/TreasuryBill.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\FormulaParser' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/FormulaParser.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\FormulaToken' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/FormulaToken.php',
+ 'PhpOffice\\PhpSpreadsheet\\Calculation\\FunctionArray' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/FunctionArray.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\Functions' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Functions.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\Information\\ErrorValue' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Information/ErrorValue.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\Information\\ExcelError' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Information/ExcelError.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\Information\\Value' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Information/Value.php',
+ 'PhpOffice\\PhpSpreadsheet\\Calculation\\Internal\\ExcelArrayPseudoFunctions' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Internal/ExcelArrayPseudoFunctions.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\Internal\\MakeMatrix' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Internal/MakeMatrix.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\Internal\\WildcardMatch' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Internal/WildcardMatch.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\Logical\\Boolean' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Logical/Boolean.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\Logical\\Conditional' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Logical/Conditional.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\Logical\\Operations' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Logical/Operations.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\Address' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Address.php',
+ 'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\ChooseRowsEtc' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/ChooseRowsEtc.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\ExcelMatch' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/ExcelMatch.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\Filter' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Filter.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\Formula' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Formula.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\HLookup' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/HLookup.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\Helpers' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Helpers.php',
+ 'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\Hstack' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Hstack.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\Hyperlink' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Hyperlink.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\Indirect' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Indirect.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\Lookup' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Lookup.php',
@@ -274,8 +296,10 @@ class ComposerStaticInit03e608fa83a14a82b3f9223977e9674e
'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\RowColumnInformation' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/RowColumnInformation.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\Selection' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Selection.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\Sort' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Sort.php',
+ 'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\TorowTocol' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/TorowTocol.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\Unique' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Unique.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\VLookup' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/VLookup.php',
+ 'PhpOffice\\PhpSpreadsheet\\Calculation\\LookupRef\\Vstack' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Vstack.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\MathTrig\\Absolute' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/MathTrig/Absolute.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\MathTrig\\Angle' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/MathTrig/Angle.php',
'PhpOffice\\PhpSpreadsheet\\Calculation\\MathTrig\\Arabic' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/MathTrig/Arabic.php',
@@ -435,6 +459,9 @@ class ComposerStaticInit03e608fa83a14a82b3f9223977e9674e
'PhpOffice\\PhpSpreadsheet\\Reader\\Security\\XmlScanner' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Security/XmlScanner.php',
'PhpOffice\\PhpSpreadsheet\\Reader\\Slk' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Slk.php',
'PhpOffice\\PhpSpreadsheet\\Reader\\Xls' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls.php',
+ 'PhpOffice\\PhpSpreadsheet\\Reader\\XlsBase' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/XlsBase.php',
+ 'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\Biff5' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/Biff5.php',
+ 'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\Biff8' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/Biff8.php',
'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\Color' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/Color.php',
'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\Color\\BIFF5' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/Color/BIFF5.php',
'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\Color\\BIFF8' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/Color/BIFF8.php',
@@ -443,7 +470,10 @@ class ComposerStaticInit03e608fa83a14a82b3f9223977e9674e
'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\DataValidationHelper' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/DataValidationHelper.php',
'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\ErrorCode' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/ErrorCode.php',
'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\Escher' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/Escher.php',
+ 'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\ListFunctions' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/ListFunctions.php',
+ 'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\LoadSpreadsheet' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/LoadSpreadsheet.php',
'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\MD5' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/MD5.php',
+ 'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\Mappings' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/Mappings.php',
'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\RC4' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/RC4.php',
'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\Style\\Border' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/Style/Border.php',
'PhpOffice\\PhpSpreadsheet\\Reader\\Xls\\Style\\CellAlignment' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Reader/Xls/Style/CellAlignment.php',
@@ -529,6 +559,8 @@ class ComposerStaticInit03e608fa83a14a82b3f9223977e9674e
'PhpOffice\\PhpSpreadsheet\\Style\\ConditionalFormatting\\ConditionalDataBarExtension' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/ConditionalFormatting/ConditionalDataBarExtension.php',
'PhpOffice\\PhpSpreadsheet\\Style\\ConditionalFormatting\\ConditionalFormatValueObject' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/ConditionalFormatting/ConditionalFormatValueObject.php',
'PhpOffice\\PhpSpreadsheet\\Style\\ConditionalFormatting\\ConditionalFormattingRuleExtension' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/ConditionalFormatting/ConditionalFormattingRuleExtension.php',
+ 'PhpOffice\\PhpSpreadsheet\\Style\\ConditionalFormatting\\ConditionalIconSet' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/ConditionalFormatting/ConditionalIconSet.php',
+ 'PhpOffice\\PhpSpreadsheet\\Style\\ConditionalFormatting\\IconSetValues' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/ConditionalFormatting/IconSetValues.php',
'PhpOffice\\PhpSpreadsheet\\Style\\ConditionalFormatting\\StyleMerger' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/ConditionalFormatting/StyleMerger.php',
'PhpOffice\\PhpSpreadsheet\\Style\\ConditionalFormatting\\Wizard' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/ConditionalFormatting/Wizard.php',
'PhpOffice\\PhpSpreadsheet\\Style\\ConditionalFormatting\\Wizard\\Blanks' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/ConditionalFormatting/Wizard/Blanks.php',
@@ -551,6 +583,8 @@ class ComposerStaticInit03e608fa83a14a82b3f9223977e9674e
'PhpOffice\\PhpSpreadsheet\\Style\\NumberFormat\\PercentageFormatter' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/NumberFormat/PercentageFormatter.php',
'PhpOffice\\PhpSpreadsheet\\Style\\NumberFormat\\Wizard\\Accounting' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/NumberFormat/Wizard/Accounting.php',
'PhpOffice\\PhpSpreadsheet\\Style\\NumberFormat\\Wizard\\Currency' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/NumberFormat/Wizard/Currency.php',
+ 'PhpOffice\\PhpSpreadsheet\\Style\\NumberFormat\\Wizard\\CurrencyBase' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/NumberFormat/Wizard/CurrencyBase.php',
+ 'PhpOffice\\PhpSpreadsheet\\Style\\NumberFormat\\Wizard\\CurrencyNegative' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/NumberFormat/Wizard/CurrencyNegative.php',
'PhpOffice\\PhpSpreadsheet\\Style\\NumberFormat\\Wizard\\Date' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/NumberFormat/Wizard/Date.php',
'PhpOffice\\PhpSpreadsheet\\Style\\NumberFormat\\Wizard\\DateTime' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/NumberFormat/Wizard/DateTime.php',
'PhpOffice\\PhpSpreadsheet\\Style\\NumberFormat\\Wizard\\DateTimeWizard' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Style/NumberFormat/Wizard/DateTimeWizard.php',
@@ -597,6 +631,7 @@ class ComposerStaticInit03e608fa83a14a82b3f9223977e9674e
'PhpOffice\\PhpSpreadsheet\\Worksheet\\SheetView' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Worksheet/SheetView.php',
'PhpOffice\\PhpSpreadsheet\\Worksheet\\Table' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Worksheet/Table.php',
'PhpOffice\\PhpSpreadsheet\\Worksheet\\Table\\Column' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Worksheet/Table/Column.php',
+ 'PhpOffice\\PhpSpreadsheet\\Worksheet\\Table\\TableDxfsStyle' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Worksheet/Table/TableDxfsStyle.php',
'PhpOffice\\PhpSpreadsheet\\Worksheet\\Table\\TableStyle' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Worksheet/Table/TableStyle.php',
'PhpOffice\\PhpSpreadsheet\\Worksheet\\Validations' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Worksheet/Validations.php',
'PhpOffice\\PhpSpreadsheet\\Worksheet\\Worksheet' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Worksheet/Worksheet.php',
@@ -634,7 +669,6 @@ class ComposerStaticInit03e608fa83a14a82b3f9223977e9674e
'PhpOffice\\PhpSpreadsheet\\Writer\\Xls\\Style\\CellAlignment' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Writer/Xls/Style/CellAlignment.php',
'PhpOffice\\PhpSpreadsheet\\Writer\\Xls\\Style\\CellBorder' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Writer/Xls/Style/CellBorder.php',
'PhpOffice\\PhpSpreadsheet\\Writer\\Xls\\Style\\CellFill' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Writer/Xls/Style/CellFill.php',
- 'PhpOffice\\PhpSpreadsheet\\Writer\\Xls\\Style\\ColorMap' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Writer/Xls/Style/ColorMap.php',
'PhpOffice\\PhpSpreadsheet\\Writer\\Xls\\Workbook' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Writer/Xls/Workbook.php',
'PhpOffice\\PhpSpreadsheet\\Writer\\Xls\\Worksheet' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Writer/Xls/Worksheet.php',
'PhpOffice\\PhpSpreadsheet\\Writer\\Xls\\Xf' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Writer/Xls/Xf.php',
@@ -647,6 +681,7 @@ class ComposerStaticInit03e608fa83a14a82b3f9223977e9674e
'PhpOffice\\PhpSpreadsheet\\Writer\\Xlsx\\DocProps' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Writer/Xlsx/DocProps.php',
'PhpOffice\\PhpSpreadsheet\\Writer\\Xlsx\\Drawing' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Writer/Xlsx/Drawing.php',
'PhpOffice\\PhpSpreadsheet\\Writer\\Xlsx\\FunctionPrefix' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Writer/Xlsx/FunctionPrefix.php',
+ 'PhpOffice\\PhpSpreadsheet\\Writer\\Xlsx\\Metadata' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Writer/Xlsx/Metadata.php',
'PhpOffice\\PhpSpreadsheet\\Writer\\Xlsx\\Rels' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Writer/Xlsx/Rels.php',
'PhpOffice\\PhpSpreadsheet\\Writer\\Xlsx\\RelsRibbon' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Writer/Xlsx/RelsRibbon.php',
'PhpOffice\\PhpSpreadsheet\\Writer\\Xlsx\\RelsVBA' => __DIR__ . '/..' . '/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Writer/Xlsx/RelsVBA.php',
@@ -1074,6 +1109,7 @@ class ComposerStaticInit03e608fa83a14a82b3f9223977e9674e
'ZipStream\\LocalFileHeader' => __DIR__ . '/..' . '/maennchen/zipstream-php/src/LocalFileHeader.php',
'ZipStream\\OperationMode' => __DIR__ . '/..' . '/maennchen/zipstream-php/src/OperationMode.php',
'ZipStream\\PackField' => __DIR__ . '/..' . '/maennchen/zipstream-php/src/PackField.php',
+ 'ZipStream\\Stream\\CallbackStreamWrapper' => __DIR__ . '/..' . '/maennchen/zipstream-php/src/Stream/CallbackStreamWrapper.php',
'ZipStream\\Time' => __DIR__ . '/..' . '/maennchen/zipstream-php/src/Time.php',
'ZipStream\\Version' => __DIR__ . '/..' . '/maennchen/zipstream-php/src/Version.php',
'ZipStream\\Zip64\\DataDescriptor' => __DIR__ . '/..' . '/maennchen/zipstream-php/src/Zip64/DataDescriptor.php',
@@ -1082,6 +1118,57 @@ class ComposerStaticInit03e608fa83a14a82b3f9223977e9674e
'ZipStream\\Zip64\\ExtendedInformationExtraField' => __DIR__ . '/..' . '/maennchen/zipstream-php/src/Zip64/ExtendedInformationExtraField.php',
'ZipStream\\ZipStream' => __DIR__ . '/..' . '/maennchen/zipstream-php/src/ZipStream.php',
'ZipStream\\Zs\\ExtendedInformationExtraField' => __DIR__ . '/..' . '/maennchen/zipstream-php/src/Zs/ExtendedInformationExtraField.php',
+ 'setasign\\Fpdi\\FpdfTpl' => __DIR__ . '/..' . '/setasign/fpdi/src/FpdfTpl.php',
+ 'setasign\\Fpdi\\FpdfTplTrait' => __DIR__ . '/..' . '/setasign/fpdi/src/FpdfTplTrait.php',
+ 'setasign\\Fpdi\\FpdfTrait' => __DIR__ . '/..' . '/setasign/fpdi/src/FpdfTrait.php',
+ 'setasign\\Fpdi\\Fpdi' => __DIR__ . '/..' . '/setasign/fpdi/src/Fpdi.php',
+ 'setasign\\Fpdi\\FpdiException' => __DIR__ . '/..' . '/setasign/fpdi/src/FpdiException.php',
+ 'setasign\\Fpdi\\FpdiTrait' => __DIR__ . '/..' . '/setasign/fpdi/src/FpdiTrait.php',
+ 'setasign\\Fpdi\\GraphicsState' => __DIR__ . '/..' . '/setasign/fpdi/src/GraphicsState.php',
+ 'setasign\\Fpdi\\Math\\Matrix' => __DIR__ . '/..' . '/setasign/fpdi/src/Math/Matrix.php',
+ 'setasign\\Fpdi\\Math\\Vector' => __DIR__ . '/..' . '/setasign/fpdi/src/Math/Vector.php',
+ 'setasign\\Fpdi\\PdfParser\\CrossReference\\AbstractReader' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/CrossReference/AbstractReader.php',
+ 'setasign\\Fpdi\\PdfParser\\CrossReference\\CrossReference' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/CrossReference/CrossReference.php',
+ 'setasign\\Fpdi\\PdfParser\\CrossReference\\CrossReferenceException' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/CrossReference/CrossReferenceException.php',
+ 'setasign\\Fpdi\\PdfParser\\CrossReference\\FixedReader' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/CrossReference/FixedReader.php',
+ 'setasign\\Fpdi\\PdfParser\\CrossReference\\LineReader' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/CrossReference/LineReader.php',
+ 'setasign\\Fpdi\\PdfParser\\CrossReference\\ReaderInterface' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/CrossReference/ReaderInterface.php',
+ 'setasign\\Fpdi\\PdfParser\\Filter\\Ascii85' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/Filter/Ascii85.php',
+ 'setasign\\Fpdi\\PdfParser\\Filter\\Ascii85Exception' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/Filter/Ascii85Exception.php',
+ 'setasign\\Fpdi\\PdfParser\\Filter\\AsciiHex' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/Filter/AsciiHex.php',
+ 'setasign\\Fpdi\\PdfParser\\Filter\\FilterException' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/Filter/FilterException.php',
+ 'setasign\\Fpdi\\PdfParser\\Filter\\FilterInterface' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/Filter/FilterInterface.php',
+ 'setasign\\Fpdi\\PdfParser\\Filter\\Flate' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/Filter/Flate.php',
+ 'setasign\\Fpdi\\PdfParser\\Filter\\FlateException' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/Filter/FlateException.php',
+ 'setasign\\Fpdi\\PdfParser\\Filter\\Lzw' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/Filter/Lzw.php',
+ 'setasign\\Fpdi\\PdfParser\\Filter\\LzwException' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/Filter/LzwException.php',
+ 'setasign\\Fpdi\\PdfParser\\PdfParser' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/PdfParser.php',
+ 'setasign\\Fpdi\\PdfParser\\PdfParserException' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/PdfParserException.php',
+ 'setasign\\Fpdi\\PdfParser\\StreamReader' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/StreamReader.php',
+ 'setasign\\Fpdi\\PdfParser\\Tokenizer' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/Tokenizer.php',
+ 'setasign\\Fpdi\\PdfParser\\Type\\PdfArray' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/Type/PdfArray.php',
+ 'setasign\\Fpdi\\PdfParser\\Type\\PdfBoolean' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/Type/PdfBoolean.php',
+ 'setasign\\Fpdi\\PdfParser\\Type\\PdfDictionary' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/Type/PdfDictionary.php',
+ 'setasign\\Fpdi\\PdfParser\\Type\\PdfHexString' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/Type/PdfHexString.php',
+ 'setasign\\Fpdi\\PdfParser\\Type\\PdfIndirectObject' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/Type/PdfIndirectObject.php',
+ 'setasign\\Fpdi\\PdfParser\\Type\\PdfIndirectObjectReference' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/Type/PdfIndirectObjectReference.php',
+ 'setasign\\Fpdi\\PdfParser\\Type\\PdfName' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/Type/PdfName.php',
+ 'setasign\\Fpdi\\PdfParser\\Type\\PdfNull' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/Type/PdfNull.php',
+ 'setasign\\Fpdi\\PdfParser\\Type\\PdfNumeric' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/Type/PdfNumeric.php',
+ 'setasign\\Fpdi\\PdfParser\\Type\\PdfStream' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/Type/PdfStream.php',
+ 'setasign\\Fpdi\\PdfParser\\Type\\PdfString' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/Type/PdfString.php',
+ 'setasign\\Fpdi\\PdfParser\\Type\\PdfToken' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/Type/PdfToken.php',
+ 'setasign\\Fpdi\\PdfParser\\Type\\PdfType' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/Type/PdfType.php',
+ 'setasign\\Fpdi\\PdfParser\\Type\\PdfTypeException' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfParser/Type/PdfTypeException.php',
+ 'setasign\\Fpdi\\PdfReader\\DataStructure\\Rectangle' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfReader/DataStructure/Rectangle.php',
+ 'setasign\\Fpdi\\PdfReader\\Page' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfReader/Page.php',
+ 'setasign\\Fpdi\\PdfReader\\PageBoundaries' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfReader/PageBoundaries.php',
+ 'setasign\\Fpdi\\PdfReader\\PdfReader' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfReader/PdfReader.php',
+ 'setasign\\Fpdi\\PdfReader\\PdfReaderException' => __DIR__ . '/..' . '/setasign/fpdi/src/PdfReader/PdfReaderException.php',
+ 'setasign\\Fpdi\\TcpdfFpdi' => __DIR__ . '/..' . '/setasign/fpdi/src/TcpdfFpdi.php',
+ 'setasign\\Fpdi\\Tcpdf\\Fpdi' => __DIR__ . '/..' . '/setasign/fpdi/src/Tcpdf/Fpdi.php',
+ 'setasign\\Fpdi\\Tfpdf\\FpdfTpl' => __DIR__ . '/..' . '/setasign/fpdi/src/Tfpdf/FpdfTpl.php',
+ 'setasign\\Fpdi\\Tfpdf\\Fpdi' => __DIR__ . '/..' . '/setasign/fpdi/src/Tfpdf/Fpdi.php',
);
public static function getInitializer(ClassLoader $loader)
diff --git a/api/vendor/composer/installed.json b/api/vendor/composer/installed.json
index 87457e80..365e4d9c 100644
--- a/api/vendor/composer/installed.json
+++ b/api/vendor/composer/installed.json
@@ -84,23 +84,23 @@
},
{
"name": "maennchen/zipstream-php",
- "version": "3.1.2",
- "version_normalized": "3.1.2.0",
+ "version": "3.2.0",
+ "version_normalized": "3.2.0.0",
"source": {
"type": "git",
"url": "https://github.com/maennchen/ZipStream-PHP.git",
- "reference": "aeadcf5c412332eb426c0f9b4485f6accba2a99f"
+ "reference": "9712d8fa4cdf9240380b01eb4be55ad8dcf71416"
},
"dist": {
"type": "zip",
- "url": "https://api.github.com/repos/maennchen/ZipStream-PHP/zipball/aeadcf5c412332eb426c0f9b4485f6accba2a99f",
- "reference": "aeadcf5c412332eb426c0f9b4485f6accba2a99f",
+ "url": "https://api.github.com/repos/maennchen/ZipStream-PHP/zipball/9712d8fa4cdf9240380b01eb4be55ad8dcf71416",
+ "reference": "9712d8fa4cdf9240380b01eb4be55ad8dcf71416",
"shasum": ""
},
"require": {
"ext-mbstring": "*",
"ext-zlib": "*",
- "php-64bit": "^8.2"
+ "php-64bit": "^8.3"
},
"require-dev": {
"brianium/paratest": "^7.7",
@@ -109,14 +109,14 @@
"guzzlehttp/guzzle": "^7.5",
"mikey179/vfsstream": "^1.6",
"php-coveralls/php-coveralls": "^2.5",
- "phpunit/phpunit": "^11.0",
+ "phpunit/phpunit": "^12.0",
"vimeo/psalm": "^6.0"
},
"suggest": {
"guzzlehttp/psr7": "^2.4",
"psr/http-message": "^2.0"
},
- "time": "2025-01-27T12:07:53+00:00",
+ "time": "2025-07-17T11:15:13+00:00",
"type": "library",
"installation-source": "dist",
"autoload": {
@@ -153,7 +153,7 @@
],
"support": {
"issues": "https://github.com/maennchen/ZipStream-PHP/issues",
- "source": "https://github.com/maennchen/ZipStream-PHP/tree/3.1.2"
+ "source": "https://github.com/maennchen/ZipStream-PHP/tree/3.2.0"
},
"funding": [
{
@@ -278,17 +278,17 @@
},
{
"name": "phpmailer/phpmailer",
- "version": "v6.10.0",
- "version_normalized": "6.10.0.0",
+ "version": "v6.11.1",
+ "version_normalized": "6.11.1.0",
"source": {
"type": "git",
"url": "https://github.com/PHPMailer/PHPMailer.git",
- "reference": "bf74d75a1fde6beaa34a0ddae2ec5fce0f72a144"
+ "reference": "d9e3b36b47f04b497a0164c5a20f92acb4593284"
},
"dist": {
"type": "zip",
- "url": "https://api.github.com/repos/PHPMailer/PHPMailer/zipball/bf74d75a1fde6beaa34a0ddae2ec5fce0f72a144",
- "reference": "bf74d75a1fde6beaa34a0ddae2ec5fce0f72a144",
+ "url": "https://api.github.com/repos/PHPMailer/PHPMailer/zipball/d9e3b36b47f04b497a0164c5a20f92acb4593284",
+ "reference": "d9e3b36b47f04b497a0164c5a20f92acb4593284",
"shasum": ""
},
"require": {
@@ -309,6 +309,7 @@
},
"suggest": {
"decomplexity/SendOauth2": "Adapter for using XOAUTH2 authentication",
+ "ext-imap": "Needed to support advanced email address parsing according to RFC822",
"ext-mbstring": "Needed to send email in multibyte encoding charset or decode encoded addresses",
"ext-openssl": "Needed for secure SMTP sending and DKIM signing",
"greew/oauth2-azure-provider": "Needed for Microsoft Azure XOAUTH2 authentication",
@@ -318,7 +319,7 @@
"symfony/polyfill-mbstring": "To support UTF-8 if the Mbstring PHP extension is not enabled (^1.2)",
"thenetworg/oauth2-azure": "Needed for Microsoft XOAUTH2 authentication"
},
- "time": "2025-04-24T15:19:31+00:00",
+ "time": "2025-09-30T11:54:53+00:00",
"type": "library",
"installation-source": "dist",
"autoload": {
@@ -350,7 +351,7 @@
"description": "PHPMailer is a full-featured email creation and transfer class for PHP",
"support": {
"issues": "https://github.com/PHPMailer/PHPMailer/issues",
- "source": "https://github.com/PHPMailer/PHPMailer/tree/v6.10.0"
+ "source": "https://github.com/PHPMailer/PHPMailer/tree/v6.11.1"
},
"funding": [
{
@@ -362,21 +363,21 @@
},
{
"name": "phpoffice/phpspreadsheet",
- "version": "2.3.8",
- "version_normalized": "2.3.8.0",
+ "version": "5.1.0",
+ "version_normalized": "5.1.0.0",
"source": {
"type": "git",
"url": "https://github.com/PHPOffice/PhpSpreadsheet.git",
- "reference": "7a700683743bf1c4a21837c84b266916f1aa7d25"
+ "reference": "fd26e45a814e94ae2aad0df757d9d1739c4bf2e0"
},
"dist": {
"type": "zip",
- "url": "https://api.github.com/repos/PHPOffice/PhpSpreadsheet/zipball/7a700683743bf1c4a21837c84b266916f1aa7d25",
- "reference": "7a700683743bf1c4a21837c84b266916f1aa7d25",
+ "url": "https://api.github.com/repos/PHPOffice/PhpSpreadsheet/zipball/fd26e45a814e94ae2aad0df757d9d1739c4bf2e0",
+ "reference": "fd26e45a814e94ae2aad0df757d9d1739c4bf2e0",
"shasum": ""
},
"require": {
- "composer/pcre": "^1 || ^2 || ^3",
+ "composer/pcre": "^1||^2||^3",
"ext-ctype": "*",
"ext-dom": "*",
"ext-fileinfo": "*",
@@ -405,9 +406,10 @@
"mitoteam/jpgraph": "^10.3",
"mpdf/mpdf": "^8.1.1",
"phpcompatibility/php-compatibility": "^9.3",
- "phpstan/phpstan": "^1.1",
- "phpstan/phpstan-phpunit": "^1.0",
- "phpunit/phpunit": "^9.6 || ^10.5",
+ "phpstan/phpstan": "^1.1 || ^2.0",
+ "phpstan/phpstan-deprecation-rules": "^1.0 || ^2.0",
+ "phpstan/phpstan-phpunit": "^1.0 || ^2.0",
+ "phpunit/phpunit": "^10.5",
"squizlabs/php_codesniffer": "^3.7",
"tecnickcom/tcpdf": "^6.5"
},
@@ -418,7 +420,7 @@
"mpdf/mpdf": "Option for rendering PDF with PDF Writer",
"tecnickcom/tcpdf": "Option for rendering PDF with PDF Writer"
},
- "time": "2025-02-08T03:01:45+00:00",
+ "time": "2025-09-04T05:34:49+00:00",
"type": "library",
"installation-source": "dist",
"autoload": {
@@ -464,7 +466,7 @@
],
"support": {
"issues": "https://github.com/PHPOffice/PhpSpreadsheet/issues",
- "source": "https://github.com/PHPOffice/PhpSpreadsheet/tree/2.3.8"
+ "source": "https://github.com/PHPOffice/PhpSpreadsheet/tree/5.1.0"
},
"install-path": "../phpoffice/phpspreadsheet"
},
@@ -740,6 +742,81 @@
},
"install-path": "../setasign/fpdf"
},
+ {
+ "name": "setasign/fpdi",
+ "version": "v2.6.4",
+ "version_normalized": "2.6.4.0",
+ "source": {
+ "type": "git",
+ "url": "https://github.com/Setasign/FPDI.git",
+ "reference": "4b53852fde2734ec6a07e458a085db627c60eada"
+ },
+ "dist": {
+ "type": "zip",
+ "url": "https://api.github.com/repos/Setasign/FPDI/zipball/4b53852fde2734ec6a07e458a085db627c60eada",
+ "reference": "4b53852fde2734ec6a07e458a085db627c60eada",
+ "shasum": ""
+ },
+ "require": {
+ "ext-zlib": "*",
+ "php": "^7.1 || ^8.0"
+ },
+ "conflict": {
+ "setasign/tfpdf": "<1.31"
+ },
+ "require-dev": {
+ "phpunit/phpunit": "^7",
+ "setasign/fpdf": "~1.8.6",
+ "setasign/tfpdf": "~1.33",
+ "squizlabs/php_codesniffer": "^3.5",
+ "tecnickcom/tcpdf": "^6.8"
+ },
+ "suggest": {
+ "setasign/fpdf": "FPDI will extend this class but as it is also possible to use TCPDF or tFPDF as an alternative. There's no fixed dependency configured."
+ },
+ "time": "2025-08-05T09:57:14+00:00",
+ "type": "library",
+ "installation-source": "dist",
+ "autoload": {
+ "psr-4": {
+ "setasign\\Fpdi\\": "src/"
+ }
+ },
+ "notification-url": "https://packagist.org/downloads/",
+ "license": [
+ "MIT"
+ ],
+ "authors": [
+ {
+ "name": "Jan Slabon",
+ "email": "jan.slabon@setasign.com",
+ "homepage": "https://www.setasign.com"
+ },
+ {
+ "name": "Maximilian Kresse",
+ "email": "maximilian.kresse@setasign.com",
+ "homepage": "https://www.setasign.com"
+ }
+ ],
+ "description": "FPDI is a collection of PHP classes facilitating developers to read pages from existing PDF documents and use them as templates in FPDF. Because it is also possible to use FPDI with TCPDF, there are no fixed dependencies defined. Please see suggestions for packages which evaluates the dependencies automatically.",
+ "homepage": "https://www.setasign.com/fpdi",
+ "keywords": [
+ "fpdf",
+ "fpdi",
+ "pdf"
+ ],
+ "support": {
+ "issues": "https://github.com/Setasign/FPDI/issues",
+ "source": "https://github.com/Setasign/FPDI/tree/v2.6.4"
+ },
+ "funding": [
+ {
+ "url": "https://tidelift.com/funding/github/packagist/setasign/fpdi",
+ "type": "tidelift"
+ }
+ ],
+ "install-path": "../setasign/fpdi"
+ },
{
"name": "stripe/stripe-php",
"version": "v17.6.0",
diff --git a/api/vendor/composer/installed.php b/api/vendor/composer/installed.php
index 4763ee94..9ca19fd2 100644
--- a/api/vendor/composer/installed.php
+++ b/api/vendor/composer/installed.php
@@ -3,7 +3,7 @@
'name' => 'your-vendor/api',
'pretty_version' => 'dev-main',
'version' => 'dev-main',
- 'reference' => 'f597c9aeb504adc2d733e5e2bd70820b06049df9',
+ 'reference' => '2b3d05c981bd6fa1e80c3459d8648bc65bea72e2',
'type' => 'project',
'install_path' => __DIR__ . '/../../',
'aliases' => array(),
@@ -20,9 +20,9 @@
'dev_requirement' => false,
),
'maennchen/zipstream-php' => array(
- 'pretty_version' => '3.1.2',
- 'version' => '3.1.2.0',
- 'reference' => 'aeadcf5c412332eb426c0f9b4485f6accba2a99f',
+ 'pretty_version' => '3.2.0',
+ 'version' => '3.2.0.0',
+ 'reference' => '9712d8fa4cdf9240380b01eb4be55ad8dcf71416',
'type' => 'library',
'install_path' => __DIR__ . '/../maennchen/zipstream-php',
'aliases' => array(),
@@ -47,18 +47,18 @@
'dev_requirement' => false,
),
'phpmailer/phpmailer' => array(
- 'pretty_version' => 'v6.10.0',
- 'version' => '6.10.0.0',
- 'reference' => 'bf74d75a1fde6beaa34a0ddae2ec5fce0f72a144',
+ 'pretty_version' => 'v6.11.1',
+ 'version' => '6.11.1.0',
+ 'reference' => 'd9e3b36b47f04b497a0164c5a20f92acb4593284',
'type' => 'library',
'install_path' => __DIR__ . '/../phpmailer/phpmailer',
'aliases' => array(),
'dev_requirement' => false,
),
'phpoffice/phpspreadsheet' => array(
- 'pretty_version' => '2.3.8',
- 'version' => '2.3.8.0',
- 'reference' => '7a700683743bf1c4a21837c84b266916f1aa7d25',
+ 'pretty_version' => '5.1.0',
+ 'version' => '5.1.0.0',
+ 'reference' => 'fd26e45a814e94ae2aad0df757d9d1739c4bf2e0',
'type' => 'library',
'install_path' => __DIR__ . '/../phpoffice/phpspreadsheet',
'aliases' => array(),
@@ -109,6 +109,15 @@
'aliases' => array(),
'dev_requirement' => false,
),
+ 'setasign/fpdi' => array(
+ 'pretty_version' => 'v2.6.4',
+ 'version' => '2.6.4.0',
+ 'reference' => '4b53852fde2734ec6a07e458a085db627c60eada',
+ 'type' => 'library',
+ 'install_path' => __DIR__ . '/../setasign/fpdi',
+ 'aliases' => array(),
+ 'dev_requirement' => false,
+ ),
'stripe/stripe-php' => array(
'pretty_version' => 'v17.6.0',
'version' => '17.6.0.0',
@@ -121,7 +130,7 @@
'your-vendor/api' => array(
'pretty_version' => 'dev-main',
'version' => 'dev-main',
- 'reference' => 'f597c9aeb504adc2d733e5e2bd70820b06049df9',
+ 'reference' => '2b3d05c981bd6fa1e80c3459d8648bc65bea72e2',
'type' => 'project',
'install_path' => __DIR__ . '/../../',
'aliases' => array(),
diff --git a/api/vendor/maennchen/zipstream-php/.gitattributes b/api/vendor/maennchen/zipstream-php/.gitattributes
deleted file mode 100644
index e058ebd0..00000000
--- a/api/vendor/maennchen/zipstream-php/.gitattributes
+++ /dev/null
@@ -1,6 +0,0 @@
-.gitignore text eol=lf
-.gitattributes text eol=lf
-*.md text eol=lf
-*.php text eol=lf
-*.yml text eol=lf
-*.xml text eol=lf
diff --git a/api/vendor/maennchen/zipstream-php/.github/CODE_OF_CONDUCT.md b/api/vendor/maennchen/zipstream-php/.github/CODE_OF_CONDUCT.md
deleted file mode 100644
index 9d75b876..00000000
--- a/api/vendor/maennchen/zipstream-php/.github/CODE_OF_CONDUCT.md
+++ /dev/null
@@ -1,132 +0,0 @@
-# Contributor Covenant Code of Conduct
-
-## Our Pledge
-
-We as members, contributors, and leaders pledge to make participation in our
-community a harassment-free experience for everyone, regardless of age, body
-size, visible or invisible disability, ethnicity, sex characteristics, gender
-identity and expression, level of experience, education, socio-economic status,
-nationality, personal appearance, race, caste, color, religion, or sexual
-identity and orientation.
-
-We pledge to act and interact in ways that contribute to an open, welcoming,
-diverse, inclusive, and healthy community.
-
-## Our Standards
-
-Examples of behavior that contributes to a positive environment for our
-community include:
-
-- Demonstrating empathy and kindness toward other people
-- Being respectful of differing opinions, viewpoints, and experiences
-- Giving and gracefully accepting constructive feedback
-- Accepting responsibility and apologizing to those affected by our mistakes,
- and learning from the experience
-- Focusing on what is best not just for us as individuals, but for the overall
- community
-
-Examples of unacceptable behavior include:
-
-- The use of sexualized language or imagery, and sexual attention or advances of
- any kind
-- Trolling, insulting or derogatory comments, and personal or political attacks
-- Public or private harassment
-- Publishing others' private information, such as a physical or email address,
- without their explicit permission
-- Other conduct which could reasonably be considered inappropriate in a
- professional setting
-
-## Enforcement Responsibilities
-
-Community leaders are responsible for clarifying and enforcing our standards of
-acceptable behavior and will take appropriate and fair corrective action in
-response to any behavior that they deem inappropriate, threatening, offensive,
-or harmful.
-
-Community leaders have the right and responsibility to remove, edit, or reject
-comments, commits, code, wiki edits, issues, and other contributions that are
-not aligned to this Code of Conduct, and will communicate reasons for moderation
-decisions when appropriate.
-
-## Scope
-
-This Code of Conduct applies within all community spaces, and also applies when
-an individual is officially representing the community in public spaces.
-Examples of representing our community include using an official e-mail address,
-posting via an official social media account, or acting as an appointed
-representative at an online or offline event.
-
-## Enforcement
-
-Instances of abusive, harassing, or otherwise unacceptable behavior may be
-reported to the community leaders responsible for enforcement at
-jonatan@maennchen.ch.
-All complaints will be reviewed and investigated promptly and fairly.
-
-All community leaders are obligated to respect the privacy and security of the
-reporter of any incident.
-
-## Enforcement Guidelines
-
-Community leaders will follow these Community Impact Guidelines in determining
-the consequences for any action they deem in violation of this Code of Conduct:
-
-### 1. Correction
-
-**Community Impact**: Use of inappropriate language or other behavior deemed
-unprofessional or unwelcome in the community.
-
-**Consequence**: A private, written warning from community leaders, providing
-clarity around the nature of the violation and an explanation of why the
-behavior was inappropriate. A public apology may be requested.
-
-### 2. Warning
-
-**Community Impact**: A violation through a single incident or series of
-actions.
-
-**Consequence**: A warning with consequences for continued behavior. No
-interaction with the people involved, including unsolicited interaction with
-those enforcing the Code of Conduct, for a specified period of time. This
-includes avoiding interactions in community spaces as well as external channels
-like social media. Violating these terms may lead to a temporary or permanent
-ban.
-
-### 3. Temporary Ban
-
-**Community Impact**: A serious violation of community standards, including
-sustained inappropriate behavior.
-
-**Consequence**: A temporary ban from any sort of interaction or public
-communication with the community for a specified period of time. No public or
-private interaction with the people involved, including unsolicited interaction
-with those enforcing the Code of Conduct, is allowed during this period.
-Violating these terms may lead to a permanent ban.
-
-### 4. Permanent Ban
-
-**Community Impact**: Demonstrating a pattern of violation of community
-standards, including sustained inappropriate behavior, harassment of an
-individual, or aggression toward or disparagement of classes of individuals.
-
-**Consequence**: A permanent ban from any sort of public interaction within the
-community.
-
-## Attribution
-
-This Code of Conduct is adapted from the [Contributor Covenant][homepage],
-version 2.1, available at
-[https://www.contributor-covenant.org/version/2/1/code_of_conduct.html][v2.1].
-
-Community Impact Guidelines were inspired by
-[Mozilla's code of conduct enforcement ladder][mozilla coc].
-
-For answers to common questions about this code of conduct, see the FAQ at
-[https://www.contributor-covenant.org/faq][faq]. Translations are available at
-[https://www.contributor-covenant.org/translations][translations].
-
-[homepage]: https://www.contributor-covenant.org
-[v2.1]: https://www.contributor-covenant.org/version/2/1/code_of_conduct.html
-[mozilla coc]: https://github.com/mozilla/diversity
-[faq]: https://www.contributor-covenant.org/faq
-[translations]: https://www.contributor-covenant.org/translations
diff --git a/api/vendor/maennchen/zipstream-php/.github/CONTRIBUTING.md b/api/vendor/maennchen/zipstream-php/.github/CONTRIBUTING.md
deleted file mode 100644
index d8caee08..00000000
--- a/api/vendor/maennchen/zipstream-php/.github/CONTRIBUTING.md
+++ /dev/null
@@ -1,139 +0,0 @@
-# Contributing to ZipStream-PHP
-
-## Welcome!
-
-We look forward to your contributions! Here are some examples how you can
-contribute:
-
-- [Report a bug](https://github.com/maennchen/ZipStream-PHP/issues/new?labels=bug&template=BUG.md)
-- [Propose a new feature](https://github.com/maennchen/ZipStream-PHP/issues/new?labels=enhancement&template=FEATURE.md)
-- [Send a pull request](https://github.com/maennchen/ZipStream-PHP/pulls)
-
-## We have a Code of Conduct
-
-Please note that this project is released with a
-[Contributor Code of Conduct](CODE_OF_CONDUCT.md). By participating in this
-project you agree to abide by its terms.
-
-## Any contributions you make will be under the MIT License
-
-When you submit code changes, your submissions are understood to be under the
-same [MIT License](https://github.com/maennchen/ZipStream-PHP/blob/main/LICENSE)
-that covers the project. By contributing to this project, you agree that your
-contributions will be licensed under its MIT License.
-
-## Write bug reports with detail, background, and sample code
-
-In your bug report, please provide the following:
-
-- A quick summary and/or background
-- Steps to reproduce
- - Be specific!
- - Give sample code if you can.
-- What you expected would happen
-- What actually happens
-- Notes (possibly including why you think this might be happening, or stuff you
-- tried that didn't work)
-
-Please do not report a bug for a version of ZIPStream-PHP that is no longer
-supported (`< 3.0.0`). Please do not report a bug if you are using a version of
-PHP that is not supported by the version of ZipStream-PHP you are using.
-
-Please post code and output as text
-([using proper markup](https://guides.github.com/features/mastering-markdown/)).
-Do not post screenshots of code or output.
-
-Please include the output of `composer info | sort`.
-
-## Workflow for Pull Requests
-
-1. Fork the repository.
-2. Create your branch from `main` if you plan to implement new functionality or
- change existing code significantly; create your branch from the oldest branch
- that is affected by the bug if you plan to fix a bug.
-3. Implement your change and add tests for it.
-4. Ensure the test suite passes.
-5. Ensure the code complies with our coding guidelines (see below).
-6. Send that pull request!
-
-Please make sure you have
-[set up your user name and email address](https://git-scm.com/book/en/v2/Getting-Started-First-Time-Git-Setup)
-for use with Git. Strings such as `silly nick name ` look really
-stupid in the commit history of a project.
-
-We encourage you to
-[sign your Git commits with your GPG key](https://docs.github.com/en/github/authenticating-to-github/signing-commits).
-
-Pull requests for new features must be based on the `main` branch.
-
-We are trying to keep backwards compatibility breaks in ZipStream-PHP to a
-minimum. Please take this into account when proposing changes.
-
-Due to time constraints, we are not always able to respond as quickly as we
-would like. Please do not take delays personal and feel free to remind us if you
-feel that we forgot to respond.
-
-## Coding Guidelines
-
-This project comes with a configuration file (located at `/psalm.yml` in the
-repository) that you can use to perform static analysis (with a focus on type
-checking):
-
-```bash
-$ .composer run test:lint
-```
-
-This project comes with a configuration file (located at
-`/.php-cs-fixer.dist.php` in the repository) that you can use to (re)format your
-source code for compliance with this project's coding guidelines:
-
-```bash
-$ composer run format
-```
-
-Please understand that we will not accept a pull request when its changes
-violate this project's coding guidelines.
-
-## Using ZipStream-PHP from a Git checkout
-
-The following commands can be used to perform the initial checkout of
-ZipStream-PHP:
-
-```bash
-$ git clone git@github.com:maennchen/ZipStream-PHP.git
-
-$ cd ZipStream-PHP
-```
-
-Install ZipStream-PHP's dependencies using [Composer](https://getcomposer.org/):
-
-```bash
-$ composer install
-$ composer run install:tools # Install phpDocumentor using phive
-```
-
-## Running ZipStream-PHP's test suite
-
-After following the steps shown above, ZipStream-PHP's test suite is run like
-this:
-
-```bash
-$ composer run test:unit
-```
-
-There's some slow tests in the test suite that test the handling of big files in
-the archives. To skip them use the following command instead:
-
-```bash
-$ composer run test:unit:fast
-```
-
-## Generating ZipStream-PHP Documentation
-
-To generate the documentation for the library, run:
-
-```bash
-$ composer run docs:generate
-```
-
-The guide documentation pages can be found in the `/guides/` directory.
diff --git a/api/vendor/maennchen/zipstream-php/.github/FUNDING.yml b/api/vendor/maennchen/zipstream-php/.github/FUNDING.yml
deleted file mode 100644
index 5a461276..00000000
--- a/api/vendor/maennchen/zipstream-php/.github/FUNDING.yml
+++ /dev/null
@@ -1 +0,0 @@
-github: maennchen
diff --git a/api/vendor/maennchen/zipstream-php/.github/ISSUE_TEMPLATE/BUG.yml b/api/vendor/maennchen/zipstream-php/.github/ISSUE_TEMPLATE/BUG.yml
deleted file mode 100644
index 0eb8cc77..00000000
--- a/api/vendor/maennchen/zipstream-php/.github/ISSUE_TEMPLATE/BUG.yml
+++ /dev/null
@@ -1,71 +0,0 @@
-name: 🐞 Bug Report
-description: Something is broken?
-labels: ["bug"]
-body:
- - type: markdown
- attributes:
- value: |
- - Create a discussion instead if you are looking for support:
- https://github.com/maennchen/ZipStream-PHP/discussions
- - type: input
- id: version
- attributes:
- label: ZipStream-PHP version
- placeholder: x.y.z
- validations:
- required: true
- - type: input
- id: php-version
- attributes:
- label: PHP version
- placeholder: x.y.z
- validations:
- required: true
- - type: checkboxes
- id: constraints
- attributes:
- label: Constraints for Bug Report
- options:
- - label: |
- I'm using a version of ZipStream that is currently supported:
- https://github.com/maennchen/ZipStream-PHP#version-support
- required: true
- - label: |
- I'm using a version of PHP that has active support:
- https://www.php.net/supported-versions.php
- required: true
- - label: |
- I'm using a version of PHP that is compatible with your used
- ZipStream version.
- required: true
- - label: |
- I'm using the latest release of the used ZipStream major version.
- required: true
- - type: textarea
- id: summary
- attributes:
- label: Summary
- description: Provide a summary describing the problem you are experiencing.
- validations:
- required: true
- - type: textarea
- id: current-behaviour
- attributes:
- label: Current behavior
- description: What is the current (buggy) behavior?
- validations:
- required: true
- - type: textarea
- id: reproduction
- attributes:
- label: How to reproduce
- description: Provide steps to reproduce the bug.
- validations:
- required: true
- - type: textarea
- id: expected-behaviour
- attributes:
- label: Expected behavior
- description: What was the expected (correct) behavior?
- validations:
- required: true
diff --git a/api/vendor/maennchen/zipstream-php/.github/ISSUE_TEMPLATE/FEATURE.yml b/api/vendor/maennchen/zipstream-php/.github/ISSUE_TEMPLATE/FEATURE.yml
deleted file mode 100644
index e5dec637..00000000
--- a/api/vendor/maennchen/zipstream-php/.github/ISSUE_TEMPLATE/FEATURE.yml
+++ /dev/null
@@ -1,11 +0,0 @@
-name: 🎉 Feature Request
-description: You have a neat idea that should be implemented?
-labels: ["enhancement"]
-body:
- - type: textarea
- id: description
- attributes:
- label: Description
- description: Provide a summary of the feature you would like to see implemented.
- validations:
- required: true
diff --git a/api/vendor/maennchen/zipstream-php/.github/PULL_REQUEST_TEMPLATE.md b/api/vendor/maennchen/zipstream-php/.github/PULL_REQUEST_TEMPLATE.md
deleted file mode 100644
index 6892c571..00000000
--- a/api/vendor/maennchen/zipstream-php/.github/PULL_REQUEST_TEMPLATE.md
+++ /dev/null
@@ -1,6 +0,0 @@
-Please go the the `Preview` tab and select the appropriate sub-template:
-
-* [🐞 Failing Test](?expand=1&template=FAILING_TEST.md)
-* [🐞 Bug Fix](?expand=1&template=FIX.md)
-* [⚙ Improvement](?expand=1&template=IMPROVEMENT.md)
-* [🎉 New Feature](?expand=1&template=NEW_FEATURE.md)
diff --git a/api/vendor/maennchen/zipstream-php/.github/PULL_REQUEST_TEMPLATE/FAILING_TEST.md b/api/vendor/maennchen/zipstream-php/.github/PULL_REQUEST_TEMPLATE/FAILING_TEST.md
deleted file mode 100644
index 24603cb6..00000000
--- a/api/vendor/maennchen/zipstream-php/.github/PULL_REQUEST_TEMPLATE/FAILING_TEST.md
+++ /dev/null
@@ -1,13 +0,0 @@
-
-
-
diff --git a/api/vendor/maennchen/zipstream-php/.github/PULL_REQUEST_TEMPLATE/FIX.md b/api/vendor/maennchen/zipstream-php/.github/PULL_REQUEST_TEMPLATE/FIX.md
deleted file mode 100644
index 77f65a08..00000000
--- a/api/vendor/maennchen/zipstream-php/.github/PULL_REQUEST_TEMPLATE/FIX.md
+++ /dev/null
@@ -1,13 +0,0 @@
-
-
-
diff --git a/api/vendor/maennchen/zipstream-php/.github/PULL_REQUEST_TEMPLATE/IMPROVEMENT.md b/api/vendor/maennchen/zipstream-php/.github/PULL_REQUEST_TEMPLATE/IMPROVEMENT.md
deleted file mode 100644
index 3ac8e310..00000000
--- a/api/vendor/maennchen/zipstream-php/.github/PULL_REQUEST_TEMPLATE/IMPROVEMENT.md
+++ /dev/null
@@ -1,9 +0,0 @@
-
-
-
diff --git a/api/vendor/maennchen/zipstream-php/.github/PULL_REQUEST_TEMPLATE/NEW_FEATURE.md b/api/vendor/maennchen/zipstream-php/.github/PULL_REQUEST_TEMPLATE/NEW_FEATURE.md
deleted file mode 100644
index ca53939c..00000000
--- a/api/vendor/maennchen/zipstream-php/.github/PULL_REQUEST_TEMPLATE/NEW_FEATURE.md
+++ /dev/null
@@ -1,9 +0,0 @@
-
-
-
diff --git a/api/vendor/maennchen/zipstream-php/.github/SECURITY.md b/api/vendor/maennchen/zipstream-php/.github/SECURITY.md
deleted file mode 100644
index 3046c310..00000000
--- a/api/vendor/maennchen/zipstream-php/.github/SECURITY.md
+++ /dev/null
@@ -1,22 +0,0 @@
-# Security Policy
-
-[](https://github.com/ossf/oss-vulnerability-guide/blob/main/finder-guide.md)
-[](https://github.com/maennchen/ZipStream-PHP/security/advisories/new)
-[](mailto:jonatan@maennchen.ch)
-
-This repository follows the
-[OpenSSF Vulnerability Disclosure guide](https://github.com/ossf/oss-vulnerability-guide/tree/main).
-You can learn more about it in the
-[Finders Guide](https://github.com/ossf/oss-vulnerability-guide/blob/main/finder-guide.md).
-
-Please report vulnerabilities via the
-[GitHub Security Vulnerability Reporting](https://github.com/maennchen/ZipStream-PHP/security/advisories/new)
-or via email to [`jonatan@maennchen.ch`](mailto:jonatan@maennchen.ch) if this does
-not work for you.
-
-Our vulnerability management team will respond within 3 working days of your
-report. If the issue is confirmed as a vulnerability, we will open a Security
-Advisory. This project follows a 90 day disclosure timeline.
-
-If you have questions about reporting security issues, email the vulnerability
-management team: [`jonatan@maennchen.ch`](mailto:jonatan@maennchen.ch)
diff --git a/api/vendor/maennchen/zipstream-php/.github/dependabot.yml b/api/vendor/maennchen/zipstream-php/.github/dependabot.yml
deleted file mode 100644
index 9d20742e..00000000
--- a/api/vendor/maennchen/zipstream-php/.github/dependabot.yml
+++ /dev/null
@@ -1,15 +0,0 @@
-version: 2
-updates:
- - package-ecosystem: "composer"
- directory: "/"
- schedule:
- interval: "daily"
- - package-ecosystem: "github-actions"
- directory: "/"
- schedule:
- interval: "weekly"
- groups:
- github-actions:
- applies-to: version-updates
- patterns:
- - "*"
diff --git a/api/vendor/maennchen/zipstream-php/.github/scorecard.yml b/api/vendor/maennchen/zipstream-php/.github/scorecard.yml
deleted file mode 100644
index 219fc0bf..00000000
--- a/api/vendor/maennchen/zipstream-php/.github/scorecard.yml
+++ /dev/null
@@ -1,14 +0,0 @@
-annotations:
- - checks:
- - fuzzing
- reasons:
- - reason: not-applicable # PHP is memory safe
- - checks:
- - packaging
- reasons:
- - reason: not-supported # Using Composer
- - checks:
- - signed-releases
- reasons:
- - reason: not-applicable # Releases are distributed via Composer
-
diff --git a/api/vendor/maennchen/zipstream-php/.github/workflows/branch_main.yml b/api/vendor/maennchen/zipstream-php/.github/workflows/branch_main.yml
deleted file mode 100644
index 15ff2782..00000000
--- a/api/vendor/maennchen/zipstream-php/.github/workflows/branch_main.yml
+++ /dev/null
@@ -1,24 +0,0 @@
-on:
- push:
- branches:
- - "main"
-
-name: "Main Branch"
-
-permissions:
- contents: read
-
-jobs:
- test:
- name: "Test"
-
- permissions:
- contents: read
- security-events: write
-
- uses: ./.github/workflows/part_test.yml
-
- docs:
- name: "Docs"
-
- uses: ./.github/workflows/part_docs.yml
diff --git a/api/vendor/maennchen/zipstream-php/.github/workflows/part_dependabot.yml b/api/vendor/maennchen/zipstream-php/.github/workflows/part_dependabot.yml
deleted file mode 100644
index 20a13a20..00000000
--- a/api/vendor/maennchen/zipstream-php/.github/workflows/part_dependabot.yml
+++ /dev/null
@@ -1,30 +0,0 @@
-on:
- workflow_call: {}
-
-name: "Dependabot"
-
-permissions:
- contents: read
-
-jobs:
- automerge_dependabot:
- name: "Automerge PRs"
-
- runs-on: ubuntu-latest
-
- permissions:
- pull-requests: write
- contents: write
-
- steps:
- - name: Harden Runner
- uses: step-security/harden-runner@cb605e52c26070c328afc4562f0b4ada7618a84e # v2.10.4
- with:
- egress-policy: audit
-
- - uses: fastify/github-action-merge-dependabot@c3bde0759d4f24db16f7b250b2122bc2df57e817 # v3.11.0
- with:
- github-token: ${{ github.token }}
- use-github-auto-merge: true
- # Major Updates need to be merged manually
- target: minor
diff --git a/api/vendor/maennchen/zipstream-php/.github/workflows/part_docs.yml b/api/vendor/maennchen/zipstream-php/.github/workflows/part_docs.yml
deleted file mode 100644
index 9b779eb5..00000000
--- a/api/vendor/maennchen/zipstream-php/.github/workflows/part_docs.yml
+++ /dev/null
@@ -1,51 +0,0 @@
-on:
- workflow_call: {}
-
-name: "Documentation"
-
-permissions:
- contents: read
-
-jobs:
- generate:
- name: "Generate"
-
- runs-on: ubuntu-latest
-
- steps:
- - name: Harden Runner
- uses: step-security/harden-runner@cb605e52c26070c328afc4562f0b4ada7618a84e # v2.10.4
- with:
- egress-policy: audit
-
- - name: Checkout Code
- uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2
- - name: SetUp PHP
- id: setup-php
- uses: shivammathur/setup-php@9e72090525849c5e82e596468b86eb55e9cc5401 # v2
- with:
- php-version: "8.3"
- tools: phive
- - name: Cache Tools
- uses: actions/cache@1bd1e32a3bdc45362d1e726936510720a7c30a57 # v4.2.0
- id: cache
- with:
- path: ~/.phive
- key: tools-${{ runner.os }}-${{ steps.setup-php.outputs.php-version }}-${{ hashFiles('**/phars.xml') }}
- restore-keys: |
- tools-${{ runner.os }}-${{ steps.setup-php.outputs.php-version }}-
- tools-${{ steps.setup-php.outputs.php-version }}-
- tools-
- - name: Install Tools
- run: composer run install:tools
- - name: Generate Docs
- run: composer run docs:generate
- - uses: actions/upload-artifact@65c4c4a1ddee5b72f698fdd19549f0f0fb45cf08 # v4.6.0
- with:
- name: docs
- path: docs
- - name: Package for GitHub Pages
- uses: actions/upload-pages-artifact@56afc609e74202658d3ffba0e8f6dda462b719fa # v3.0.1
- with:
- path: docs
-
diff --git a/api/vendor/maennchen/zipstream-php/.github/workflows/part_release.yml b/api/vendor/maennchen/zipstream-php/.github/workflows/part_release.yml
deleted file mode 100644
index 112d72a4..00000000
--- a/api/vendor/maennchen/zipstream-php/.github/workflows/part_release.yml
+++ /dev/null
@@ -1,94 +0,0 @@
-on:
- workflow_call:
- inputs:
- releaseName:
- required: true
- type: string
- stable:
- required: false
- type: boolean
- default: false
-
-name: "Release"
-
-permissions:
- contents: read
-
-jobs:
- create:
- name: Create Release
-
- runs-on: ubuntu-latest
-
- permissions:
- contents: write
-
- steps:
- - name: Harden Runner
- uses: step-security/harden-runner@cb605e52c26070c328afc4562f0b4ada7618a84e # v2.10.4
- with:
- egress-policy: audit
-
- - name: Create prerelease
- if: ${{ !inputs.stable }}
- env:
- GITHUB_TOKEN: ${{ github.token }}
- run: |
- gh release create \
- --repo ${{ github.repository }} \
- --title ${{ inputs.releaseName }} \
- --prerelease \
- --generate-notes \
- ${{ inputs.releaseName }}
-
- - name: Create release
- if: ${{ inputs.stable }}
- env:
- GITHUB_TOKEN: ${{ github.token }}
- run: |
- gh release create \
- --repo ${{ github.repository }} \
- --title ${{ inputs.releaseName }} \
- --generate-notes \
- ${{ inputs.releaseName }}
-
- upload_release:
- name: "Upload"
-
- needs: ["create"]
-
- runs-on: ubuntu-latest
-
- permissions:
- id-token: write
- contents: write
- attestations: write
-
- steps:
- - name: Harden Runner
- uses: step-security/harden-runner@cb605e52c26070c328afc4562f0b4ada7618a84e # v2.10.4
- with:
- egress-policy: audit
-
- - uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2
- - uses: actions/download-artifact@fa0a91b85d4f404e444e00e005971372dc801d16 # v4.1.8
- with:
- name: docs
- path: docs
- - run: |
- tar -czvf docs.tar.gz docs
- - name: "Attest Documentation"
- id: attestation
- uses: actions/attest-build-provenance@520d128f165991a6c774bcb264f323e3d70747f4 # v2.2.0
- with:
- subject-path: "docs.tar.gz"
- - name: Copy Attestation
- run: cp "$ATTESTATION" docs.tar.gz.sigstore
- env:
- ATTESTATION: "${{ steps.attestation.outputs.bundle-path }}"
- - name: Upload
- env:
- GITHUB_TOKEN: ${{ github.token }}
- run: |
- gh release upload --clobber "${{ github.ref_name }}" \
- docs.tar.gz docs.tar.gz.sigstore
diff --git a/api/vendor/maennchen/zipstream-php/.github/workflows/part_test.yml b/api/vendor/maennchen/zipstream-php/.github/workflows/part_test.yml
deleted file mode 100644
index d4f8180a..00000000
--- a/api/vendor/maennchen/zipstream-php/.github/workflows/part_test.yml
+++ /dev/null
@@ -1,181 +0,0 @@
-on:
- workflow_call:
-
-name: "Test"
-
-permissions:
- contents: read
-
-jobs:
- phpunit:
- name: PHPUnit (PHP ${{ matrix.php }} on ${{ matrix.os }})
-
- runs-on: ${{ matrix.os }}
-
- continue-on-error: ${{ matrix.experimental }}
-
- strategy:
- fail-fast: false
- matrix:
- php: ["8.2", "8.3", "8.4"]
- os: [ubuntu-latest]
- experimental: [false]
- include:
- - php: nightly
- os: ubuntu-latest
- experimental: true
- - php: "8.4"
- os: windows-latest
- experimental: false
- - php: "8.4"
- os: macos-latest
- experimental: false
-
- steps:
- - name: Harden Runner
- uses: step-security/harden-runner@cb605e52c26070c328afc4562f0b4ada7618a84e # v2.10.4
- with:
- egress-policy: audit
-
- - name: Checkout Code
- uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2
- - name: SetUp PHP
- id: setup-php
- uses: shivammathur/setup-php@9e72090525849c5e82e596468b86eb55e9cc5401 # v2
- with:
- php-version: "${{ matrix.php }}"
- tools: phpunit
- coverage: xdebug
- extensions: xdebug,zip
- - name: Get composer cache directory
- id: composer-cache-common
- if: "${{ runner.os != 'Windows' }}"
- run: echo "dir=$(composer config cache-files-dir)" >> $GITHUB_OUTPUT
- - name: Get composer cache directory
- id: composer-cache-windows
- if: "${{ runner.os == 'Windows' }}"
- run: echo "dir=$(composer config cache-files-dir)" >> $env:GITHUB_OUTPUT
- - name: Cache Deps
- uses: actions/cache@1bd1e32a3bdc45362d1e726936510720a7c30a57 # v4.2.0
- id: cache
- with:
- path: ${{ steps.composer-cache-common.outputs.dir }}${{ steps.composer-cache-windows.outputs.dir }}
- key: deps-${{ runner.os }}-${{ steps.setup-php.outputs.php-version }}-composer-${{ hashFiles('**/composer.lock') }}
- restore-keys: |
- deps-${{ runner.os }}-${{ steps.setup-php.outputs.php-version }}-composer-
- deps-${{ runner.os }}-${{ steps.setup-php.outputs.php-version }}-
- deps-${{ steps.setup-php.outputs.php-version }}-
- deps-
- - name: Install Deps
- if: matrix.php != 'nightly'
- run: composer install --prefer-dist
- - name: Install Deps (ignore PHP requirement)
- if: matrix.php == 'nightly'
- run: composer install --prefer-dist --ignore-platform-req=php+
- - name: Run PHPUnit
- run: composer run test:unit:cov
- - name: Upload coverage results to Coveralls
- env:
- COVERALLS_REPO_TOKEN: ${{ secrets.GITHUB_TOKEN }}
- COVERALLS_PARALLEL: true
- COVERALLS_FLAG_NAME: ${{ runner.os }}-${{ steps.setup-php.outputs.php-version }}
- run: composer run coverage:report
- continue-on-error: ${{ matrix.experimental }}
-
- mark_coverage_done:
- needs: ["phpunit"]
-
- runs-on: ubuntu-latest
-
- steps:
- - name: Harden Runner
- uses: step-security/harden-runner@cb605e52c26070c328afc4562f0b4ada7618a84e # v2.10.4
- with:
- egress-policy: audit
-
- - name: Coveralls Finished
- uses: coverallsapp/github-action@648a8eb78e6d50909eff900e4ec85cab4524a45b # v2.3.6
- with:
- github-token: ${{ secrets.github_token }}
- parallel-finished: true
-
- psalm:
- name: Run Psalm
-
- runs-on: "ubuntu-latest"
-
- permissions:
- security-events: write
-
- steps:
- - name: Harden Runner
- uses: step-security/harden-runner@cb605e52c26070c328afc4562f0b4ada7618a84e # v2.10.4
- with:
- egress-policy: audit
-
- - name: Checkout Code
- uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2
- - name: SetUp PHP
- id: setup-php
- uses: shivammathur/setup-php@9e72090525849c5e82e596468b86eb55e9cc5401 # v2
- with:
- php-version: "8.3"
- - name: Get composer cache directory
- id: composer-cache
- run: echo "dir=$(composer config cache-files-dir)" >> $GITHUB_OUTPUT
- - name: Cache Deps
- uses: actions/cache@1bd1e32a3bdc45362d1e726936510720a7c30a57 # v4.2.0
- id: cache
- with:
- path: ${{ steps.composer-cache.outputs.dir }}
- key: deps-${{ runner.os }}-${{ steps.setup-php.outputs.php-version }}-composer-${{ hashFiles('**/composer.lock') }}
- restore-keys: |
- deps-${{ runner.os }}-${{ steps.setup-php.outputs.php-version }}-composer-
- deps-${{ runner.os }}-${{ steps.setup-php.outputs.php-version }}-
- deps-${{ steps.setup-php.outputs.php-version }}-
- deps-
- - name: Install Deps
- run: composer install --prefer-dist
- - name: Run Psalm
- run: composer run test:lint -- --report=results.sarif
- - name: "Upload SARIF"
- uses: github/codeql-action/upload-sarif@f6091c0113d1dcf9b98e269ee48e8a7e51b7bdd4 # v3
- with:
- sarif_file: results.sarif
-
- php-cs:
- name: Run PHP-CS
-
- runs-on: "ubuntu-latest"
-
- steps:
- - name: Harden Runner
- uses: step-security/harden-runner@cb605e52c26070c328afc4562f0b4ada7618a84e # v2.10.4
- with:
- egress-policy: audit
-
- - name: Checkout Code
- uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2
- - name: SetUp PHP
- id: setup-php
- uses: shivammathur/setup-php@9e72090525849c5e82e596468b86eb55e9cc5401 # v2
- with:
- php-version: "8.3"
- - name: Get composer cache directory
- id: composer-cache
- run: echo "dir=$(composer config cache-files-dir)" >> $GITHUB_OUTPUT
- - name: Cache Deps
- uses: actions/cache@1bd1e32a3bdc45362d1e726936510720a7c30a57 # v4.2.0
- id: cache
- with:
- path: ${{ steps.composer-cache.outputs.dir }}
- key: deps-${{ runner.os }}-${{ steps.setup-php.outputs.php-version }}-composer-${{ hashFiles('**/composer.lock') }}
- restore-keys: |
- deps-${{ runner.os }}-${{ steps.setup-php.outputs.php-version }}-composer-
- deps-${{ runner.os }}-${{ steps.setup-php.outputs.php-version }}-
- deps-${{ steps.setup-php.outputs.php-version }}-
- deps-
- - name: Install Deps
- run: composer install --prefer-dist
- - name: Run PHP-CS
- run: composer run test:formatted
diff --git a/api/vendor/maennchen/zipstream-php/.github/workflows/pr.yml b/api/vendor/maennchen/zipstream-php/.github/workflows/pr.yml
deleted file mode 100644
index d21f3986..00000000
--- a/api/vendor/maennchen/zipstream-php/.github/workflows/pr.yml
+++ /dev/null
@@ -1,50 +0,0 @@
-on:
- pull_request:
- branches:
- - "*"
- workflow_dispatch: {}
-
-name: "Pull Request"
-
-permissions:
- contents: read
-
-jobs:
- test:
- name: "Test"
-
- permissions:
- contents: read
- security-events: write
-
- uses: ./.github/workflows/part_test.yml
-
- docs:
- name: "Docs"
-
- uses: ./.github/workflows/part_docs.yml
-
- dependabot:
- name: "Dependabot"
-
- if: ${{ github.actor == 'dependabot[bot]'}}
-
- permissions:
- pull-requests: write
- contents: write
-
- uses: ./.github/workflows/part_dependabot.yml
-
- dependency-review:
- name: Dependency Review
- runs-on: ubuntu-latest
- steps:
- - name: Harden Runner
- uses: step-security/harden-runner@cb605e52c26070c328afc4562f0b4ada7618a84e # v2.10.4
- with:
- egress-policy: audit
-
- - name: 'Checkout Repository'
- uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2
- - name: 'Dependency Review'
- uses: actions/dependency-review-action@3b139cfc5fae8b618d3eae3675e383bb1769c019 # v4.5.0
diff --git a/api/vendor/maennchen/zipstream-php/.github/workflows/scorecard.yml b/api/vendor/maennchen/zipstream-php/.github/workflows/scorecard.yml
deleted file mode 100644
index c1d08a21..00000000
--- a/api/vendor/maennchen/zipstream-php/.github/workflows/scorecard.yml
+++ /dev/null
@@ -1,78 +0,0 @@
-# This workflow uses actions that are not certified by GitHub. They are provided
-# by a third-party and are governed by separate terms of service, privacy
-# policy, and support documentation.
-
-name: Scorecard supply-chain security
-on:
- # For Branch-Protection check. Only the default branch is supported. See
- # https://github.com/ossf/scorecard/blob/main/docs/checks.md#branch-protection
- branch_protection_rule:
- # To guarantee Maintained check is occasionally updated. See
- # https://github.com/ossf/scorecard/blob/main/docs/checks.md#maintained
- schedule:
- - cron: '28 11 * * 3'
- push:
- branches: [ "main" ]
-
-# Declare default permissions as read only.
-permissions: read-all
-
-jobs:
- analysis:
- name: Scorecard analysis
- runs-on: ubuntu-latest
- permissions:
- # Needed to upload the results to code-scanning dashboard.
- security-events: write
- # Needed to publish results and get a badge (see publish_results below).
- id-token: write
- # Uncomment the permissions below if installing in a private repository.
- # contents: read
- # actions: read
-
- steps:
- - name: Harden Runner
- uses: step-security/harden-runner@cb605e52c26070c328afc4562f0b4ada7618a84e # v2.10.4
- with:
- egress-policy: audit
-
- - name: "Checkout code"
- uses: actions/checkout@11bd71901bbe5b1630ceea73d27597364c9af683 # v4.2.2
- with:
- persist-credentials: false
-
- - name: "Run analysis"
- uses: ossf/scorecard-action@62b2cac7ed8198b15735ed49ab1e5cf35480ba46 # v2.4.0
- with:
- results_file: results.sarif
- results_format: sarif
- # (Optional) "write" PAT token. Uncomment the `repo_token` line below if:
- # - you want to enable the Branch-Protection check on a *public* repository, or
- # - you are installing Scorecard on a *private* repository
- # To create the PAT, follow the steps in https://github.com/ossf/scorecard-action?tab=readme-ov-file#authentication-with-fine-grained-pat-optional.
- # repo_token: ${{ secrets.SCORECARD_TOKEN }}
-
- # Public repositories:
- # - Publish results to OpenSSF REST API for easy access by consumers
- # - Allows the repository to include the Scorecard badge.
- # - See https://github.com/ossf/scorecard-action#publishing-results.
- # For private repositories:
- # - `publish_results` will always be set to `false`, regardless
- # of the value entered here.
- publish_results: true
-
- # Upload the results as artifacts (optional). Commenting out will disable uploads of run results in SARIF
- # format to the repository Actions tab.
- - name: "Upload artifact"
- uses: actions/upload-artifact@65c4c4a1ddee5b72f698fdd19549f0f0fb45cf08 # v4.6.0
- with:
- name: SARIF file
- path: results.sarif
- retention-days: 5
-
- # Upload the results to GitHub's code scanning dashboard (optional).
- # Commenting out will disable upload of results to your repo's Code Scanning dashboard
- - name: "Upload to code-scanning"
- uses: github/codeql-action/upload-sarif@f6091c0113d1dcf9b98e269ee48e8a7e51b7bdd4 # v3.28.5
- with:
- sarif_file: results.sarif
diff --git a/api/vendor/maennchen/zipstream-php/.github/workflows/tag-beta.yml b/api/vendor/maennchen/zipstream-php/.github/workflows/tag-beta.yml
deleted file mode 100644
index b3399454..00000000
--- a/api/vendor/maennchen/zipstream-php/.github/workflows/tag-beta.yml
+++ /dev/null
@@ -1,29 +0,0 @@
-on:
- push:
- tags:
- - "[0-9]+.[0-9]+.[0-9]+-beta.[0-9]+"
-
-name: "Beta Tag"
-
-permissions:
- contents: read
-
-jobs:
- docs:
- name: "Docs"
-
- uses: ./.github/workflows/part_docs.yml
-
- release:
- name: "Release"
-
- needs: ["docs"]
-
- permissions:
- id-token: write
- contents: write
- attestations: write
-
- uses: ./.github/workflows/part_release.yml
- with:
- releaseName: "${{ github.ref_name }}"
diff --git a/api/vendor/maennchen/zipstream-php/.github/workflows/tag-stable.yml b/api/vendor/maennchen/zipstream-php/.github/workflows/tag-stable.yml
deleted file mode 100644
index dfc14383..00000000
--- a/api/vendor/maennchen/zipstream-php/.github/workflows/tag-stable.yml
+++ /dev/null
@@ -1,55 +0,0 @@
-on:
- push:
- tags:
- - "[0-9]+.[0-9]+.[0-9]+"
-
-name: "Stable Tag"
-
-permissions:
- contents: read
-
-jobs:
- docs:
- name: "Docs"
-
- uses: ./.github/workflows/part_docs.yml
-
- release:
- name: "Release"
-
- needs: ["docs"]
-
- permissions:
- id-token: write
- contents: write
- attestations: write
-
- uses: ./.github/workflows/part_release.yml
- with:
- releaseName: "${{ github.ref_name }}"
- stable: true
-
- deploy_pages:
- name: "Deploy to GitHub Pages"
-
- needs: ["release", "docs"]
-
- runs-on: ubuntu-latest
-
- permissions:
- pages: write
- id-token: write
-
- environment:
- name: github-pages
- url: ${{ steps.deployment.outputs.page_url }}
-
- steps:
- - name: Harden Runner
- uses: step-security/harden-runner@cb605e52c26070c328afc4562f0b4ada7618a84e # v2.10.4
- with:
- egress-policy: audit
-
- - name: Deploy to GitHub Pages
- id: deployment
- uses: actions/deploy-pages@d6db90164ac5ed86f2b6aed7e0febac5b3c0c03e # v4.0.5
diff --git a/api/vendor/maennchen/zipstream-php/.gitignore b/api/vendor/maennchen/zipstream-php/.gitignore
deleted file mode 100644
index e52a4987..00000000
--- a/api/vendor/maennchen/zipstream-php/.gitignore
+++ /dev/null
@@ -1,12 +0,0 @@
-/composer.lock
-/cov
-/coverage.clover.xml
-/docs
-.idea
-/.php-cs-fixer.cache
-/.phpdoc/cache
-/.phpunit.result.cache
-/phpunit.xml
-/.phpunit.cache
-/tools
-/vendor
diff --git a/api/vendor/maennchen/zipstream-php/.phive/phars.xml b/api/vendor/maennchen/zipstream-php/.phive/phars.xml
index c958402b..10f0df2d 100644
--- a/api/vendor/maennchen/zipstream-php/.phive/phars.xml
+++ b/api/vendor/maennchen/zipstream-php/.phive/phars.xml
@@ -1,4 +1,4 @@
-
+
diff --git a/api/vendor/maennchen/zipstream-php/.php-cs-fixer.dist.php b/api/vendor/maennchen/zipstream-php/.php-cs-fixer.dist.php
index 9d47c384..31c8f0ce 100644
--- a/api/vendor/maennchen/zipstream-php/.php-cs-fixer.dist.php
+++ b/api/vendor/maennchen/zipstream-php/.php-cs-fixer.dist.php
@@ -28,7 +28,8 @@ return $config->setRules([
'@PER' => true,
'@PER:risky' => true,
'@PHP83Migration' => true,
- '@PHP84Migration' => true,
+ // Enable once PHP 8.4 is the minimum version
+ // '@PHP84Migration' => true,
'@PHPUnit84Migration:risky' => true,
'array_syntax' => ['syntax' => 'short'],
'class_attributes_separation' => true,
diff --git a/api/vendor/maennchen/zipstream-php/.tool-versions b/api/vendor/maennchen/zipstream-php/.tool-versions
index 150c1ee4..abc068cc 100644
--- a/api/vendor/maennchen/zipstream-php/.tool-versions
+++ b/api/vendor/maennchen/zipstream-php/.tool-versions
@@ -1 +1 @@
-php 8.4.3
+php 8.4.10
diff --git a/api/vendor/maennchen/zipstream-php/README.md b/api/vendor/maennchen/zipstream-php/README.md
index 1e6d6798..3e527f0a 100644
--- a/api/vendor/maennchen/zipstream-php/README.md
+++ b/api/vendor/maennchen/zipstream-php/README.md
@@ -64,6 +64,50 @@ $zip->addFileFromPath(
$zip->finish();
```
+### Callback Output
+
+You can stream ZIP data to a custom callback function instead of directly to the browser:
+
+```php
+use ZipStream\ZipStream;
+use ZipStream\Stream\CallbackStreamWrapper;
+
+// Stream to a callback function with proper file handling
+$outputFile = fopen('output.zip', 'wb');
+$backupFile = fopen('backup.zip', 'wb');
+
+$zip = new ZipStream(
+ outputStream: CallbackStreamWrapper::open(function (string $data) use ($outputFile, $backupFile) {
+ // Handle ZIP data as it's generated
+ fwrite($outputFile, $data);
+
+ // Send to multiple destinations efficiently
+ echo $data; // Browser
+ fwrite($backupFile, $data); // Backup file
+ }),
+ sendHttpHeaders: false,
+);
+
+$zip->addFile('hello.txt', 'Hello World!');
+$zip->finish();
+
+// Clean up resources
+fclose($outputFile);
+fclose($backupFile);
+```
+
+## Questions
+
+**💬 Questions? Please Read This First!**
+
+If you have a question about using this library, please *do not email the
+authors directly*. Instead, head over to the
+[GitHub Discussions](https://github.com/maennchen/ZipStream-PHP/discussions)
+page — your question might already be answered there! Using Discussions helps
+build a shared knowledge base, so others can also benefit from the answers. If
+you need dedicated 1:1 support, check out the options available on
+[@maennchen's sponsorship page](https://github.com/sponsors/maennchen?frequency=one-time&sponsor=maennchen).
+
## Upgrade to version 3.1.2
- Minimum PHP Version: `8.2`
@@ -76,7 +120,7 @@ $zip->finish();
- Only 64bit Architecture is supported.
- The class `ZipStream\Option\Method` has been replaced with the enum
`ZipStream\CompressionMethod`.
-- Most clases have been flagged as `@internal` and should not be used from the
+- Most classes have been flagged as `@internal` and should not be used from the
outside.
If you're using internal resources to extend this library, please open an
issue so that a clean interface can be added & published.
@@ -88,7 +132,7 @@ $zip->finish();
### Archive Options
- The class `ZipStream\Option\Archive` has been replaced in favor of named
- arguments in the `ZipStream\ZipStream` constuctor.
+ arguments in the `ZipStream\ZipStream` constructor.
- The archive options `largeFileSize` & `largeFileMethod` has been removed. If
you want different `compressionMethods` based on the file size, you'll have to
implement this yourself.
@@ -101,7 +145,7 @@ $zip->finish();
filesizes this way.
- The archive option `deflateLevel` has been replaced with the option
`defaultDeflateLevel` and can be overridden for every file.
-- The first argument (`name`) of the `ZipStream\ZipStream` constuctor has been
+- The first argument (`name`) of the `ZipStream\ZipStream` constructor has been
replaced with the named argument `outputName`.
- Headers are now also sent if the `outputName` is empty. If you do not want to
automatically send http headers, set `sendHttpHeaders` to `false`.
diff --git a/api/vendor/maennchen/zipstream-php/composer.json b/api/vendor/maennchen/zipstream-php/composer.json
index 6ecd503a..529c2445 100644
--- a/api/vendor/maennchen/zipstream-php/composer.json
+++ b/api/vendor/maennchen/zipstream-php/composer.json
@@ -22,12 +22,12 @@
}
],
"require": {
- "php-64bit": "^8.2",
+ "php-64bit": "^8.3",
"ext-mbstring": "*",
"ext-zlib": "*"
},
"require-dev": {
- "phpunit/phpunit": "^11.0",
+ "phpunit/phpunit": "^12.0",
"guzzlehttp/guzzle": "^7.5",
"ext-zip": "*",
"mikey179/vfsstream": "^1.6",
@@ -57,7 +57,7 @@
"test:formatted": "@format --dry-run --stop-on-violation --using-cache=no",
"test:lint": "psalm --stats --show-info=true --find-unused-psalm-suppress",
"coverage:report": "php-coveralls --coverage_clover=coverage.clover.xml --json_path=coveralls-upload.json --insecure",
- "install:tools": "phive install --trust-gpg-keys 0x67F861C3D889C656 --trust-gpg-keys 0x8AC0BAA79732DD42",
+ "install:tools": "phive install --trust-gpg-keys 0x67F861C3D889C656 --trust-gpg-keys 0x8AC0BAA79732DD42 --trust-gpg-keys 0x6DA3ACC4991FFAE5",
"docs:generate": "tools/phpdocumentor --sourcecode"
},
"autoload": {
diff --git a/api/vendor/maennchen/zipstream-php/guides/ContentLength.rst b/api/vendor/maennchen/zipstream-php/guides/ContentLength.rst
index 21fea34d..3b75dd76 100644
--- a/api/vendor/maennchen/zipstream-php/guides/ContentLength.rst
+++ b/api/vendor/maennchen/zipstream-php/guides/ContentLength.rst
@@ -7,7 +7,7 @@ using the options ``SIMULATION_STRICT`` or ``SIMULATION_LAX`` in the
In the ``SIMULATION_STRICT`` mode, ``ZipStream`` will not allow to calculate the
size based on reading the whole file. ``SIMULATION_LAX`` will read the whole
-file if neccessary.
+file if necessary.
``SIMULATION_STRICT`` is therefore useful to make sure that the size can be
calculated efficiently.
diff --git a/api/vendor/maennchen/zipstream-php/guides/Options.rst b/api/vendor/maennchen/zipstream-php/guides/Options.rst
index 5e92e94d..373afef2 100644
--- a/api/vendor/maennchen/zipstream-php/guides/Options.rst
+++ b/api/vendor/maennchen/zipstream-php/guides/Options.rst
@@ -12,11 +12,14 @@ Here is the full list of options available to you. You can also have a look at
$zip = new ZipStream(
// Define output stream
- // (argument is eiter a resource or implementing
+ // (argument is either a resource or implementing
// `Psr\Http\Message\StreamInterface`)
//
// Setup with `psr/http-message` & `guzzlehttp/psr7` dependencies
// required when using `Psr\Http\Message\StreamInterface`.
+ //
+ // Can also use CallbackStreamWrapper for custom output handling:
+ // outputStream: CallbackStreamWrapper::open(function($data) { /* handle data */ }),
outputStream: $filePointer,
// Set the deflate level (default is 6; use -1 to disable it)
diff --git a/api/vendor/maennchen/zipstream-php/guides/StreamOutput.rst b/api/vendor/maennchen/zipstream-php/guides/StreamOutput.rst
index 9f3165b7..60593f9c 100644
--- a/api/vendor/maennchen/zipstream-php/guides/StreamOutput.rst
+++ b/api/vendor/maennchen/zipstream-php/guides/StreamOutput.rst
@@ -37,3 +37,77 @@ Stream to S3 Bucket
$zip->finish();
fclose($zipFile);
+
+Stream to Callback Function
+---------------------------
+
+The CallbackStreamWrapper allows you to stream ZIP data to a custom callback function,
+enabling flexible output handling such as streaming to multiple destinations,
+progress tracking, or data transformation.
+
+.. code-block:: php
+
+ use ZipStream\ZipStream;
+ use ZipStream\Stream\CallbackStreamWrapper;
+
+ // Example 1: Stream to multiple destinations with proper file handling
+ $backupFile = fopen('backup.zip', 'wb');
+ $logFile = fopen('transfer.log', 'ab');
+
+ $zip = new ZipStream(
+ outputStream: CallbackStreamWrapper::open(function (string $data) use ($backupFile, $logFile) {
+ // Send to browser
+ echo $data;
+
+ // Save to file efficiently
+ fwrite($backupFile, $data);
+
+ // Log transfer progress
+ fwrite($logFile, "Transferred " . strlen($data) . " bytes\n");
+ }),
+ sendHttpHeaders: false,
+ );
+
+ $zip->addFile('hello.txt', 'Hello World!');
+ $zip->finish();
+
+ // Clean up resources
+ fclose($backupFile);
+ fclose($logFile);
+
+.. code-block:: php
+
+ // Example 2: Progress tracking
+ $totalBytes = 0;
+ $zip = new ZipStream(
+ outputStream: CallbackStreamWrapper::open(function (string $data) use (&$totalBytes) {
+ $totalBytes += strlen($data);
+ reportProgress($totalBytes); // Report progress to your tracking system
+
+ // Your actual output handling
+ echo $data;
+ }),
+ sendHttpHeaders: false,
+ );
+
+ $zip->addFile('large_file.txt', str_repeat('A', 10000));
+ $zip->finish();
+
+.. code-block:: php
+
+ // Example 3: Data transformation using PHP stream filters
+ // For data transformations, prefer PHP's built-in stream filters
+ $outputStream = fopen('php://output', 'w');
+ stream_filter_append($outputStream, 'convert.base64-encode');
+
+ $zip = new ZipStream(
+ outputStream: $outputStream,
+ sendHttpHeaders: false,
+ );
+
+ $zip->addFile('secret.txt', 'Confidential data');
+ $zip->finish();
+ fclose($outputStream);
+
+.. note::
+ For data transformations, PHP's built-in stream filters are preferred over callback transformations. Stream filters operate at the stream level and maintain data integrity. You can register custom filters using ``stream_filter_register()`` for specialized transformations.
diff --git a/api/vendor/maennchen/zipstream-php/guides/index.rst b/api/vendor/maennchen/zipstream-php/guides/index.rst
index 48f465ae..70211c0d 100644
--- a/api/vendor/maennchen/zipstream-php/guides/index.rst
+++ b/api/vendor/maennchen/zipstream-php/guides/index.rst
@@ -39,7 +39,7 @@ as well:
If ``composer install`` yields the following error, your installation is missing
the `mbstring extension `_,
either `install it `_
-or run the follwoing command:
+or run the following command:
.. code-block::
Your requirements could not be resolved to an installable set of packages.
@@ -120,7 +120,7 @@ It is recommended to extract with another tool like
`7-zip `_.
See `#146 `_.
-It is the responsability of the client code to make sure that files are not
+It is the responsibility of the client code to make sure that files are not
saved with the same path, as it is not possible for the library to figure it out
while streaming a zip.
See `#154 `_.
diff --git a/api/vendor/maennchen/zipstream-php/src/CompressionMethod.php b/api/vendor/maennchen/zipstream-php/src/CompressionMethod.php
index 51e43637..2e1289cc 100644
--- a/api/vendor/maennchen/zipstream-php/src/CompressionMethod.php
+++ b/api/vendor/maennchen/zipstream-php/src/CompressionMethod.php
@@ -4,6 +4,9 @@ declare(strict_types=1);
namespace ZipStream;
+/**
+ * @api
+ */
enum CompressionMethod: int
{
/**
diff --git a/api/vendor/maennchen/zipstream-php/src/Exception.php b/api/vendor/maennchen/zipstream-php/src/Exception.php
index 2e81e307..b3f42e5f 100644
--- a/api/vendor/maennchen/zipstream-php/src/Exception.php
+++ b/api/vendor/maennchen/zipstream-php/src/Exception.php
@@ -4,4 +4,7 @@ declare(strict_types=1);
namespace ZipStream;
+/**
+ * @api
+ */
abstract class Exception extends \Exception {}
diff --git a/api/vendor/maennchen/zipstream-php/src/Exception/DosTimeOverflowException.php b/api/vendor/maennchen/zipstream-php/src/Exception/DosTimeOverflowException.php
index b8d05080..360b5f83 100644
--- a/api/vendor/maennchen/zipstream-php/src/Exception/DosTimeOverflowException.php
+++ b/api/vendor/maennchen/zipstream-php/src/Exception/DosTimeOverflowException.php
@@ -8,7 +8,9 @@ use DateTimeInterface;
use ZipStream\Exception;
/**
- * This Exception gets invoked if a file wasn't found
+ * This Exception gets invoked if a DOS time is overflowing
+ *
+ * @api
*/
class DosTimeOverflowException extends Exception
{
diff --git a/api/vendor/maennchen/zipstream-php/src/Exception/FileNotFoundException.php b/api/vendor/maennchen/zipstream-php/src/Exception/FileNotFoundException.php
index 350a7bfe..b9b1a2b1 100644
--- a/api/vendor/maennchen/zipstream-php/src/Exception/FileNotFoundException.php
+++ b/api/vendor/maennchen/zipstream-php/src/Exception/FileNotFoundException.php
@@ -8,6 +8,8 @@ use ZipStream\Exception;
/**
* This Exception gets invoked if a file wasn't found
+ *
+ * @api
*/
class FileNotFoundException extends Exception
{
diff --git a/api/vendor/maennchen/zipstream-php/src/Exception/FileNotReadableException.php b/api/vendor/maennchen/zipstream-php/src/Exception/FileNotReadableException.php
index 93d0c6c6..29504237 100644
--- a/api/vendor/maennchen/zipstream-php/src/Exception/FileNotReadableException.php
+++ b/api/vendor/maennchen/zipstream-php/src/Exception/FileNotReadableException.php
@@ -7,7 +7,9 @@ namespace ZipStream\Exception;
use ZipStream\Exception;
/**
- * This Exception gets invoked if a file wasn't found
+ * This Exception gets invoked if a file isn't readable
+ *
+ * @api
*/
class FileNotReadableException extends Exception
{
diff --git a/api/vendor/maennchen/zipstream-php/src/Exception/FileSizeIncorrectException.php b/api/vendor/maennchen/zipstream-php/src/Exception/FileSizeIncorrectException.php
index 11f0b67b..e7006697 100644
--- a/api/vendor/maennchen/zipstream-php/src/Exception/FileSizeIncorrectException.php
+++ b/api/vendor/maennchen/zipstream-php/src/Exception/FileSizeIncorrectException.php
@@ -8,6 +8,8 @@ use ZipStream\Exception;
/**
* This Exception gets invoked if a file is not as large as it was specified.
+ *
+ * @api
*/
class FileSizeIncorrectException extends Exception
{
diff --git a/api/vendor/maennchen/zipstream-php/src/Exception/OverflowException.php b/api/vendor/maennchen/zipstream-php/src/Exception/OverflowException.php
index 09bdafb2..d96c4417 100644
--- a/api/vendor/maennchen/zipstream-php/src/Exception/OverflowException.php
+++ b/api/vendor/maennchen/zipstream-php/src/Exception/OverflowException.php
@@ -8,6 +8,8 @@ use ZipStream\Exception;
/**
* This Exception gets invoked if a counter value exceeds storage size
+ *
+ * @api
*/
class OverflowException extends Exception
{
diff --git a/api/vendor/maennchen/zipstream-php/src/Exception/ResourceActionException.php b/api/vendor/maennchen/zipstream-php/src/Exception/ResourceActionException.php
index cbd9b0bb..81a2ea59 100644
--- a/api/vendor/maennchen/zipstream-php/src/Exception/ResourceActionException.php
+++ b/api/vendor/maennchen/zipstream-php/src/Exception/ResourceActionException.php
@@ -8,6 +8,8 @@ use ZipStream\Exception;
/**
* This Exception gets invoked if a resource like `fread` returns false
+ *
+ * @api
*/
class ResourceActionException extends Exception
{
diff --git a/api/vendor/maennchen/zipstream-php/src/Exception/SimulationFileUnknownException.php b/api/vendor/maennchen/zipstream-php/src/Exception/SimulationFileUnknownException.php
index 717c1aaf..240f8251 100644
--- a/api/vendor/maennchen/zipstream-php/src/Exception/SimulationFileUnknownException.php
+++ b/api/vendor/maennchen/zipstream-php/src/Exception/SimulationFileUnknownException.php
@@ -9,6 +9,8 @@ use ZipStream\Exception;
/**
* This Exception gets invoked if a strict simulation is executed and the file
* information can't be determined without reading the entire file.
+ *
+ * @api
*/
class SimulationFileUnknownException extends Exception
{
diff --git a/api/vendor/maennchen/zipstream-php/src/Exception/StreamNotReadableException.php b/api/vendor/maennchen/zipstream-php/src/Exception/StreamNotReadableException.php
index c1446735..b058f55d 100644
--- a/api/vendor/maennchen/zipstream-php/src/Exception/StreamNotReadableException.php
+++ b/api/vendor/maennchen/zipstream-php/src/Exception/StreamNotReadableException.php
@@ -8,6 +8,8 @@ use ZipStream\Exception;
/**
* This Exception gets invoked if a stream can't be read.
+ *
+ * @api
*/
class StreamNotReadableException extends Exception
{
diff --git a/api/vendor/maennchen/zipstream-php/src/Exception/StreamNotSeekableException.php b/api/vendor/maennchen/zipstream-php/src/Exception/StreamNotSeekableException.php
index 606f11f1..ecfd6e3f 100644
--- a/api/vendor/maennchen/zipstream-php/src/Exception/StreamNotSeekableException.php
+++ b/api/vendor/maennchen/zipstream-php/src/Exception/StreamNotSeekableException.php
@@ -9,6 +9,8 @@ use ZipStream\Exception;
/**
* This Exception gets invoked if a non seekable stream is
* provided and zero headers are disabled.
+ *
+ * @api
*/
class StreamNotSeekableException extends Exception
{
diff --git a/api/vendor/maennchen/zipstream-php/src/File.php b/api/vendor/maennchen/zipstream-php/src/File.php
index 1498e4d6..148fc55b 100644
--- a/api/vendor/maennchen/zipstream-php/src/File.php
+++ b/api/vendor/maennchen/zipstream-php/src/File.php
@@ -18,7 +18,7 @@ use ZipStream\Exception\StreamNotSeekableException;
/**
* @internal
*/
-class File
+final class File
{
private const CHUNKED_READ_BLOCK_SIZE = 0x1000000;
diff --git a/api/vendor/maennchen/zipstream-php/src/OperationMode.php b/api/vendor/maennchen/zipstream-php/src/OperationMode.php
index dd650f07..ae45e318 100644
--- a/api/vendor/maennchen/zipstream-php/src/OperationMode.php
+++ b/api/vendor/maennchen/zipstream-php/src/OperationMode.php
@@ -6,6 +6,8 @@ namespace ZipStream;
/**
* ZipStream execution operation modes
+ *
+ * @api
*/
enum OperationMode
{
diff --git a/api/vendor/maennchen/zipstream-php/src/PackField.php b/api/vendor/maennchen/zipstream-php/src/PackField.php
index 892b4009..bf783403 100644
--- a/api/vendor/maennchen/zipstream-php/src/PackField.php
+++ b/api/vendor/maennchen/zipstream-php/src/PackField.php
@@ -10,7 +10,7 @@ use RuntimeException;
* @internal
* TODO: Make class readonly when requiring PHP 8.2 exclusively
*/
-class PackField
+final class PackField
{
public const MAX_V = 0xFFFFFFFF;
diff --git a/api/vendor/maennchen/zipstream-php/src/Version.php b/api/vendor/maennchen/zipstream-php/src/Version.php
index c014f8a1..483d25a5 100644
--- a/api/vendor/maennchen/zipstream-php/src/Version.php
+++ b/api/vendor/maennchen/zipstream-php/src/Version.php
@@ -4,6 +4,9 @@ declare(strict_types=1);
namespace ZipStream;
+/**
+ * @api
+ */
enum Version: int
{
case STORE = 0x000A; // 1.00
diff --git a/api/vendor/maennchen/zipstream-php/src/ZipStream.php b/api/vendor/maennchen/zipstream-php/src/ZipStream.php
index 698ffbb3..d7b939dd 100644
--- a/api/vendor/maennchen/zipstream-php/src/ZipStream.php
+++ b/api/vendor/maennchen/zipstream-php/src/ZipStream.php
@@ -64,6 +64,8 @@ use ZipStream\Exception\ResourceActionException;
* // write archive footer to stream
* $zip->finish();
* ```
+ *
+ * @api
*/
class ZipStream
{
@@ -691,7 +693,7 @@ class ZipStream
/**
* Write zip footer to stream.
*
- * The clase is left in an unusable state after `finish`.
+ * The class is left in an unusable state after `finish`.
*
* ##### Example
*
diff --git a/api/vendor/phpmailer/phpmailer/README.md b/api/vendor/phpmailer/phpmailer/README.md
index 862a4e1a..51c97517 100644
--- a/api/vendor/phpmailer/phpmailer/README.md
+++ b/api/vendor/phpmailer/phpmailer/README.md
@@ -48,7 +48,7 @@ This software is distributed under the [LGPL 2.1](https://www.gnu.org/licenses/o
PHPMailer is available on [Packagist](https://packagist.org/packages/phpmailer/phpmailer) (using semantic versioning), and installation via [Composer](https://getcomposer.org) is the recommended way to install PHPMailer. Just add this line to your `composer.json` file:
```json
-"phpmailer/phpmailer": "^6.10.0"
+"phpmailer/phpmailer": "^6.11.1"
```
or run
diff --git a/api/vendor/phpmailer/phpmailer/VERSION b/api/vendor/phpmailer/phpmailer/VERSION
index cf79bf90..fac714a3 100644
--- a/api/vendor/phpmailer/phpmailer/VERSION
+++ b/api/vendor/phpmailer/phpmailer/VERSION
@@ -1 +1 @@
-6.10.0
+6.11.1
diff --git a/api/vendor/phpmailer/phpmailer/composer.json b/api/vendor/phpmailer/phpmailer/composer.json
index 7b008b7c..e762d59d 100644
--- a/api/vendor/phpmailer/phpmailer/composer.json
+++ b/api/vendor/phpmailer/phpmailer/composer.json
@@ -49,14 +49,15 @@
},
"suggest": {
"decomplexity/SendOauth2": "Adapter for using XOAUTH2 authentication",
+ "ext-imap": "Needed to support advanced email address parsing according to RFC822",
"ext-mbstring": "Needed to send email in multibyte encoding charset or decode encoded addresses",
"ext-openssl": "Needed for secure SMTP sending and DKIM signing",
"greew/oauth2-azure-provider": "Needed for Microsoft Azure XOAUTH2 authentication",
"hayageek/oauth2-yahoo": "Needed for Yahoo XOAUTH2 authentication",
"league/oauth2-google": "Needed for Google XOAUTH2 authentication",
"psr/log": "For optional PSR-3 debug logging",
- "thenetworg/oauth2-azure": "Needed for Microsoft XOAUTH2 authentication",
- "symfony/polyfill-mbstring": "To support UTF-8 if the Mbstring PHP extension is not enabled (^1.2)"
+ "symfony/polyfill-mbstring": "To support UTF-8 if the Mbstring PHP extension is not enabled (^1.2)",
+ "thenetworg/oauth2-azure": "Needed for Microsoft XOAUTH2 authentication"
},
"autoload": {
"psr-4": {
@@ -71,6 +72,7 @@
"license": "LGPL-2.1-only",
"scripts": {
"check": "./vendor/bin/phpcs",
+ "style": "./vendor/bin/phpcbf",
"test": "./vendor/bin/phpunit --no-coverage",
"coverage": "./vendor/bin/phpunit",
"lint": [
diff --git a/api/vendor/phpmailer/phpmailer/language/phpmailer.lang-es.php b/api/vendor/phpmailer/phpmailer/language/phpmailer.lang-es.php
index 4e74bfb7..a871824f 100644
--- a/api/vendor/phpmailer/phpmailer/language/phpmailer.lang-es.php
+++ b/api/vendor/phpmailer/phpmailer/language/phpmailer.lang-es.php
@@ -9,7 +9,7 @@
*/
$PHPMAILER_LANG['authenticate'] = 'Error SMTP: Imposible autentificar.';
-$PHPMAILER_LANG['buggy_php'] = 'Tu versión de PHP está afectada por un bug que puede resultar en mensajes corruptos. Para arreglarlo, cambia a enviar usando SMTP, deshabilita la opción mail.add_x_header en tu php.ini, cambia a MacOS o Linux, o actualiza tu PHP a la versión 7.0.17+ o 7.1.3+.';
+$PHPMAILER_LANG['buggy_php'] = 'Tu versión de PHP ha sido afectada por un bug que puede resultar en mensajes corruptos. Para arreglarlo, cambia a enviar usando SMTP, deshabilita la opción mail.add_x_header en tu php.ini, cambia a MacOS o Linux, o actualiza tu PHP a la versión 7.0.17+ o 7.1.3+.';
$PHPMAILER_LANG['connect_host'] = 'Error SMTP: Imposible conectar al servidor SMTP.';
$PHPMAILER_LANG['data_not_accepted'] = 'Error SMTP: Datos no aceptados.';
$PHPMAILER_LANG['empty_message'] = 'El cuerpo del mensaje está vacío.';
@@ -18,7 +18,7 @@ $PHPMAILER_LANG['execute'] = 'Imposible ejecutar: ';
$PHPMAILER_LANG['extension_missing'] = 'Extensión faltante: ';
$PHPMAILER_LANG['file_access'] = 'Imposible acceder al archivo: ';
$PHPMAILER_LANG['file_open'] = 'Error de Archivo: Imposible abrir el archivo: ';
-$PHPMAILER_LANG['from_failed'] = 'La(s) siguiente(s) direcciones de remitente fallaron: ';
+$PHPMAILER_LANG['from_failed'] = 'La siguiente dirección de remitente falló: ';
$PHPMAILER_LANG['instantiate'] = 'Imposible crear una instancia de la función Mail.';
$PHPMAILER_LANG['invalid_address'] = 'Imposible enviar: dirección de email inválido: ';
$PHPMAILER_LANG['invalid_header'] = 'Nombre o valor de encabezado no válido';
@@ -34,3 +34,5 @@ $PHPMAILER_LANG['smtp_connect_failed'] = 'SMTP Connect() falló.';
$PHPMAILER_LANG['smtp_detail'] = 'Detalle: ';
$PHPMAILER_LANG['smtp_error'] = 'Error del servidor SMTP: ';
$PHPMAILER_LANG['variable_set'] = 'No se pudo configurar la variable: ';
+$PHPMAILER_LANG['imap_recommended'] = 'No se recomienda usar el analizador de direcciones simplificado. Instala la extensión IMAP de PHP para un análisis RFC822 más completo.';
+$PHPMAILER_LANG['deprecated_argument'] = 'El argumento $useimap ha quedado obsoleto';
diff --git a/api/vendor/phpmailer/phpmailer/src/PHPMailer.php b/api/vendor/phpmailer/phpmailer/src/PHPMailer.php
index 2444bcf3..0a8711f4 100644
--- a/api/vendor/phpmailer/phpmailer/src/PHPMailer.php
+++ b/api/vendor/phpmailer/phpmailer/src/PHPMailer.php
@@ -561,9 +561,9 @@ class PHPMailer
* string $body the email body
* string $from email address of sender
* string $extra extra information of possible use
- * "smtp_transaction_id' => last smtp transaction id
+ * 'smtp_transaction_id' => last smtp transaction id
*
- * @var string
+ * @var callable|callable-string
*/
public $action_function = '';
@@ -711,7 +711,7 @@ class PHPMailer
*
* @var array
*/
- protected $language = [];
+ protected static $language = [];
/**
* The number of errors encountered.
@@ -768,7 +768,7 @@ class PHPMailer
*
* @var string
*/
- const VERSION = '6.10.0';
+ const VERSION = '6.11.1';
/**
* Error severity: message only, continue processing.
@@ -1102,7 +1102,7 @@ class PHPMailer
//At-sign is missing.
$error_message = sprintf(
'%s (%s): %s',
- $this->lang('invalid_address'),
+ self::lang('invalid_address'),
$kind,
$address
);
@@ -1187,7 +1187,7 @@ class PHPMailer
if (!in_array($kind, ['to', 'cc', 'bcc', 'Reply-To'])) {
$error_message = sprintf(
'%s: %s',
- $this->lang('Invalid recipient kind'),
+ self::lang('Invalid recipient kind'),
$kind
);
$this->setError($error_message);
@@ -1201,7 +1201,7 @@ class PHPMailer
if (!static::validateAddress($address)) {
$error_message = sprintf(
'%s (%s): %s',
- $this->lang('invalid_address'),
+ self::lang('invalid_address'),
$kind,
$address
);
@@ -1220,12 +1220,16 @@ class PHPMailer
return true;
}
- } elseif (!array_key_exists(strtolower($address), $this->ReplyTo)) {
- $this->ReplyTo[strtolower($address)] = [$address, $name];
+ } else {
+ foreach ($this->ReplyTo as $replyTo) {
+ if (0 === strcasecmp($replyTo[0], $address)) {
+ return false;
+ }
+ }
+ $this->ReplyTo[] = [$address, $name];
return true;
}
-
return false;
}
@@ -1238,15 +1242,18 @@ class PHPMailer
* @see https://www.andrew.cmu.edu/user/agreen1/testing/mrbs/web/Mail/RFC822.php A more careful implementation
*
* @param string $addrstr The address list string
- * @param bool $useimap Whether to use the IMAP extension to parse the list
+ * @param null $useimap Deprecated argument since 6.11.0.
* @param string $charset The charset to use when decoding the address list string.
*
* @return array
*/
- public static function parseAddresses($addrstr, $useimap = true, $charset = self::CHARSET_ISO88591)
+ public static function parseAddresses($addrstr, $useimap = null, $charset = self::CHARSET_ISO88591)
{
+ if ($useimap !== null) {
+ trigger_error(self::lang('deprecated_argument'), E_USER_DEPRECATED);
+ }
$addresses = [];
- if ($useimap && function_exists('imap_rfc822_parse_adrlist')) {
+ if (function_exists('imap_rfc822_parse_adrlist')) {
//Use this built-in parser if it's available
$list = imap_rfc822_parse_adrlist($addrstr, '');
// Clear any potential IMAP errors to get rid of notices being thrown at end of script.
@@ -1256,20 +1263,13 @@ class PHPMailer
'.SYNTAX-ERROR.' !== $address->host &&
static::validateAddress($address->mailbox . '@' . $address->host)
) {
- //Decode the name part if it's present and encoded
+ //Decode the name part if it's present and maybe encoded
if (
- property_exists($address, 'personal') &&
- //Check for a Mbstring constant rather than using extension_loaded, which is sometimes disabled
- defined('MB_CASE_UPPER') &&
- preg_match('/^=\?.*\?=$/s', $address->personal)
+ property_exists($address, 'personal')
+ && is_string($address->personal)
+ && $address->personal !== ''
) {
- $origCharset = mb_internal_encoding();
- mb_internal_encoding($charset);
- //Undo any RFC2047-encoded spaces-as-underscores
- $address->personal = str_replace('_', '=20', $address->personal);
- //Decode the name
- $address->personal = mb_decode_mimeheader($address->personal);
- mb_internal_encoding($origCharset);
+ $address->personal = static::decodeHeader($address->personal, $charset);
}
$addresses[] = [
@@ -1280,40 +1280,51 @@ class PHPMailer
}
} else {
//Use this simpler parser
- $list = explode(',', $addrstr);
- foreach ($list as $address) {
- $address = trim($address);
- //Is there a separate name part?
- if (strpos($address, '<') === false) {
- //No separate name, just use the whole thing
- if (static::validateAddress($address)) {
- $addresses[] = [
- 'name' => '',
- 'address' => $address,
- ];
- }
- } else {
- list($name, $email) = explode('<', $address);
- $email = trim(str_replace('>', '', $email));
- $name = trim($name);
- if (static::validateAddress($email)) {
- //Check for a Mbstring constant rather than using extension_loaded, which is sometimes disabled
- //If this name is encoded, decode it
- if (defined('MB_CASE_UPPER') && preg_match('/^=\?.*\?=$/s', $name)) {
- $origCharset = mb_internal_encoding();
- mb_internal_encoding($charset);
- //Undo any RFC2047-encoded spaces-as-underscores
- $name = str_replace('_', '=20', $name);
- //Decode the name
- $name = mb_decode_mimeheader($name);
- mb_internal_encoding($origCharset);
- }
- $addresses[] = [
- //Remove any surrounding quotes and spaces from the name
- 'name' => trim($name, '\'" '),
- 'address' => $email,
- ];
- }
+ $addresses = static::parseSimplerAddresses($addrstr, $charset);
+ }
+
+ return $addresses;
+ }
+
+ /**
+ * Parse a string containing one or more RFC822-style comma-separated email addresses
+ * with the form "display name " into an array of name/address pairs.
+ * Uses a simpler parser that does not require the IMAP extension but doesnt support
+ * the full RFC822 spec. For full RFC822 support, use the PHP IMAP extension.
+ *
+ * @param string $addrstr The address list string
+ * @param string $charset The charset to use when decoding the address list string.
+ *
+ * @return array
+ */
+ protected static function parseSimplerAddresses($addrstr, $charset)
+ {
+ // Emit a runtime notice to recommend using the IMAP extension for full RFC822 parsing
+ trigger_error(self::lang('imap_recommended'), E_USER_NOTICE);
+
+ $addresses = [];
+ $list = explode(',', $addrstr);
+ foreach ($list as $address) {
+ $address = trim($address);
+ //Is there a separate name part?
+ if (strpos($address, '<') === false) {
+ //No separate name, just use the whole thing
+ if (static::validateAddress($address)) {
+ $addresses[] = [
+ 'name' => '',
+ 'address' => $address,
+ ];
+ }
+ } else {
+ $parsed = static::parseEmailString($address);
+ $email = $parsed['email'];
+ if (static::validateAddress($email)) {
+ $name = static::decodeHeader($parsed['name'], $charset);
+ $addresses[] = [
+ //Remove any surrounding quotes and spaces from the name
+ 'name' => trim($name, '\'" '),
+ 'address' => $email,
+ ];
}
}
}
@@ -1321,6 +1332,42 @@ class PHPMailer
return $addresses;
}
+ /**
+ * Parse a string containing an email address with an optional name
+ * and divide it into a name and email address.
+ *
+ * @param string $input The email with name.
+ *
+ * @return array{name: string, email: string}
+ */
+ private static function parseEmailString($input)
+ {
+ $input = trim((string)$input);
+
+ if ($input === '') {
+ return ['name' => '', 'email' => ''];
+ }
+
+ $pattern = '/^\s*(?:(?:"([^"]*)"|\'([^\']*)\'|([^<]*?))\s*)?<\s*([^>]+)\s*>\s*$/';
+ if (preg_match($pattern, $input, $matches)) {
+ $name = '';
+ // Double quotes including special scenarios.
+ if (isset($matches[1]) && $matches[1] !== '') {
+ $name = $matches[1];
+ // Single quotes including special scenarios.
+ } elseif (isset($matches[2]) && $matches[2] !== '') {
+ $name = $matches[2];
+ // Simplest scenario, name and email are in the format "Name ".
+ } elseif (isset($matches[3])) {
+ $name = trim($matches[3]);
+ }
+
+ return ['name' => $name, 'email' => trim($matches[4])];
+ }
+
+ return ['name' => '', 'email' => $input];
+ }
+
/**
* Set the From and FromName properties.
*
@@ -1334,6 +1381,10 @@ class PHPMailer
*/
public function setFrom($address, $name = '', $auto = true)
{
+ if (is_null($name)) {
+ //Helps avoid a deprecation warning in the preg_replace() below
+ $name = '';
+ }
$address = trim((string)$address);
$name = trim(preg_replace('/[\r\n]+/', '', $name)); //Strip breaks and trim
//Don't validate now addresses with IDN. Will be done in send().
@@ -1345,7 +1396,7 @@ class PHPMailer
) {
$error_message = sprintf(
'%s (From): %s',
- $this->lang('invalid_address'),
+ self::lang('invalid_address'),
$address
);
$this->setError($error_message);
@@ -1601,7 +1652,7 @@ class PHPMailer
&& ini_get('mail.add_x_header') === '1'
&& stripos(PHP_OS, 'WIN') === 0
) {
- trigger_error($this->lang('buggy_php'), E_USER_WARNING);
+ trigger_error(self::lang('buggy_php'), E_USER_WARNING);
}
try {
@@ -1631,7 +1682,7 @@ class PHPMailer
call_user_func_array([$this, 'addAnAddress'], $params);
}
if (count($this->to) + count($this->cc) + count($this->bcc) < 1) {
- throw new Exception($this->lang('provide_address'), self::STOP_CRITICAL);
+ throw new Exception(self::lang('provide_address'), self::STOP_CRITICAL);
}
//Validate From, Sender, and ConfirmReadingTo addresses
@@ -1648,7 +1699,7 @@ class PHPMailer
if (!static::validateAddress($this->{$address_kind})) {
$error_message = sprintf(
'%s (%s): %s',
- $this->lang('invalid_address'),
+ self::lang('invalid_address'),
$address_kind,
$this->{$address_kind}
);
@@ -1670,7 +1721,7 @@ class PHPMailer
$this->setMessageType();
//Refuse to send an empty message unless we are specifically allowing it
if (!$this->AllowEmpty && empty($this->Body)) {
- throw new Exception($this->lang('empty_message'), self::STOP_CRITICAL);
+ throw new Exception(self::lang('empty_message'), self::STOP_CRITICAL);
}
//Trim subject consistently
@@ -1809,8 +1860,10 @@ class PHPMailer
} else {
$sendmailFmt = '%s -oi -f%s -t';
}
+ } elseif ($this->Mailer === 'qmail') {
+ $sendmailFmt = '%s';
} else {
- //allow sendmail to choose a default envelope sender. It may
+ //Allow sendmail to choose a default envelope sender. It may
//seem preferable to force it to use the From header as with
//SMTP, but that introduces new problems (see
//), and
@@ -1828,33 +1881,35 @@ class PHPMailer
foreach ($this->SingleToArray as $toAddr) {
$mail = @popen($sendmail, 'w');
if (!$mail) {
- throw new Exception($this->lang('execute') . $this->Sendmail, self::STOP_CRITICAL);
+ throw new Exception(self::lang('execute') . $this->Sendmail, self::STOP_CRITICAL);
}
$this->edebug("To: {$toAddr}");
fwrite($mail, 'To: ' . $toAddr . "\n");
fwrite($mail, $header);
fwrite($mail, $body);
$result = pclose($mail);
- $addrinfo = static::parseAddresses($toAddr, true, $this->CharSet);
- $this->doCallback(
- ($result === 0),
- [[$addrinfo['address'], $addrinfo['name']]],
- $this->cc,
- $this->bcc,
- $this->Subject,
- $body,
- $this->From,
- []
- );
+ $addrinfo = static::parseAddresses($toAddr, null, $this->CharSet);
+ foreach ($addrinfo as $addr) {
+ $this->doCallback(
+ ($result === 0),
+ [[$addr['address'], $addr['name']]],
+ $this->cc,
+ $this->bcc,
+ $this->Subject,
+ $body,
+ $this->From,
+ []
+ );
+ }
$this->edebug("Result: " . ($result === 0 ? 'true' : 'false'));
if (0 !== $result) {
- throw new Exception($this->lang('execute') . $this->Sendmail, self::STOP_CRITICAL);
+ throw new Exception(self::lang('execute') . $this->Sendmail, self::STOP_CRITICAL);
}
}
} else {
$mail = @popen($sendmail, 'w');
if (!$mail) {
- throw new Exception($this->lang('execute') . $this->Sendmail, self::STOP_CRITICAL);
+ throw new Exception(self::lang('execute') . $this->Sendmail, self::STOP_CRITICAL);
}
fwrite($mail, $header);
fwrite($mail, $body);
@@ -1871,7 +1926,7 @@ class PHPMailer
);
$this->edebug("Result: " . ($result === 0 ? 'true' : 'false'));
if (0 !== $result) {
- throw new Exception($this->lang('execute') . $this->Sendmail, self::STOP_CRITICAL);
+ throw new Exception(self::lang('execute') . $this->Sendmail, self::STOP_CRITICAL);
}
}
@@ -2010,17 +2065,19 @@ class PHPMailer
if ($this->SingleTo && count($toArr) > 1) {
foreach ($toArr as $toAddr) {
$result = $this->mailPassthru($toAddr, $this->Subject, $body, $header, $params);
- $addrinfo = static::parseAddresses($toAddr, true, $this->CharSet);
- $this->doCallback(
- $result,
- [[$addrinfo['address'], $addrinfo['name']]],
- $this->cc,
- $this->bcc,
- $this->Subject,
- $body,
- $this->From,
- []
- );
+ $addrinfo = static::parseAddresses($toAddr, null, $this->CharSet);
+ foreach ($addrinfo as $addr) {
+ $this->doCallback(
+ $result,
+ [[$addr['address'], $addr['name']]],
+ $this->cc,
+ $this->bcc,
+ $this->Subject,
+ $body,
+ $this->From,
+ []
+ );
+ }
}
} else {
$result = $this->mailPassthru($to, $this->Subject, $body, $header, $params);
@@ -2030,7 +2087,7 @@ class PHPMailer
ini_set('sendmail_from', $old_from);
}
if (!$result) {
- throw new Exception($this->lang('instantiate'), self::STOP_CRITICAL);
+ throw new Exception(self::lang('instantiate'), self::STOP_CRITICAL);
}
return true;
@@ -2116,12 +2173,12 @@ class PHPMailer
$header = static::stripTrailingWSP($header) . static::$LE . static::$LE;
$bad_rcpt = [];
if (!$this->smtpConnect($this->SMTPOptions)) {
- throw new Exception($this->lang('smtp_connect_failed'), self::STOP_CRITICAL);
+ throw new Exception(self::lang('smtp_connect_failed'), self::STOP_CRITICAL);
}
//If we have recipient addresses that need Unicode support,
//but the server doesn't support it, stop here
if ($this->UseSMTPUTF8 && !$this->smtp->getServerExt('SMTPUTF8')) {
- throw new Exception($this->lang('no_smtputf8'), self::STOP_CRITICAL);
+ throw new Exception(self::lang('no_smtputf8'), self::STOP_CRITICAL);
}
//Sender already validated in preSend()
if ('' === $this->Sender) {
@@ -2133,7 +2190,7 @@ class PHPMailer
$this->smtp->xclient($this->SMTPXClient);
}
if (!$this->smtp->mail($smtp_from)) {
- $this->setError($this->lang('from_failed') . $smtp_from . ' : ' . implode(',', $this->smtp->getError()));
+ $this->setError(self::lang('from_failed') . $smtp_from . ' : ' . implode(',', $this->smtp->getError()));
throw new Exception($this->ErrorInfo, self::STOP_CRITICAL);
}
@@ -2155,7 +2212,7 @@ class PHPMailer
//Only send the DATA command if we have viable recipients
if ((count($this->all_recipients) > count($bad_rcpt)) && !$this->smtp->data($header . $body)) {
- throw new Exception($this->lang('data_not_accepted'), self::STOP_CRITICAL);
+ throw new Exception(self::lang('data_not_accepted'), self::STOP_CRITICAL);
}
$smtp_transaction_id = $this->smtp->getLastTransactionID();
@@ -2186,7 +2243,7 @@ class PHPMailer
foreach ($bad_rcpt as $bad) {
$errstr .= $bad['to'] . ': ' . $bad['error'];
}
- throw new Exception($this->lang('recipients_failed') . $errstr, self::STOP_CONTINUE);
+ throw new Exception(self::lang('recipients_failed') . $errstr, self::STOP_CONTINUE);
}
return true;
@@ -2240,7 +2297,7 @@ class PHPMailer
$hostinfo
)
) {
- $this->edebug($this->lang('invalid_hostentry') . ' ' . trim($hostentry));
+ $this->edebug(self::lang('invalid_hostentry') . ' ' . trim($hostentry));
//Not a valid host entry
continue;
}
@@ -2252,7 +2309,7 @@ class PHPMailer
//Check the host name is a valid name or IP address before trying to use it
if (!static::isValidHost($hostinfo[2])) {
- $this->edebug($this->lang('invalid_host') . ' ' . $hostinfo[2]);
+ $this->edebug(self::lang('invalid_host') . ' ' . $hostinfo[2]);
continue;
}
$prefix = '';
@@ -2272,7 +2329,7 @@ class PHPMailer
if (static::ENCRYPTION_STARTTLS === $secure || static::ENCRYPTION_SMTPS === $secure) {
//Check for an OpenSSL constant rather than using extension_loaded, which is sometimes disabled
if (!$sslext) {
- throw new Exception($this->lang('extension_missing') . 'openssl', self::STOP_CRITICAL);
+ throw new Exception(self::lang('extension_missing') . 'openssl', self::STOP_CRITICAL);
}
}
$host = $hostinfo[2];
@@ -2324,7 +2381,7 @@ class PHPMailer
$this->oauth
)
) {
- throw new Exception($this->lang('authenticate'));
+ throw new Exception(self::lang('authenticate'));
}
return true;
@@ -2374,7 +2431,7 @@ class PHPMailer
*
* @return bool Returns true if the requested language was loaded, false otherwise.
*/
- public function setLanguage($langcode = 'en', $lang_path = '')
+ public static function setLanguage($langcode = 'en', $lang_path = '')
{
//Backwards compatibility for renamed language codes
$renamed_langcodes = [
@@ -2423,6 +2480,9 @@ class PHPMailer
'smtp_error' => 'SMTP server error: ',
'variable_set' => 'Cannot set or reset variable: ',
'no_smtputf8' => 'Server does not support SMTPUTF8 needed to send to Unicode addresses',
+ 'imap_recommended' => 'Using simplified address parser is not recommended. ' .
+ 'Install the PHP IMAP extension for full RFC822 parsing.',
+ 'deprecated_argument' => 'Argument $useimap is deprecated',
];
if (empty($lang_path)) {
//Calculate an absolute path so it can work if CWD is not here
@@ -2489,7 +2549,7 @@ class PHPMailer
}
}
}
- $this->language = $PHPMAILER_LANG;
+ self::$language = $PHPMAILER_LANG;
return $foundlang; //Returns false if language not found
}
@@ -2501,11 +2561,11 @@ class PHPMailer
*/
public function getTranslations()
{
- if (empty($this->language)) {
- $this->setLanguage(); // Set the default language.
+ if (empty(self::$language)) {
+ self::setLanguage(); // Set the default language.
}
- return $this->language;
+ return self::$language;
}
/**
@@ -2928,10 +2988,6 @@ class PHPMailer
//Create unique IDs and preset boundaries
$this->setBoundaries();
- if ($this->sign_key_file) {
- $body .= $this->getMailMIME() . static::$LE;
- }
-
$this->setWordWrap();
$bodyEncoding = $this->Encoding;
@@ -2963,6 +3019,12 @@ class PHPMailer
if (static::ENCODING_BASE64 !== $altBodyEncoding && static::hasLineLongerThanMax($this->AltBody)) {
$altBodyEncoding = static::ENCODING_QUOTED_PRINTABLE;
}
+
+ if ($this->sign_key_file) {
+ $this->Encoding = $bodyEncoding;
+ $body .= $this->getMailMIME() . static::$LE;
+ }
+
//Use this as a preamble in all multipart message types
$mimepre = '';
switch ($this->message_type) {
@@ -3144,12 +3206,12 @@ class PHPMailer
if ($this->isError()) {
$body = '';
if ($this->exceptions) {
- throw new Exception($this->lang('empty_message'), self::STOP_CRITICAL);
+ throw new Exception(self::lang('empty_message'), self::STOP_CRITICAL);
}
} elseif ($this->sign_key_file) {
try {
if (!defined('PKCS7_TEXT')) {
- throw new Exception($this->lang('extension_missing') . 'openssl');
+ throw new Exception(self::lang('extension_missing') . 'openssl');
}
$file = tempnam(sys_get_temp_dir(), 'srcsign');
@@ -3187,7 +3249,7 @@ class PHPMailer
$body = $parts[1];
} else {
@unlink($signed);
- throw new Exception($this->lang('signing') . openssl_error_string());
+ throw new Exception(self::lang('signing') . openssl_error_string());
}
} catch (Exception $exc) {
$body = '';
@@ -3332,7 +3394,7 @@ class PHPMailer
) {
try {
if (!static::fileIsAccessible($path)) {
- throw new Exception($this->lang('file_access') . $path, self::STOP_CONTINUE);
+ throw new Exception(self::lang('file_access') . $path, self::STOP_CONTINUE);
}
//If a MIME type is not specified, try to work it out from the file name
@@ -3345,7 +3407,7 @@ class PHPMailer
$name = $filename;
}
if (!$this->validateEncoding($encoding)) {
- throw new Exception($this->lang('encoding') . $encoding);
+ throw new Exception(self::lang('encoding') . $encoding);
}
$this->attachment[] = [
@@ -3506,11 +3568,11 @@ class PHPMailer
{
try {
if (!static::fileIsAccessible($path)) {
- throw new Exception($this->lang('file_open') . $path, self::STOP_CONTINUE);
+ throw new Exception(self::lang('file_open') . $path, self::STOP_CONTINUE);
}
$file_buffer = file_get_contents($path);
if (false === $file_buffer) {
- throw new Exception($this->lang('file_open') . $path, self::STOP_CONTINUE);
+ throw new Exception(self::lang('file_open') . $path, self::STOP_CONTINUE);
}
$file_buffer = $this->encodeString($file_buffer, $encoding);
@@ -3563,9 +3625,9 @@ class PHPMailer
$encoded = $this->encodeQP($str);
break;
default:
- $this->setError($this->lang('encoding') . $encoding);
+ $this->setError(self::lang('encoding') . $encoding);
if ($this->exceptions) {
- throw new Exception($this->lang('encoding') . $encoding);
+ throw new Exception(self::lang('encoding') . $encoding);
}
break;
}
@@ -3671,6 +3733,42 @@ class PHPMailer
return trim(static::normalizeBreaks($encoded));
}
+ /**
+ * Decode an RFC2047-encoded header value
+ * Attempts multiple strategies so it works even when the mbstring extension is disabled.
+ *
+ * @param string $value The header value to decode
+ * @param string $charset The target charset to convert to, defaults to ISO-8859-1 for BC
+ *
+ * @return string The decoded header value
+ */
+ public static function decodeHeader($value, $charset = self::CHARSET_ISO88591)
+ {
+ if (!is_string($value) || $value === '') {
+ return '';
+ }
+ // Detect the presence of any RFC2047 encoded-words
+ $hasEncodedWord = (bool) preg_match('/=\?.*\?=/s', $value);
+ if ($hasEncodedWord && defined('MB_CASE_UPPER')) {
+ $origCharset = mb_internal_encoding();
+ // Always decode to UTF-8 to provide a consistent, modern output encoding.
+ mb_internal_encoding($charset);
+ if (PHP_VERSION_ID < 80300) {
+ // Undo any RFC2047-encoded spaces-as-underscores.
+ $value = str_replace('_', '=20', $value);
+ } else {
+ // PHP 8.3+ already interprets underscores as spaces. Remove additional
+ // linear whitespace between adjacent encoded words to avoid double spacing.
+ $value = preg_replace('/(\?=)\s+(=\?)/', '$1$2', $value);
+ }
+ // Decode the header value
+ $value = mb_decode_mimeheader($value);
+ mb_internal_encoding($origCharset);
+ }
+
+ return $value;
+ }
+
/**
* Check if a string contains multi-byte characters.
*
@@ -3840,7 +3938,7 @@ class PHPMailer
}
if (!$this->validateEncoding($encoding)) {
- throw new Exception($this->lang('encoding') . $encoding);
+ throw new Exception(self::lang('encoding') . $encoding);
}
//Append to $attachment array
@@ -3899,7 +3997,7 @@ class PHPMailer
) {
try {
if (!static::fileIsAccessible($path)) {
- throw new Exception($this->lang('file_access') . $path, self::STOP_CONTINUE);
+ throw new Exception(self::lang('file_access') . $path, self::STOP_CONTINUE);
}
//If a MIME type is not specified, try to work it out from the file name
@@ -3908,7 +4006,7 @@ class PHPMailer
}
if (!$this->validateEncoding($encoding)) {
- throw new Exception($this->lang('encoding') . $encoding);
+ throw new Exception(self::lang('encoding') . $encoding);
}
$filename = (string) static::mb_pathinfo($path, PATHINFO_BASENAME);
@@ -3974,7 +4072,7 @@ class PHPMailer
}
if (!$this->validateEncoding($encoding)) {
- throw new Exception($this->lang('encoding') . $encoding);
+ throw new Exception(self::lang('encoding') . $encoding);
}
//Append to $attachment array
@@ -4231,7 +4329,7 @@ class PHPMailer
}
if (strpbrk($name . $value, "\r\n") !== false) {
if ($this->exceptions) {
- throw new Exception($this->lang('invalid_header'));
+ throw new Exception(self::lang('invalid_header'));
}
return false;
@@ -4255,15 +4353,15 @@ class PHPMailer
if ('smtp' === $this->Mailer && null !== $this->smtp) {
$lasterror = $this->smtp->getError();
if (!empty($lasterror['error'])) {
- $msg .= ' ' . $this->lang('smtp_error') . $lasterror['error'];
+ $msg .= ' ' . self::lang('smtp_error') . $lasterror['error'];
if (!empty($lasterror['detail'])) {
- $msg .= ' ' . $this->lang('smtp_detail') . $lasterror['detail'];
+ $msg .= ' ' . self::lang('smtp_detail') . $lasterror['detail'];
}
if (!empty($lasterror['smtp_code'])) {
- $msg .= ' ' . $this->lang('smtp_code') . $lasterror['smtp_code'];
+ $msg .= ' ' . self::lang('smtp_code') . $lasterror['smtp_code'];
}
if (!empty($lasterror['smtp_code_ex'])) {
- $msg .= ' ' . $this->lang('smtp_code_ex') . $lasterror['smtp_code_ex'];
+ $msg .= ' ' . self::lang('smtp_code_ex') . $lasterror['smtp_code_ex'];
}
}
}
@@ -4388,21 +4486,21 @@ class PHPMailer
*
* @return string
*/
- protected function lang($key)
+ protected static function lang($key)
{
- if (count($this->language) < 1) {
- $this->setLanguage(); //Set the default language
+ if (count(self::$language) < 1) {
+ self::setLanguage(); //Set the default language
}
- if (array_key_exists($key, $this->language)) {
+ if (array_key_exists($key, self::$language)) {
if ('smtp_connect_failed' === $key) {
//Include a link to troubleshooting docs on SMTP connection failure.
//This is by far the biggest cause of support questions
//but it's usually not PHPMailer's fault.
- return $this->language[$key] . ' https://github.com/PHPMailer/PHPMailer/wiki/Troubleshooting';
+ return self::$language[$key] . ' https://github.com/PHPMailer/PHPMailer/wiki/Troubleshooting';
}
- return $this->language[$key];
+ return self::$language[$key];
}
//Return the key as a fallback
@@ -4417,7 +4515,7 @@ class PHPMailer
*/
private function getSmtpErrorMessage($base_key)
{
- $message = $this->lang($base_key);
+ $message = self::lang($base_key);
$error = $this->smtp->getError();
if (!empty($error['error'])) {
$message .= ' ' . $error['error'];
@@ -4461,7 +4559,7 @@ class PHPMailer
//Ensure name is not empty, and that neither name nor value contain line breaks
if (empty($name) || strpbrk($name . $value, "\r\n") !== false) {
if ($this->exceptions) {
- throw new Exception($this->lang('invalid_header'));
+ throw new Exception(self::lang('invalid_header'));
}
return false;
@@ -4854,7 +4952,7 @@ class PHPMailer
return true;
}
- $this->setError($this->lang('variable_set') . $name);
+ $this->setError(self::lang('variable_set') . $name);
return false;
}
@@ -4992,7 +5090,7 @@ class PHPMailer
{
if (!defined('PKCS7_TEXT')) {
if ($this->exceptions) {
- throw new Exception($this->lang('extension_missing') . 'openssl');
+ throw new Exception(self::lang('extension_missing') . 'openssl');
}
return '';
diff --git a/api/vendor/phpmailer/phpmailer/src/POP3.php b/api/vendor/phpmailer/phpmailer/src/POP3.php
index 1190a1e2..2c2cf789 100644
--- a/api/vendor/phpmailer/phpmailer/src/POP3.php
+++ b/api/vendor/phpmailer/phpmailer/src/POP3.php
@@ -46,7 +46,7 @@ class POP3
*
* @var string
*/
- const VERSION = '6.10.0';
+ const VERSION = '6.11.1';
/**
* Default POP3 port number.
diff --git a/api/vendor/phpmailer/phpmailer/src/SMTP.php b/api/vendor/phpmailer/phpmailer/src/SMTP.php
index 7226ee93..3772c94a 100644
--- a/api/vendor/phpmailer/phpmailer/src/SMTP.php
+++ b/api/vendor/phpmailer/phpmailer/src/SMTP.php
@@ -35,7 +35,7 @@ class SMTP
*
* @var string
*/
- const VERSION = '6.10.0';
+ const VERSION = '6.11.1';
/**
* SMTP line break constant.
@@ -205,6 +205,7 @@ class SMTP
'Haraka' => '/[\d]{3} Message Queued \((.*)\)/',
'ZoneMTA' => '/[\d]{3} Message queued as (.*)/',
'Mailjet' => '/[\d]{3} OK queued as (.*)/',
+ 'Gsmtp' => '/[\d]{3} 2\.0\.0 OK (.*) - gsmtp/',
];
/**
@@ -633,10 +634,41 @@ class SMTP
return false;
}
$oauth = $OAuth->getOauth64();
-
- //Start authentication
- if (!$this->sendCommand('AUTH', 'AUTH XOAUTH2 ' . $oauth, 235)) {
- return false;
+ /*
+ * An SMTP command line can have a maximum length of 512 bytes, including the command name,
+ * so the base64-encoded OAUTH token has a maximum length of:
+ * 512 - 13 (AUTH XOAUTH2) - 2 (CRLF) = 497 bytes
+ * If the token is longer than that, the command and the token must be sent separately as described in
+ * https://www.rfc-editor.org/rfc/rfc4954#section-4
+ */
+ if ($oauth === '') {
+ //Sending an empty auth token is legitimate, but it must be encoded as '='
+ //to indicate it's not a 2-part command
+ if (!$this->sendCommand('AUTH', 'AUTH XOAUTH2 =', 235)) {
+ return false;
+ }
+ } elseif (strlen($oauth) <= 497) {
+ //Authenticate using a token in the initial-response part
+ if (!$this->sendCommand('AUTH', 'AUTH XOAUTH2 ' . $oauth, 235)) {
+ return false;
+ }
+ } else {
+ //The token is too long, so we need to send it in two parts.
+ //Send the auth command without a token and expect a 334
+ if (!$this->sendCommand('AUTH', 'AUTH XOAUTH2', 334)) {
+ return false;
+ }
+ //Send the token
+ if (!$this->sendCommand('OAuth TOKEN', $oauth, [235, 334])) {
+ return false;
+ }
+ //If the server answers with 334, send an empty line and wait for a 235
+ if (
+ substr($this->last_reply, 0, 3) === '334'
+ && $this->sendCommand('AUTH End', '', 235)
+ ) {
+ return false;
+ }
}
break;
default:
@@ -1309,7 +1341,16 @@ class SMTP
//stream_select returns false when the `select` system call is interrupted
//by an incoming signal, try the select again
- if (stripos($message, 'interrupted system call') !== false) {
+ if (
+ stripos($message, 'interrupted system call') !== false ||
+ (
+ // on applications with a different locale than english, the message above is not found because
+ // it's translated. So we also check for the SOCKET_EINTR constant which is defined under
+ // Windows and UNIX-like platforms (if available on the platform).
+ defined('SOCKET_EINTR') &&
+ stripos($message, 'stream_select(): Unable to select [' . SOCKET_EINTR . ']') !== false
+ )
+ ) {
$this->edebug(
'SMTP -> get_lines(): retrying stream_select',
self::DEBUG_LOWLEVEL
diff --git a/api/vendor/phpoffice/phpspreadsheet/CHANGELOG.md b/api/vendor/phpoffice/phpspreadsheet/CHANGELOG.md
index 8084e7b0..295f4fe6 100644
--- a/api/vendor/phpoffice/phpspreadsheet/CHANGELOG.md
+++ b/api/vendor/phpoffice/phpspreadsheet/CHANGELOG.md
@@ -3,36 +3,267 @@
All notable changes to this project will be documented in this file.
The format is based on [Keep a Changelog](https://keepachangelog.com)
-and this project adheres to [Semantic Versioning](https://semver.org).
+and this project adheres to [Semantic Versioning](https://semver.org). Thia is always true of the master branch. Some earlier branches remain supported and security fixes are applied to them; if the security fix represents a breaking change, it may have to be applied as a minor or patch version.
-# 2025-02-07 - 2.3.8
+## TBD - 5.2.0
-### Fixed
+### Added
-- Xls writer Parser Mishandling True/False Argument. Backport of [PR #4333](https://github.com/PHPOffice/PhpSpreadsheet/pull/4333)
-- Xls writer Parser Parse By Character Not Byte. Backport of [PR #4344](https://github.com/PHPOffice/PhpSpreadsheet/pull/4344)
+- Nothing yet.
-# 2025-01-26 - 2.3.7
+### Removed
-### Fixed
+- Nothing yet.
-- Backported security patch for control characters in protocol.
-- Use Composer\Pcre in Xls/Parser. Partial backport of [PR #4203](https://github.com/PHPOffice/PhpSpreadsheet/pull/4203)
+### Changed
-# 2025-01-11 - 2.3.6
+- Nothing yet.
+
+### Moved
+
+- Nothing yet.
### Deprecated
-- Worksheet::getHashCode is no longer needed.
+- Nothing yet.
### Fixed
-- Backported security patch for Html navigation.
-- Change hash code for worksheet. Backport of [PR #4207](https://github.com/PHPOffice/PhpSpreadsheet/pull/4207)
-- Retitling cloned worksheets. Backport of [PR #4302](https://github.com/PHPOffice/PhpSpreadsheet/pull/4302)
+- Nothing yet.
+## 2025-09-03 - 5.1.0
-# 2024-12-26 - 2.3.5
+### Added
+
+- Add Conditional Formatting with IconSet (Xlsx only). [Issue #4560](https://github.com/PHPOffice/PhpSpreadsheet/issues/4560) [PR #4574](https://github.com/PHPOffice/PhpSpreadsheet/pull/4574)
+- Copy cell adjusting formula. [Issue #1203](https://github.com/PHPOffice/PhpSpreadsheet/issues/1203) [PR #4577](https://github.com/PHPOffice/PhpSpreadsheet/pull/4577)
+- splitRange and ProtectedRange. [Issue #1457](https://github.com/PHPOffice/PhpSpreadsheet/issues/1457) [PR #4580](https://github.com/PHPOffice/PhpSpreadsheet/pull/4580)
+- Option to create Blank Sheet if LoadSheetsOnly doesn't find any. [PR #4618](https://github.com/PHPOffice/PhpSpreadsheet/pull/4618)
+
+### Fixed
+
+- Google-only formulas exported from Google Sheets. [Issue #1637](https://github.com/PHPOffice/PhpSpreadsheet/issues/1637) [PR #4579](https://github.com/PHPOffice/PhpSpreadsheet/pull/4579)
+- Maximum column width. [PR #4581](https://github.com/PHPOffice/PhpSpreadsheet/pull/4581)
+- PrintArea after row/column delete. [Issue #2912](https://github.com/PHPOffice/PhpSpreadsheet/issues/2912) [PR #4598](https://github.com/PHPOffice/PhpSpreadsheet/pull/4598)
+- Remove deprecated imagedestroy call. [PR #4625](https://github.com/PHPOffice/PhpSpreadsheet/pull/4625)
+- Excel 2007 problem with newlines. [Issue #4619](https://github.com/PHPOffice/PhpSpreadsheet/issues/4619) [PR #4620](https://github.com/PHPOffice/PhpSpreadsheet/pull/4620)
+- Compatibility changes for Php 8.5. [PR #4601](https://github.com/PHPOffice/PhpSpreadsheet/pull/4601) [PR #4611](https://github.com/PHPOffice/PhpSpreadsheet/pull/4611)
+
+## 2025-08-10 - 5.0.0
+
+### Breaking Changes
+
+- Images will be loaded from an external source (e.g. http://example.com/img.png) only if the reader is explicitly set to allow it via `$reader->setAllowExternalImages(true)`. We do not believe that loading of external images is a widely used feature.
+- Deletion of items deprecated in Release 4. See "removed" below.
+- Move some properties from Base Reader to Html Reader. [PR #4551](https://github.com/PHPOffice/PhpSpreadsheet/pull/4551)
+- DefaultValueBinder will treat integers with more than 15 digits as strings. [Issue #4522](https://github.com/PHPOffice/PhpSpreadsheet/issues/4522) [PR #4527](https://github.com/PHPOffice/PhpSpreadsheet/pull/4527)
+
+### Removed
+
+- Theme public constants COLOR_SCHEME_2013_PLUS_NAME (use COLOR_SCHEME_2013_2022_NAME) and COLOR_SCHEME_2013_PLUS (use COLOR_SCHEME_2013_2022).
+
+### Fixed
+
+- Additional floating-point precision changes. [Issue #1324](https://github.com/PHPOffice/PhpSpreadsheet/issues/1324) [PR #4575](https://github.com/PHPOffice/PhpSpreadsheet/pull/4575)
+- Header/Footer images expand location. [Issue #484](https://github.com/PHPOffice/PhpSpreadsheet/issues/484) [Issue #1318](https://github.com/PHPOffice/PhpSpreadsheet/issues/1318) [PR #4572](https://github.com/PHPOffice/PhpSpreadsheet/pull/4572)
+- Create uninitialized cell if used in calculation. [Issue #4558](https://github.com/PHPOffice/PhpSpreadsheet/issues/4558) [Issue #4530](https://github.com/PHPOffice/PhpSpreadsheet/issues/4530) [PR #4565](https://github.com/PHPOffice/PhpSpreadsheet/pull/4565)
+- Shared/Date::isDateTime handle cells which calculate as arrays. [Issue #4557](https://github.com/PHPOffice/PhpSpreadsheet/issues/4557) [PR #4562](https://github.com/PHPOffice/PhpSpreadsheet/pull/4562)
+- Xlsx Writer eliminate xml:space from non-text nodes. [Issue #4542](https://github.com/PHPOffice/PhpSpreadsheet/issues/4542) [PR #4556](https://github.com/PHPOffice/PhpSpreadsheet/pull/4556)
+
+## 2025-07-23 - 4.5.0
+
+### Added
+
+- Add to all readers the option to allow or forbid fetching external images. This is unconditionally allowed now. The default will be set to "allow", so no code changes are necessary. However, we are giving consideration to changing the default. [PR #4543](https://github.com/PHPOffice/PhpSpreadsheet/pull/4543)
+- Address Excel Inappropriate Number Format Substitution. [PR #4532](https://github.com/PHPOffice/PhpSpreadsheet/pull/4532)
+
+### Fixed
+
+- Html Writer Conditional Formatting Inline Css. [Issue #4539](https://github.com/PHPOffice/PhpSpreadsheet/issues/4539) [PR #4541](https://github.com/PHPOffice/PhpSpreadsheet/pull/4541)
+- Do not use htmlspecialchars when formatting XML. [Issue #4537](https://github.com/PHPOffice/PhpSpreadsheet/issues/4537) [PR #4540](https://github.com/PHPOffice/PhpSpreadsheet/pull/4540)
+- Writer Html/Pdf support RTL alignment of tables. [Issue #1104](https://github.com/PHPOffice/PhpSpreadsheet/issues/1104) [PR #4535](https://github.com/PHPOffice/PhpSpreadsheet/pull/4535)
+- Xlsx Reader use dynamic arrays if spreadsheet did so. [PR #4533](https://github.com/PHPOffice/PhpSpreadsheet/pull/4533)
+- Ods Reader Nested table-row. [Issue #4528](https://github.com/PHPOffice/PhpSpreadsheet/issues/4528) [Issue #2507](https://github.com/PHPOffice/PhpSpreadsheet/issues/2507) [PR #4531](https://github.com/PHPOffice/PhpSpreadsheet/pull/4531)
+- Recognize application/x-empty mimetype. [Issue #4521](https://github.com/PHPOffice/PhpSpreadsheet/issues/4521) [PR #4524](https://github.com/PHPOffice/PhpSpreadsheet/pull/4524)
+- Micro-optimization in getSheetByName. [PR #4499](https://github.com/PHPOffice/PhpSpreadsheet/pull/4499)
+- Bug in resizeMatricesExtend. [Issue #4451](https://github.com/PHPOffice/PhpSpreadsheet/issues/4451) [PR #4474](https://github.com/PHPOffice/PhpSpreadsheet/pull/4474)
+- Allow Replace of Dummy Function with Custom Function. [PR #4544](https://github.com/PHPOffice/PhpSpreadsheet/pull/4544)
+- Preserve 0x0a in Strings if Desired. [Issue #347](https://github.com/PHPOffice/PhpSpreadsheet/issues/347) [PR #4536](https://github.com/PHPOffice/PhpSpreadsheet/pull/4536)
+
+## 2025-06-22 - 4.4.0
+
+### Added
+
+- VSTACK and HSTACK. [Issue #4485](https://github.com/PHPOffice/PhpSpreadsheet/issues/4485) [PR #4492](https://github.com/PHPOffice/PhpSpreadsheet/pull/4492)
+- TOCOL and TOROW. [PR #4493](https://github.com/PHPOffice/PhpSpreadsheet/pull/4493)
+- Support Current Office Theme. [PR #4500](https://github.com/PHPOffice/PhpSpreadsheet/pull/4500)
+
+### Deprecated
+
+- Theme constants COLOR_SCHEME_2013_PLUS_NAME (use COLOR_SCHEME_2013_2022_NAME) and COLOR_SCHEME_2013_PLUS (use COLOR_SCHEME_2013_2022).
+
+### Fixed
+
+- Various Writers RichText TextElement Should Inherit Cell Style. [Issue #1154](https://github.com/PHPOffice/PhpSpreadsheet/issues/1154) [PR #4487](https://github.com/PHPOffice/PhpSpreadsheet/pull/4487)
+- Minor Changes to FILTER function. [PR #4491](https://github.com/PHPOffice/PhpSpreadsheet/pull/4491)
+- Allow Xlsx Reader/Writer to support Font Charset. [Issue #2760](https://github.com/PHPOffice/PhpSpreadsheet/issues/2760) [PR #4501](https://github.com/PHPOffice/PhpSpreadsheet/pull/4501)
+- AutoColor for LibreOffice Dark Mode [Discussion 4502](https://github.com/PHPOffice/PhpSpreadsheet/discussions/4502) [PR #4503](https://github.com/PHPOffice/PhpSpreadsheet/pull/4503)
+- Xlsx Style Writer Minor Refactoring. [PR #4508](https://github.com/PHPOffice/PhpSpreadsheet/pull/4508)
+- Allow Xlsx Reader to Specify ParseHuge. [Issue #4260](https://github.com/PHPOffice/PhpSpreadsheet/issues/4260) [PR #4515](https://github.com/PHPOffice/PhpSpreadsheet/pull/4515)
+
+## 2025-05-26 - 4.3.1
+
+### Fixed
+
+- Regression in Date::stringToExcel. [Issue #4488](https://github.com/PHPOffice/PhpSpreadsheet/issues/4488) [PR #4489](https://github.com/PHPOffice/PhpSpreadsheet/pull/4489)
+
+## 2025-05-25 - 4.3.0
+
+### Added
+
+- Xml Reader recognize indents. [Issue #4448](https://github.com/PHPOffice/PhpSpreadsheet/issues/4448) [PR #4449](https://github.com/PHPOffice/PhpSpreadsheet/pull/4449)
+
+### Changed
+
+- Phpstan Level 10.
+
+### Fixed
+
+- Micro-optimization for excelToDateTimeObject. [Issue #4438](https://github.com/PHPOffice/PhpSpreadsheet/issues/4438) [PR #4442](https://github.com/PHPOffice/PhpSpreadsheet/pull/4442)
+- Removing Columns/Rows Containing Merged Cells. [Issue #282](https://github.com/PHPOffice/PhpSpreadsheet/issues/282) [PR #4465](https://github.com/PHPOffice/PhpSpreadsheet/pull/4465)
+- Print Area and Row Break. [Issue #1275](https://github.com/PHPOffice/PhpSpreadsheet/issues/1275) [PR #4450](https://github.com/PHPOffice/PhpSpreadsheet/pull/4450)
+- Copy Styles after insertNewColumnBefore. [Issue #1425](https://github.com/PHPOffice/PhpSpreadsheet/issues/1425) [PR #4468](https://github.com/PHPOffice/PhpSpreadsheet/pull/4468)
+- Xls Writer Treat Hyperlink Starting with # as Internal. [Issue #56](https://github.com/PHPOffice/PhpSpreadsheet/issues/56) [PR #4453](https://github.com/PHPOffice/PhpSpreadsheet/pull/4453)
+- More Precision for Float to String Casts. [Issue #3899](https://github.com/PHPOffice/PhpSpreadsheet/issues/3899) [PR #4479](https://github.com/PHPOffice/PhpSpreadsheet/pull/4479)
+- Hyperlink Styles. [Issue #1632](https://github.com/PHPOffice/PhpSpreadsheet/issues/1632) [PR #4478](https://github.com/PHPOffice/PhpSpreadsheet/pull/4478)
+- ODS Handling of Ceiling and Floor. [Issue #477](https://github.com/PHPOffice/PhpSpreadsheet/issues/407) [PR #4466](https://github.com/PHPOffice/PhpSpreadsheet/pull/4466)
+- Xlsx Reader Do Not Process Printer Settings for Dataonly. [Issue #4477](https://github.com/PHPOffice/PhpSpreadsheet/issues/4477) [PR #4480](https://github.com/PHPOffice/PhpSpreadsheet/pull/4480)
+
+## 2025-04-16 - 4.2.0
+
+### Added
+
+- Add ability to add custom functions to Calculation. [PR #4390](https://github.com/PHPOffice/PhpSpreadsheet/pull/4390)
+- Add FormulaRange to IgnoredErrors. [PR #4393](https://github.com/PHPOffice/PhpSpreadsheet/pull/4393)
+- TextGrid improvements. [PR #4418](https://github.com/PHPOffice/PhpSpreadsheet/pull/4418)
+- Permit read to class which extends Spreadsheet. [Discussion #4402](https://github.com/PHPOffice/PhpSpreadsheet/discussions/4402) [PR #4404](https://github.com/PHPOffice/PhpSpreadsheet/pull/4404)
+- Conditional and table formatting support for html writer [PR #4412](https://github.com/PHPOffice/PhpSpreadsheet/pull/4412)
+
+### Changed
+
+- Phpstan Version 2. [PR #4384](https://github.com/PHPOffice/PhpSpreadsheet/pull/4384)
+- Start migration to Phpstan level 9. [PR #4396](https://github.com/PHPOffice/PhpSpreadsheet/pull/4396)
+- Calculation locale logic moved to separate class. [PR #4398](https://github.com/PHPOffice/PhpSpreadsheet/pull/4398)
+- TREND_POLYNOMIAL_* and TREND_BEST_FIT do not work, and are changed to throw Exceptions if attempted. (TREND_BEST_FIT_NO_POLY works.) An attempt to use an unknown trend type will now also throw an exception. [Issue #4400](https://github.com/PHPOffice/PhpSpreadsheet/issues/4400) [PR #4339](https://github.com/PHPOffice/PhpSpreadsheet/pull/4339)
+- Month parameter of DATE function will now return VALUE if an ordinal string (e.g. '3rd') is used, but will accept bool or null. [PR #4420](https://github.com/PHPOffice/PhpSpreadsheet/pull/4420)
+
+### Fixed
+
+- Ignore fractional part of Drawing Shadow Alpha. [Issue #4415](https://github.com/PHPOffice/PhpSpreadsheet/issues/4415) [PR #4417](https://github.com/PHPOffice/PhpSpreadsheet/pull/4417)
+- BIN2DEC, OCT2DEC, and HEX2DEC return numbers rather than strings. [Issue #4383](https://github.com/PHPOffice/PhpSpreadsheet/issues/4383) [PR #4389](https://github.com/PHPOffice/PhpSpreadsheet/pull/4389)
+- Fix TREND_BEST_FIT_NO_POLY. [Issue #4400](https://github.com/PHPOffice/PhpSpreadsheet/issues/4400) [PR #4339](https://github.com/PHPOffice/PhpSpreadsheet/pull/4339)
+- Ods Reader No DataType for Null Value. [Issue #4435](https://github.com/PHPOffice/PhpSpreadsheet/issues/4435) [PR #4436](https://github.com/PHPOffice/PhpSpreadsheet/pull/4436)
+- Column widths not preserved when using read filter. [Issue #4416](https://github.com/PHPOffice/PhpSpreadsheet/issues/4416) [PR #4423](https://github.com/PHPOffice/PhpSpreadsheet/pull/4423)
+- Fix typo in Style exportArray quotePrefix. [Issue #4422](https://github.com/PHPOffice/PhpSpreadsheet/issues/4422) [PR #4424](https://github.com/PHPOffice/PhpSpreadsheet/pull/4424)
+- Tweak Spreadsheet clone. [PR #4419](https://github.com/PHPOffice/PhpSpreadsheet/pull/4419)
+- Better handling of Chart DisplayBlanksAs. [Issue #4411](https://github.com/PHPOffice/PhpSpreadsheet/issues/4411) [PR #4414](https://github.com/PHPOffice/PhpSpreadsheet/pull/4414)
+
+## 2025-03-02 - 4.1.0
+
+### Added
+
+- Support Justify Last Line. [Issue #4374](https://github.com/PHPOffice/PhpSpreadsheet/issues/4374) [PR #4373](https://github.com/PHPOffice/PhpSpreadsheet/pull/4373)
+- Allow Spreadsheet clone. [PR #4370](https://github.com/PHPOffice/PhpSpreadsheet/pull/4370)
+
+### Changed
+
+- ListWorksheetInfo will now return sheetState (visible, hidden, veryHidden). [Issue #4345](https://github.com/PHPOffice/PhpSpreadsheet/issues/4345) [PR #4366](https://github.com/PHPOffice/PhpSpreadsheet/pull/4366)
+- Start migration to Phpstan 2. [PR #4359](https://github.com/PHPOffice/PhpSpreadsheet/pull/4359)
+- IOFactory identify can return, and createReader and CreateWriter can accept, a class name rather than a file type. [Issue #4357](https://github.com/PHPOffice/PhpSpreadsheet/issues/4357) [PR #4361](https://github.com/PHPOffice/PhpSpreadsheet/pull/4361)
+
+### Fixed
+
+- Refactor Helper/Html. [PR #4359](https://github.com/PHPOffice/PhpSpreadsheet/pull/4359)
+- Handle #REF! as Argument to AVERAGEIF/COUNTIF/SUMIF. [Issue #4381](https://github.com/PHPOffice/PhpSpreadsheet/issues/4381) [PR #4382](https://github.com/PHPOffice/PhpSpreadsheet/pull/4382)
+- Ignore ignoredErrors when not applicable. [Issue #4375](https://github.com/PHPOffice/PhpSpreadsheet/issues/4375) [PR #4377](https://github.com/PHPOffice/PhpSpreadsheet/pull/4377)
+- Better handling of defined names on sheets whose titles include apostrophes. [Issue #4356](https://github.com/PHPOffice/PhpSpreadsheet/issues/4356) [Issue #4362](https://github.com/PHPOffice/PhpSpreadsheet/issues/4362) [Issue #4376](https://github.com/PHPOffice/PhpSpreadsheet/issues/4376) [PR #4360](https://github.com/PHPOffice/PhpSpreadsheet/pull/4360)
+- Partial solution for removing rows or columns that include edge ranges. [Issue #1449](https://github.com/PHPOffice/PhpSpreadsheet/issues/1449) [PR #3528](https://github.com/PHPOffice/PhpSpreadsheet/pull/3528)
+- Prefer mb_str_split to str_split. [PR #3341](https://github.com/PHPOffice/PhpSpreadsheet/pull/3341)
+
+## 2025-02-08 - 4.0.0
+
+### BREAKING CHANGES
+
+- Data Validations will be stored by worksheet, not cell. Index can be one or more cells or cell ranges. [Issue #797](https://github.com/PHPOffice/PhpSpreadsheet/issues/797) [Issue #4091](https://github.com/PHPOffice/PhpSpreadsheet/issues/4091) [Issue #4206](https://github.com/PHPOffice/PhpSpreadsheet/issues/4206) [PR #4240](https://github.com/PHPOffice/PhpSpreadsheet/pull/4240)
+- Conditional Formatting adds Priority property and handles overlapping ranges better. [Issue #4312](https://github.com/PHPOffice/PhpSpreadsheet/issues/4312) [Issue #4318](https://github.com/PHPOffice/PhpSpreadsheet/issues/4318) [PR #4314](https://github.com/PHPOffice/PhpSpreadsheet/pull/4314)
+- Csv Reader will no longer auto-detect Mac line endings by default. Prior behavior can be explicitly enabled via `setTestAutoDetect(true)`, and it will not be possible at all with Php9+. [Issue #4092](https://github.com/PHPOffice/PhpSpreadsheet/issues/4092) [PR #4340](https://github.com/PHPOffice/PhpSpreadsheet/pull/4340)
+- Html Writer will now use "better boolean" logic. Booleans will now be output by default as TRUE/FALSE rather than 1/null-string. Prior behavior can be explicitly enabled via `setBetterBoolean(false)`. [PR #4340](https://github.com/PHPOffice/PhpSpreadsheet/pull/4340)
+- Xlsx Writer will now use false as the default for `forceFullCalc`. This affects writes with `preCalculateFormulas` set to false. Prior behavior can be explicitly enabled via `setForceFullCalc(null)`.[PR #4340](https://github.com/PHPOffice/PhpSpreadsheet/pull/4340)
+- Deletion of items deprecated in Release 3. See "removed" below.
+
+### Added
+
+- Pdf Charts and Drawings. [Discussion #4129](https://github.com/PHPOffice/PhpSpreadsheet/discussions/4129) [Discussion #4168](https://github.com/PHPOffice/PhpSpreadsheet/discussions/4168) [PR #4327](https://github.com/PHPOffice/PhpSpreadsheet/pull/4327)
+- Allow spreadsheet serialization. [Discussion #4324](https://github.com/PHPOffice/PhpSpreadsheet/discussions/4324) [Issue #1741](https://github.com/PHPOffice/PhpSpreadsheet/issues/1741) [Issue #1757](https://github.com/PHPOffice/PhpSpreadsheet/issues/1757) [PR #4326](https://github.com/PHPOffice/PhpSpreadsheet/pull/4326)
+
+### Removed
+
+- Worksheet::getStyles - no replacement. [PR #4330](https://github.com/PHPOffice/PhpSpreadsheet/pull/4330)
+- The following items were deprecated in release 3 and are now removed.
+- Drawing::setIsUrl - no replacement.
+- Settings::setLibXmlLoaderOptions() and Settings::getLibXmlLoaderOptions() - no replacement.
+- Worksheet::getHashCode - no replacement.
+- IReader::SKIP_EMPTY_CELLS - use its alias IGNORE_EMPTY_CELLS instead.
+- Worksheet::getProtectedCells - use getProtectedCellRanges instead.
+- Writer/Html::isMpdf property - use instanceof Mpdf instead.
+
+### Changed
+
+- Nothing yet.
+
+### Moved
+
+- Nothing yet.
+
+### Deprecated
+
+- Nothing yet.
+
+### Fixed
+
+- Xls writer Parser Mishandling True/False Argument. [Issue #4331](https://github.com/PHPOffice/PhpSpreadsheet/issues/4331) [PR #4333](https://github.com/PHPOffice/PhpSpreadsheet/pull/4333)
+- Xls writer Parser Parse By Character Not Byte. [PR #4344](https://github.com/PHPOffice/PhpSpreadsheet/pull/4344)
+- Minor changes to dynamic array calculations exposed by using explicit array return types in some tests. [PR #4328](https://github.com/PHPOffice/PhpSpreadsheet/pull/4328)
+
+## 2025-01-26 - 3.9.0
+
+### Added
+
+- Methods to get style for row or column. [PR #4317](https://github.com/PHPOffice/PhpSpreadsheet/pull/4317)
+- Method for duplicating worksheet in spreadsheet. [PR #4315](https://github.com/PHPOffice/PhpSpreadsheet/pull/4315)
+
+### Fixed
+
+- Security patch for control characters in protocol.
+- Ods Reader Sheet Names with Period. [Issue #4311](https://github.com/PHPOffice/PhpSpreadsheet/issues/4311) [PR #4313](https://github.com/PHPOffice/PhpSpreadsheet/pull/4313)
+- Mpdf and Tcpdf Hidden Columns and Merged Cells. [Issue #4319](https://github.com/PHPOffice/PhpSpreadsheet/issues/4319) [PR #4320](https://github.com/PHPOffice/PhpSpreadsheet/pull/4320)
+- Html Writer Allow mailto. [Issue #4316](https://github.com/PHPOffice/PhpSpreadsheet/issues/4316) [PR #4322](https://github.com/PHPOffice/PhpSpreadsheet/pull/4322)
+- Use composer/pcre rather than preg_* in Writer. [PR #4323](https://github.com/PHPOffice/PhpSpreadsheet/pull/4323)
+
+## 2025-01-11 - 3.8.0
+
+### Added
+
+- CHOOSECOLS, CHOOSEROWS, DROP, TAKE, and EXPAND. [PR #4286](https://github.com/PHPOffice/PhpSpreadsheet/pull/4286)
+
+### Fixed
+
+- Security patch for Html navigation.
+- Xlsx Reader Shared Formula with Boolean Result. Partial solution for [Issue #4280](https://github.com/PHPOffice/PhpSpreadsheet/issues/4280) [PR #4281](https://github.com/PHPOffice/PhpSpreadsheet/pull/4281)
+- Retitling cloned Worksheets. [Issue #641](https://github.com/PHPOffice/PhpSpreadsheet/issues/641) [PR #4302](https://github.com/PHPOffice/PhpSpreadsheet/pull/4302)
+- Extremely limited support for GROUPBY function. Partial response to [Issue #4282](https://github.com/PHPOffice/PhpSpreadsheet/issues/4282) [PR #4283](https://github.com/PHPOffice/PhpSpreadsheet/pull/4283)
+
+## 2024-12-26 - 3.7.0
### Deprecated
@@ -40,58 +271,145 @@ and this project adheres to [Semantic Versioning](https://semver.org).
### Fixed
-- More context options may be needed for http(s) image. Backport of [PR #4276](https://github.com/PHPOffice/PhpSpreadsheet/pull/4276)
-- Backported security patches for Samples.
-- Backported security patches for Html Writer.
+- Security patches for Samples.
+- Security patches for Html Writer.
+- Avoid unexpected charset in currency symbol. [PR #4279](https://github.com/PHPOffice/PhpSpreadsheet/pull/4279)
+- Add forceFullCalc option to Xlsx Writer. [Issue #4269](https://github.com/PHPOffice/PhpSpreadsheet/issues/4269) [PR #4271](https://github.com/PHPOffice/PhpSpreadsheet/pull/4271)
+- More context options may be needed for http(s) image. [Php issue 17121](https://github.com/php/php-src/issues/17121) [PR #4276](https://github.com/PHPOffice/PhpSpreadsheet/pull/4276)
+- Coverage-related tweaks to Xls Reader. [PR #4277](https://github.com/PHPOffice/PhpSpreadsheet/pull/4277)
+- Several fixed to ODS Writer. [Issue #4261](https://github.com/PHPOffice/PhpSpreadsheet/issues/4261) [PR #4263](https://github.com/PHPOffice/PhpSpreadsheet/pull/4263) [PR #4264](https://github.com/PHPOffice/PhpSpreadsheet/pull/4264) [PR #4266](https://github.com/PHPOffice/PhpSpreadsheet/pull/4266)
-## 2024-12-08 - 2.3.4
+## 2024-12-08 - 3.6.0
-### Fixed
+### Added
-- Fix Minor Break Handling Drawings. Backport of [PR #4244](https://github.com/PHPOffice/PhpSpreadsheet/pull/4244)
-- Swapped Row and Column Indexes in Reference Helper. Backport of [PR #4247](https://github.com/PHPOffice/PhpSpreadsheet/pull/4247)
-- Upgrade locked version of Dompdf (Php8.4 compatibility).
-- Remove unnecessary files from Composer package.
-
-## 2024-11-22 - 2.3.3
+- Nothing yet.
### Changed
-- Settings::libXmlLoaderOptions is ignored. Backport of [PR #4233](https://github.com/PHPOffice/PhpSpreadsheet/pull/4233)
+- Nothing yet.
+
+### Moved
+
+- Nothing yet.
+
+### Deprecated
+
+- Nothing yet.
+
+### Fixed
+
+- Html Reader/Writer Better Handling of Booleans. [PR #4257](https://github.com/PHPOffice/PhpSpreadsheet/pull/4257)
+- Fill Patterns/Colors When Xml Attributes are Missing. [Issue #4248](https://github.com/PHPOffice/PhpSpreadsheet/issues/4248) [PR #4250](https://github.com/PHPOffice/PhpSpreadsheet/pull/4250)
+- Remove Unneccesary files from Composer Package. [PR #4262](https://github.com/PHPOffice/PhpSpreadsheet/pull/4262)
+- Swapped row and column indexes in ReferenceHelper. [Issue #4246](https://github.com/PHPOffice/PhpSpreadsheet/issues/4246) [PR #4247](https://github.com/PHPOffice/PhpSpreadsheet/pull/4247)
+- Fix minor break handling drawings. [Issue #4241](https://github.com/PHPOffice/PhpSpreadsheet/issues/4241) [PR #4244](https://github.com/PHPOffice/PhpSpreadsheet/pull/4244)
+- Ignore cell formatting when the format is a single @. [Issue #4242](https://github.com/PHPOffice/PhpSpreadsheet/issues/4242) [PR #4243](https://github.com/PHPOffice/PhpSpreadsheet/pull/4243)
+- Upgrade Dompdf to Php-8.4 compatible version [PR #4267](https://github.com/PHPOffice/PhpSpreadsheet/pull/4267)
+
+## 2024-11-22 - 3.5.0
+
+### Added
+
+- Nothing yet.
+
+### Changed
+
+- Settings::libXmlLoaderOptions is ignored. [PR #4233](https://github.com/PHPOffice/PhpSpreadsheet/pull/4233)
+
+### Moved
+
+- Nothing yet.
### Deprecated
- Settings::setLibXmlLoaderOptions() and Settings::getLibXmlLoaderOptions() are no longer needed - no replacement.
-
-## 2024-11-10 - 2.3.2
+- Worksheet::getHashCode is no longer needed.
### Fixed
-- 2.3.1 omitted.
-- Backported security patches.
-- Write ignoredErrors Tag Before Drawings. Backport of [PR #4212](https://github.com/PHPOffice/PhpSpreadsheet/pull/4212) intended for 3.4.0.
-- Changes to ROUNDDOWN/ROUNDUP/TRUNC. Backport of [PR #4214](https://github.com/PHPOffice/PhpSpreadsheet/pull/4214) intended for 3.4.0.
+- Add support for `` tag when converting HTML to RichText. [Issue #4223](https://github.com/PHPOffice/PhpSpreadsheet/issues/4223) [PR #4224](https://github.com/PHPOffice/PhpSpreadsheet/pull/4224)
+- Change hash code for worksheet. [Issue #4192](https://github.com/PHPOffice/PhpSpreadsheet/issues/4192) [PR #4207](https://github.com/PHPOffice/PhpSpreadsheet/pull/4207)
+
+## 2024-11-10 - 3.4.0
+
+### Security Fix
+
+- Several security patches.
### Added
-- Method to Test Whether Csv Will Be Affected by Php9. Backport of [PR #4189](https://github.com/PHPOffice/PhpSpreadsheet/pull/4189) intended for 3.4.0.
-
-## 2024-09-29 - 2.3.0
-
-### Fixed
-
-- Backported security patches.
-- Improve Xlsx Reader speed (backport of PR #4153 intended for 3.0.0). [Issue #3917](https://github.com/PHPOffice/PhpSpreadsheet/issues/3917)
-- Change to Csv Reader (see below under Deprecated). Backport of PR #4162 intended for 3.0.0. [Issue #4161](https://github.com/PHPOffice/PhpSpreadsheet/issues/4161)
-- Tweak to AMORDEGRC. Backport of PR #4164 intended for 3.0.0.
+- Add Dynamic valueBinder Property to Spreadsheet and Readers. [Issue #1395](https://github.com/PHPOffice/PhpSpreadsheet/issues/1395) [PR #4185](https://github.com/PHPOffice/PhpSpreadsheet/pull/4185)
+- Allow Omitting Chart Border. [Issue #562](https://github.com/PHPOffice/PhpSpreadsheet/issues/562) [PR #4188](https://github.com/PHPOffice/PhpSpreadsheet/pull/4188)
+- Method to Test Whether Csv Will Be Affected by Php9. [PR #4189](https://github.com/PHPOffice/PhpSpreadsheet/pull/4189)
### Changed
+- Refactor Xls Reader. [PR #4118](https://github.com/PHPOffice/PhpSpreadsheet/pull/4118)
+
+### Deprecated
+
+- IReader::SKIP_EMPTY_CELLS - use its alias IGNORE_EMPTY_CELLS instead.
+- Worksheet::getProtectedCells was deprecated in release 2, but was not properly documented, and not removed in release 3. Use getProtectedCellRanges instead.
+- Writer/Html::isMpdf property was deprecated in release 2, but was not properly documented, and not removed in release 3. Use instanceof Mpdf instead.
+
+### Moved
+
+- Nothing yet.
+
+### Fixed
+
+- Xls Writer Condtional Rules Applied to Whole Rows or Columns. [Issue #3185](https://github.com/PHPOffice/PhpSpreadsheet/issues/3185) [PR #4152](https://github.com/PHPOffice/PhpSpreadsheet/pull/4152)
+- Xlsx Writer Duplicate ContentTypes Entry for Background Image. [Issue #4179](https://github.com/PHPOffice/PhpSpreadsheet/issues/4179) [PR #4180](https://github.com/PHPOffice/PhpSpreadsheet/pull/4180)
+- Check strictNullComparison outside of loops. [PR #3347](https://github.com/PHPOffice/PhpSpreadsheet/pull/3347)
+- SUMIFS Does Not Require xlfn. [Issue #4182](https://github.com/PHPOffice/PhpSpreadsheet/issues/4182) [PR #4186](https://github.com/PHPOffice/PhpSpreadsheet/pull/4186)
+- Image Transparency/Opacity with Html Reader Changes. [Discussion #4117](https://github.com/PHPOffice/PhpSpreadsheet/discussions/4117) [PR #4142](https://github.com/PHPOffice/PhpSpreadsheet/pull/4142)
+- Option to Write Hyperlink Rather Than Label to Csv. [Issue #1412](https://github.com/PHPOffice/PhpSpreadsheet/issues/1412) [PR #4151](https://github.com/PHPOffice/PhpSpreadsheet/pull/4151)
+- Invalid Html Due to Cached Filesize. [Issue #1107](https://github.com/PHPOffice/PhpSpreadsheet/issues/1107) [PR #4184](https://github.com/PHPOffice/PhpSpreadsheet/pull/4184)
+- Excel 2003 Allows Html Entities. [Issue #2157](https://github.com/PHPOffice/PhpSpreadsheet/issues/2157) [PR #4187](https://github.com/PHPOffice/PhpSpreadsheet/pull/4187)
+- Changes to ROUNDDOWN/ROUNDUP/TRUNC. [Issue #4213](https://github.com/PHPOffice/PhpSpreadsheet/issues/4213) [PR #4214](https://github.com/PHPOffice/PhpSpreadsheet/pull/4214)
+- Writer Xlsx ignoredErrors Before Drawings. [Issue #4200](https://github.com/PHPOffice/PhpSpreadsheet/issues/4200) [Issue #4145](https://github.com/PHPOffice/PhpSpreadsheet/issues/4145) [PR #4212](https://github.com/PHPOffice/PhpSpreadsheet/pull/4212)
+- Allow ANCHORARRAY as Data Validation list. [Issue #4197](https://github.com/PHPOffice/PhpSpreadsheet/issues/4197) [PR #4203](https://github.com/PHPOffice/PhpSpreadsheet/pull/4203)
+
+## 2024-09-29 - 3.3.0 (no 3.0.\*, 3.1.\*, 3.2.\*)
+
+### Dynamic Arrays
+
+- Support for Excel dynamic arrays is added. It is an opt-in feature, so our hope is that there will be no BC breaks, but it is a very large change. Full support is added for Xlsx. It is emulated as Ctrl-Shift-Enter arrays for Ods read and write and Excel2003 and Gnumeric read. Html/Pdf and Csv writers will populate cells on output if they are the result of array formulas. No support is added for Xls or Slk.
+
+### Added
+
+- Excel Dynamic Arrays. [Issue #3901](https://github.com/PHPOffice/PhpSpreadsheet/issues/3901) [Issue #3659](https://github.com/PHPOffice/PhpSpreadsheet/issues/3659) [Issue #1834](https://github.com/PHPOffice/PhpSpreadsheet/issues/1834) [PR #3962](https://github.com/PHPOffice/PhpSpreadsheet/pull/3962)
+- String Value Binder Allow Setting "Ignore Number Stored as Text". [PR #4141](https://github.com/PHPOffice/PhpSpreadsheet/pull/4141)
+
+### Changed
+
+- Xlsx Reader default datatype when none is specified in Xml is changed from string to numeric, which is how Excel treats it. There is expected to be little impact because DefaultValueBinder and AdvancedValueBinder correct mis-identification as string, and StringValueBinder usually expects string. [PR #4139](https://github.com/PHPOffice/PhpSpreadsheet/pull/4139)
+- Currency and Accounting Wizards are changed to act like Excel, and a new CurrencyBase Wizard is added for for non-Excel formats. [Issue #4125](https://github.com/PHPOffice/PhpSpreadsheet/issues/4125) [Issue #4124](https://github.com/PHPOffice/PhpSpreadsheet/issues/4124) [PR #4127](https://github.com/PHPOffice/PhpSpreadsheet/pull/4127)
- Images will not be added to spreadsheet if they cannot be validated as images.
### Deprecated
-- Php8.4 will deprecate the escape parameter of fgetcsv. Csv Reader is affected by this; code is changed to be unaffected, but this will mean a breaking change is coming with Php9. Any code which uses the default escape value of backslash will fail in Php9. It is recommended to explicitly set the escape value to null string before then.
+- Nothing yet.
+
+### Removed
+
+- The following items were deprecated in release 2 and are now removed.
+- Writer\Xls\Style\ColorMap (no longer needed).
+- Reader\Xml::trySimpleXMLLoadString (should not have been public, no public replacement).
+- Calculation\Calculation::_translateFormulaToLocale (use method name translateFormulaToLocale without leading underscore).
+- Calculation\Calculation::_translateFormulaToEnglish (use method name translateFormulaToEnglish without leading underscore).
+
+### Moved
+
+- Nothing yet.
+
+### Fixed
+
+- Several security patches.
+- Xls Reader Some Ranges Not Handled Properly. [Issue #1570](https://github.com/PHPOffice/PhpSpreadsheet/issues/1570) [PR #4140](https://github.com/PHPOffice/PhpSpreadsheet/pull/4140)
+- Better Handling of legacyDrawing Xml. [Issue #4105](https://github.com/PHPOffice/PhpSpreadsheet/issues/4105) [PR #4122](https://github.com/PHPOffice/PhpSpreadsheet/pull/4122)
+- Improve Xlsx Reader Speed. [Issue #3917](https://github.com/PHPOffice/PhpSpreadsheet/issues/3917) [PR #4153](https://github.com/PHPOffice/PhpSpreadsheet/pull/4153)
## 2024-08-07 - 2.2.2
diff --git a/api/vendor/phpoffice/phpspreadsheet/CONTRIBUTING.md b/api/vendor/phpoffice/phpspreadsheet/CONTRIBUTING.md
index e89e99ec..6cbf9fb4 100644
--- a/api/vendor/phpoffice/phpspreadsheet/CONTRIBUTING.md
+++ b/api/vendor/phpoffice/phpspreadsheet/CONTRIBUTING.md
@@ -4,10 +4,11 @@ If you would like to contribute, here are some notes and guidelines:
- All new development should be on feature/fix branches, which are then merged to the `master` branch once stable and approved; so the `master` branch is always the most up-to-date, working code
- If you are going to submit a pull request, please fork from `master`, and submit your pull request back as a fix/feature branch referencing the GitHub issue number
+ - Install (development) dependencies by running `composer install` inside your PhpSpreadsheet clone.
- The code must work with all PHP versions that we support.
- You can call `composer versions` to test version compatibility.
- Code style should be maintained.
- - `composer style` will identify any issues with Coding Style`.
+ - `composer style` will identify any issues with Coding Style.
- `composer fix` will fix most issues with Coding Style.
- All code changes must be validated by `composer check`.
- Please include Unit Tests to verify that a bug exists, and that this PR fixes it.
@@ -39,7 +40,10 @@ This makes it easier to see exactly what is being tested when reviewing the PR.
2. Tag subject must be the version number, eg: `1.2.3`
3. Tag body must be a copy-paste of the changelog entries.
3. Push the tag with `git push --tags`, GitHub Actions will create a GitHub release automatically, and the release details will automatically be sent to packagist.
-4. Github seems to remove markdown headings in the Release Notes, so you should edit to restore these.
-
-> **Note:** Tagged releases are made from the `master` branch. Only in an emergency should a tagged release be made from the `release` branch. (i.e. cherry-picked hot-fixes.)
+4. By default, Github removes markdown headings in the Release Notes. You can either edit to restore these, or, probably preferably, change the default comment character on your system - `git config core.commentChar ";"`.
+> **Note:** Tagged releases are made from the `master` branch. Only in an emergency should a tagged release be made from the `release` branch. (i.e. cherry-picked hot-fixes.) However, there are 4 branches which have been updated to apply security patches, and those may be tagged if future security updates are needed.
+- release1291 (no further updates aside from security patches, including code changes needed for Php 8.5 compatibility)
+- release210 (no further updates aside from security patches, including code changes needed for Php 8.5 compatibility)
+- release222
+- release390
diff --git a/api/vendor/phpoffice/phpspreadsheet/README.md b/api/vendor/phpoffice/phpspreadsheet/README.md
index 84b4b7be..8286e3a5 100644
--- a/api/vendor/phpoffice/phpspreadsheet/README.md
+++ b/api/vendor/phpoffice/phpspreadsheet/README.md
@@ -1,8 +1,7 @@
# PhpSpreadsheet
[](https://github.com/PHPOffice/PhpSpreadsheet/actions)
-[](https://scrutinizer-ci.com/g/PHPOffice/PhpSpreadsheet/?branch=master)
-[](https://scrutinizer-ci.com/g/PHPOffice/PhpSpreadsheet/?branch=master)
+[](https://coveralls.io/github/PHPOffice/PhpSpreadsheet?branch=master)
[](https://packagist.org/packages/phpoffice/phpspreadsheet)
[](https://packagist.org/packages/phpoffice/phpspreadsheet)
[](https://packagist.org/packages/phpoffice/phpspreadsheet)
@@ -11,6 +10,17 @@
PhpSpreadsheet is a library written in pure PHP and offers a set of classes that
allow you to read and write various spreadsheet file formats such as Excel and LibreOffice Calc.
+This is the master branch, and is maintained for security and bug fixes.
+
+## PHP Version Support
+
+LTS: For maintained branches, support for PHP versions will only be maintained for a period of six months beyond the
+[end of life](https://www.php.net/supported-versions) of that PHP version.
+
+Currently the required PHP minimum version is PHP __8.1__, and we [will support that version](https://www.php.net/supported-versions.php) until 30th June 2026.
+
+See the `composer.json` for other requirements.
+
## Installation
See the [install instructions](https://phpspreadsheet.readthedocs.io/en/latest/#installation).
diff --git a/api/vendor/phpoffice/phpspreadsheet/composer.json b/api/vendor/phpoffice/phpspreadsheet/composer.json
index cb00ebf4..bf03b9a7 100644
--- a/api/vendor/phpoffice/phpspreadsheet/composer.json
+++ b/api/vendor/phpoffice/phpspreadsheet/composer.json
@@ -15,6 +15,7 @@
"platform": {
"php" : "8.1.99"
},
+ "process-timeout": 600,
"sort-packages": true,
"allow-plugins": {
"dealerdirect/phpcodesniffer-composer-installer": true
@@ -45,12 +46,12 @@
],
"scripts": {
"check": [
- "./bin/check-phpdoc-types",
+ "php bin/check-phpdoc-types.php",
"phpcs samples/ src/ tests/ --report=checkstyle",
- "phpcs samples/ src/ tests/ --standard=PHPCompatibility --runtime-set testVersion 8.0- -n",
+ "phpcs samples/ src/ tests/ --standard=PHPCompatibility --runtime-set testVersion 8.0- --exclude=PHPCompatibility.Variables.ForbiddenThisUseContexts -n",
"php-cs-fixer fix --ansi --dry-run --diff",
- "phpunit --color=always",
- "phpstan analyse --ansi --memory-limit=2048M"
+ "phpstan analyse --ansi --memory-limit=2048M",
+ "phpunit --color=always"
],
"style": [
"phpcs samples/ src/ tests/ --report=checkstyle",
@@ -61,7 +62,7 @@
"php-cs-fixer fix"
],
"versions": [
- "phpcs samples/ src/ tests/ --standard=PHPCompatibility --runtime-set testVersion 8.0- -n"
+ "phpcs samples/ src/ tests/ --standard=PHPCompatibility --runtime-set testVersion 8.0- --exclude=PHPCompatibility.Variables.ForbiddenThisUseContexts -n"
]
},
"require": {
@@ -79,7 +80,7 @@
"ext-xmlwriter": "*",
"ext-zip": "*",
"ext-zlib": "*",
- "composer/pcre": "^1 || ^2 || ^3",
+ "composer/pcre": "^1||^2||^3",
"maennchen/zipstream-php": "^2.1 || ^3.0",
"markbaker/complex": "^3.0",
"markbaker/matrix": "^3.0",
@@ -94,9 +95,10 @@
"mitoteam/jpgraph": "^10.3",
"mpdf/mpdf": "^8.1.1",
"phpcompatibility/php-compatibility": "^9.3",
- "phpstan/phpstan": "^1.1",
- "phpstan/phpstan-phpunit": "^1.0",
- "phpunit/phpunit": "^9.6 || ^10.5",
+ "phpstan/phpstan": "^1.1 || ^2.0",
+ "phpstan/phpstan-phpunit": "^1.0 || ^2.0",
+ "phpstan/phpstan-deprecation-rules": "^1.0 || ^2.0",
+ "phpunit/phpunit": "^10.5",
"squizlabs/php_codesniffer": "^3.7",
"tecnickcom/tcpdf": "^6.5"
},
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/ArrayEnabled.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/ArrayEnabled.php
index 7b78b6f5..038b7161 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/ArrayEnabled.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/ArrayEnabled.php
@@ -12,21 +12,25 @@ trait ArrayEnabled
private static ArrayArgumentHelper $arrayArgumentHelper;
/**
- * @param array|false $arguments Can be changed to array for Php8.1+
+ * @param mixed[] $arguments
*/
- private static function initialiseHelper($arguments): void
+ private static function initialiseHelper(array $arguments): void
{
if (self::$initializationNeeded === true) {
self::$arrayArgumentHelper = new ArrayArgumentHelper();
self::$initializationNeeded = false;
}
- self::$arrayArgumentHelper->initialise(($arguments === false) ? [] : $arguments);
+ self::$arrayArgumentHelper->initialise($arguments);
}
/**
* Handles array argument processing when the function accepts a single argument that can be an array argument.
* Example use for:
* DAYOFMONTH() or FACT().
+ *
+ * @param mixed[] $values
+ *
+ * @return mixed[]
*/
protected static function evaluateSingleArgumentArray(callable $method, array $values): array
{
@@ -43,6 +47,8 @@ trait ArrayEnabled
* and any of them can be an array argument.
* Example use for:
* ROUND() or DATE().
+ *
+ * @return mixed[]
*/
protected static function evaluateArrayArguments(callable $method, mixed ...$arguments): array
{
@@ -58,6 +64,8 @@ trait ArrayEnabled
* Example use for:
* NETWORKDAYS() or CONCATENATE(), where the last argument is a matrix (or a series of values) that need
* to be treated as a such rather than as an array arguments.
+ *
+ * @return mixed[]
*/
protected static function evaluateArrayArgumentsSubset(callable $method, int $limit, mixed ...$arguments): array
{
@@ -80,6 +88,8 @@ trait ArrayEnabled
* Example use for:
* Z.TEST() or INDEX(), where the first argument 1 is a matrix that needs to be treated as a dataset
* rather than as an array argument.
+ *
+ * @return mixed[]
*/
protected static function evaluateArrayArgumentsSubsetFrom(callable $method, int $start, mixed ...$arguments): array
{
@@ -105,6 +115,8 @@ trait ArrayEnabled
* Example use for:
* HLOOKUP() and VLOOKUP(), where argument 1 is a matrix that needs to be treated as a database
* rather than as an array argument.
+ *
+ * @return mixed[]
*/
protected static function evaluateArrayArgumentsIgnore(callable $method, int $ignore, mixed ...$arguments): array
{
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/BinaryComparison.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/BinaryComparison.php
index e4bc156a..1f697946 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/BinaryComparison.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/BinaryComparison.php
@@ -14,13 +14,15 @@ class BinaryComparison
/**
* Compare two strings in the same way as strcmp() except that lowercase come before uppercase letters.
*
- * @param null|string $str1 First string value for the comparison
- * @param null|string $str2 Second string value for the comparison
+ * @param mixed $str1 First string value for the comparison, expect ?string
+ * @param mixed $str2 Second string value for the comparison, expect ?string
*/
- private static function strcmpLowercaseFirst(?string $str1, ?string $str2): int
+ private static function strcmpLowercaseFirst(mixed $str1, mixed $str2): int
{
- $inversedStr1 = StringHelper::strCaseReverse($str1 ?? '');
- $inversedStr2 = StringHelper::strCaseReverse($str2 ?? '');
+ $str1 = StringHelper::convertToString($str1);
+ $str2 = StringHelper::convertToString($str2);
+ $inversedStr1 = StringHelper::strCaseReverse($str1);
+ $inversedStr2 = StringHelper::strCaseReverse($str2);
return strcmp($inversedStr1, $inversedStr2);
}
@@ -28,12 +30,15 @@ class BinaryComparison
/**
* PHP8.1 deprecates passing null to strcmp.
*
- * @param null|string $str1 First string value for the comparison
- * @param null|string $str2 Second string value for the comparison
+ * @param mixed $str1 First string value for the comparison, expect ?string
+ * @param mixed $str2 Second string value for the comparison, expect ?string
*/
- private static function strcmpAllowNull(?string $str1, ?string $str2): int
+ private static function strcmpAllowNull(mixed $str1, mixed $str2): int
{
- return strcmp($str1 ?? '', $str2 ?? '');
+ $str1 = StringHelper::convertToString($str1);
+ $str2 = StringHelper::convertToString($str2);
+
+ return strcmp($str1, $str2);
}
public static function compare(mixed $operand1, mixed $operand2, string $operator): bool
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Calculation.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Calculation.php
index 8d622d4e..c5772492 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Calculation.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Calculation.php
@@ -15,15 +15,16 @@ use PhpOffice\PhpSpreadsheet\Cell\DataType;
use PhpOffice\PhpSpreadsheet\DefinedName;
use PhpOffice\PhpSpreadsheet\NamedRange;
use PhpOffice\PhpSpreadsheet\ReferenceHelper;
-use PhpOffice\PhpSpreadsheet\Shared;
+use PhpOffice\PhpSpreadsheet\Shared\StringHelper;
use PhpOffice\PhpSpreadsheet\Spreadsheet;
use PhpOffice\PhpSpreadsheet\Worksheet\Worksheet;
use ReflectionClassConstant;
use ReflectionMethod;
use ReflectionParameter;
use Throwable;
+use TypeError;
-class Calculation
+class Calculation extends CalculationLocale
{
/** Constants */
/** Regular Expressions */
@@ -34,11 +35,11 @@ class Calculation
// Opening bracket
const CALCULATION_REGEXP_OPENBRACE = '\(';
// Function (allow for the old @ symbol that could be used to prefix a function, but we'll ignore it)
- const CALCULATION_REGEXP_FUNCTION = '@?(?:_xlfn\.)?(?:_xlws\.)?([\p{L}][\p{L}\p{N}\.]*)[\s]*\(';
- // Strip xlfn and xlws prefixes from function name
- const CALCULATION_REGEXP_STRIP_XLFN_XLWS = '/(_xlfn[.])?(_xlws[.])?(?=[\p{L}][\p{L}\p{N}\.]*[\s]*[(])/';
+ const CALCULATION_REGEXP_FUNCTION = '@?(?:_xlfn\.)?(?:_xlws\.)?((?:__xludf\.)?[\p{L}][\p{L}\p{N}\.]*)[\s]*\(';
// Cell reference (cell or range of cells, with or without a sheet reference)
const CALCULATION_REGEXP_CELLREF = '((([^\s,!&%^\/\*\+<>=:`-]*)|(\'(?:[^\']|\'[^!])+?\')|(\"(?:[^\"]|\"[^!])+?\"))!)?\$?\b([a-z]{1,3})\$?(\d{1,7})(?![\w.])';
+ // Used only to detect spill operator #
+ const CALCULATION_REGEXP_CELLREF_SPILL = '/' . self::CALCULATION_REGEXP_CELLREF . '#/i';
// Cell reference (with or without a sheet reference) ensuring absolute/relative
const CALCULATION_REGEXP_CELLREF_RELATIVE = '((([^\s\(,!&%^\/\*\+<>=:`-]*)|(\'(?:[^\']|\'[^!])+?\')|(\"(?:[^\"]|\"[^!])+?\"))!)?(\$?\b[a-z]{1,3})(\$?\d{1,7})(?![\w.])';
const CALCULATION_REGEXP_COLUMN_RANGE = '(((([^\s\(,!&%^\/\*\+<>=:`-]*)|(\'(?:[^\']|\'[^!])+?\')|(\".(?:[^\"]|\"[^!])?\"))!)?(\$?[a-z]{1,3})):(?![.*])';
@@ -50,7 +51,7 @@ class Calculation
// Defined Names: Named Range of cells, or Named Formulae
const CALCULATION_REGEXP_DEFINEDNAME = '((([^\s,!&%^\/\*\+<>=-]*)|(\'(?:[^\']|\'[^!])+?\')|(\"(?:[^\"]|\"[^!])+?\"))!)?([_\p{L}][_\p{L}\p{N}\.]*)';
// Structured Reference (Fully Qualified and Unqualified)
- const CALCULATION_REGEXP_STRUCTURED_REFERENCE = '([\p{L}_\\\\][\p{L}\p{N}\._]+)?(\[(?:[^\d\]+-])?)';
+ const CALCULATION_REGEXP_STRUCTURED_REFERENCE = '([\p{L}_\\\][\p{L}\p{N}\._]+)?(\[(?:[^\d\]+-])?)';
// Error
const CALCULATION_REGEXP_ERROR = '\#[A-Z][A-Z0_\/]*[!\?]?';
@@ -59,14 +60,12 @@ class Calculation
const RETURN_ARRAY_AS_VALUE = 'value';
const RETURN_ARRAY_AS_ARRAY = 'array';
- const FORMULA_OPEN_FUNCTION_BRACE = '(';
- const FORMULA_CLOSE_FUNCTION_BRACE = ')';
- const FORMULA_OPEN_MATRIX_BRACE = '{';
- const FORMULA_CLOSE_MATRIX_BRACE = '}';
- const FORMULA_STRING_QUOTE = '"';
-
+ /** Preferable to use instance variable instanceArrayReturnType rather than this static property. */
private static string $returnArrayAsType = self::RETURN_ARRAY_AS_VALUE;
+ /** Preferable to use this instance variable rather than static returnArrayAsType */
+ private ?string $instanceArrayReturnType = null;
+
/**
* Instance of this class.
*/
@@ -79,6 +78,8 @@ class Calculation
/**
* Calculation cache.
+ *
+ * @var mixed[]
*/
private array $calculationCache = [];
@@ -120,21 +121,19 @@ class Calculation
private bool $suppressFormulaErrors = false;
+ private bool $processingAnchorArray = false;
+
/**
* Error message for any error that was raised/thrown by the calculation engine.
*/
public ?string $formulaError = null;
- /**
- * Reference Helper.
- */
- private static ReferenceHelper $referenceHelper;
-
/**
* An array of the nested cell references accessed by the calculation engine, used for the debug log.
*/
private CyclicReferenceStack $cyclicReferenceStack;
+ /** @var mixed[] */
private array $cellStack = [];
/**
@@ -151,51 +150,11 @@ class Calculation
*/
public int $cyclicFormulaCount = 1;
- /**
- * The current locale setting.
- */
- private static string $localeLanguage = 'en_us'; // US English (default locale)
-
- /**
- * List of available locale settings
- * Note that this is read for the locale subdirectory only when requested.
- *
- * @var string[]
- */
- private static array $validLocaleLanguages = [
- 'en', // English (default language)
- ];
-
- /**
- * Locale-specific argument separator for function arguments.
- */
- private static string $localeArgumentSeparator = ',';
-
- private static array $localeFunctions = [];
-
- /**
- * Locale-specific translations for Excel constants (True, False and Null).
- *
- * @var array
- */
- private static array $localeBoolean = [
- 'TRUE' => 'TRUE',
- 'FALSE' => 'FALSE',
- 'NULL' => 'NULL',
- ];
-
- public static function getLocaleBoolean(string $index): string
- {
- return self::$localeBoolean[$index];
- }
-
/**
* Excel constant string translations to their PHP equivalents
* Constant conversion from text name/value to actual (datatyped) value.
- *
- * @var array
*/
- private static array $excelConstants = [
+ private const EXCEL_CONSTANTS = [
'TRUE' => true,
'FALSE' => false,
'NULL' => null,
@@ -203,2659 +162,18 @@ class Calculation
public static function keyInExcelConstants(string $key): bool
{
- return array_key_exists($key, self::$excelConstants);
+ return array_key_exists($key, self::EXCEL_CONSTANTS);
}
public static function getExcelConstants(string $key): bool|null
{
- return self::$excelConstants[$key];
+ return self::EXCEL_CONSTANTS[$key];
}
- /**
- * Array of functions usable on Spreadsheet.
- * In theory, this could be const rather than static;
- * however, Phpstan breaks trying to analyze it when attempted.
- */
- private static array $phpSpreadsheetFunctions = [
- 'ABS' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Absolute::class, 'evaluate'],
- 'argumentCount' => '1',
- ],
- 'ACCRINT' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\Securities\AccruedInterest::class, 'periodic'],
- 'argumentCount' => '4-8',
- ],
- 'ACCRINTM' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\Securities\AccruedInterest::class, 'atMaturity'],
- 'argumentCount' => '3-5',
- ],
- 'ACOS' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Trig\Cosine::class, 'acos'],
- 'argumentCount' => '1',
- ],
- 'ACOSH' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Trig\Cosine::class, 'acosh'],
- 'argumentCount' => '1',
- ],
- 'ACOT' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Trig\Cotangent::class, 'acot'],
- 'argumentCount' => '1',
- ],
- 'ACOTH' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Trig\Cotangent::class, 'acoth'],
- 'argumentCount' => '1',
- ],
- 'ADDRESS' => [
- 'category' => Category::CATEGORY_LOOKUP_AND_REFERENCE,
- 'functionCall' => [LookupRef\Address::class, 'cell'],
- 'argumentCount' => '2-5',
- ],
- 'AGGREGATE' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '3+',
- ],
- 'AMORDEGRC' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\Amortization::class, 'AMORDEGRC'],
- 'argumentCount' => '6,7',
- ],
- 'AMORLINC' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\Amortization::class, 'AMORLINC'],
- 'argumentCount' => '6,7',
- ],
- 'ANCHORARRAY' => [
- 'category' => Category::CATEGORY_UNCATEGORISED,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '*',
- ],
- 'AND' => [
- 'category' => Category::CATEGORY_LOGICAL,
- 'functionCall' => [Logical\Operations::class, 'logicalAnd'],
- 'argumentCount' => '1+',
- ],
- 'ARABIC' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Arabic::class, 'evaluate'],
- 'argumentCount' => '1',
- ],
- 'AREAS' => [
- 'category' => Category::CATEGORY_LOOKUP_AND_REFERENCE,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '1',
- ],
- 'ARRAYTOTEXT' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Text::class, 'fromArray'],
- 'argumentCount' => '1,2',
- ],
- 'ASC' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '1',
- ],
- 'ASIN' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Trig\Sine::class, 'asin'],
- 'argumentCount' => '1',
- ],
- 'ASINH' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Trig\Sine::class, 'asinh'],
- 'argumentCount' => '1',
- ],
- 'ATAN' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Trig\Tangent::class, 'atan'],
- 'argumentCount' => '1',
- ],
- 'ATAN2' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Trig\Tangent::class, 'atan2'],
- 'argumentCount' => '2',
- ],
- 'ATANH' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Trig\Tangent::class, 'atanh'],
- 'argumentCount' => '1',
- ],
- 'AVEDEV' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Averages::class, 'averageDeviations'],
- 'argumentCount' => '1+',
- ],
- 'AVERAGE' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Averages::class, 'average'],
- 'argumentCount' => '1+',
- ],
- 'AVERAGEA' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Averages::class, 'averageA'],
- 'argumentCount' => '1+',
- ],
- 'AVERAGEIF' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Conditional::class, 'AVERAGEIF'],
- 'argumentCount' => '2,3',
- ],
- 'AVERAGEIFS' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Conditional::class, 'AVERAGEIFS'],
- 'argumentCount' => '3+',
- ],
- 'BAHTTEXT' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '1',
- ],
- 'BASE' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Base::class, 'evaluate'],
- 'argumentCount' => '2,3',
- ],
- 'BESSELI' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\BesselI::class, 'BESSELI'],
- 'argumentCount' => '2',
- ],
- 'BESSELJ' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\BesselJ::class, 'BESSELJ'],
- 'argumentCount' => '2',
- ],
- 'BESSELK' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\BesselK::class, 'BESSELK'],
- 'argumentCount' => '2',
- ],
- 'BESSELY' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\BesselY::class, 'BESSELY'],
- 'argumentCount' => '2',
- ],
- 'BETADIST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\Beta::class, 'distribution'],
- 'argumentCount' => '3-5',
- ],
- 'BETA.DIST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '4-6',
- ],
- 'BETAINV' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\Beta::class, 'inverse'],
- 'argumentCount' => '3-5',
- ],
- 'BETA.INV' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\Beta::class, 'inverse'],
- 'argumentCount' => '3-5',
- ],
- 'BIN2DEC' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ConvertBinary::class, 'toDecimal'],
- 'argumentCount' => '1',
- ],
- 'BIN2HEX' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ConvertBinary::class, 'toHex'],
- 'argumentCount' => '1,2',
- ],
- 'BIN2OCT' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ConvertBinary::class, 'toOctal'],
- 'argumentCount' => '1,2',
- ],
- 'BINOMDIST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\Binomial::class, 'distribution'],
- 'argumentCount' => '4',
- ],
- 'BINOM.DIST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\Binomial::class, 'distribution'],
- 'argumentCount' => '4',
- ],
- 'BINOM.DIST.RANGE' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\Binomial::class, 'range'],
- 'argumentCount' => '3,4',
- ],
- 'BINOM.INV' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\Binomial::class, 'inverse'],
- 'argumentCount' => '3',
- ],
- 'BITAND' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\BitWise::class, 'BITAND'],
- 'argumentCount' => '2',
- ],
- 'BITOR' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\BitWise::class, 'BITOR'],
- 'argumentCount' => '2',
- ],
- 'BITXOR' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\BitWise::class, 'BITXOR'],
- 'argumentCount' => '2',
- ],
- 'BITLSHIFT' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\BitWise::class, 'BITLSHIFT'],
- 'argumentCount' => '2',
- ],
- 'BITRSHIFT' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\BitWise::class, 'BITRSHIFT'],
- 'argumentCount' => '2',
- ],
- 'BYCOL' => [
- 'category' => Category::CATEGORY_LOGICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '*',
- ],
- 'BYROW' => [
- 'category' => Category::CATEGORY_LOGICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '*',
- ],
- 'CEILING' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Ceiling::class, 'ceiling'],
- 'argumentCount' => '1-2', // 2 for Excel, 1-2 for Ods/Gnumeric
- ],
- 'CEILING.MATH' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Ceiling::class, 'math'],
- 'argumentCount' => '1-3',
- ],
- 'CEILING.PRECISE' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Ceiling::class, 'precise'],
- 'argumentCount' => '1,2',
- ],
- 'CELL' => [
- 'category' => Category::CATEGORY_INFORMATION,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '1,2',
- ],
- 'CHAR' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\CharacterConvert::class, 'character'],
- 'argumentCount' => '1',
- ],
- 'CHIDIST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\ChiSquared::class, 'distributionRightTail'],
- 'argumentCount' => '2',
- ],
- 'CHISQ.DIST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\ChiSquared::class, 'distributionLeftTail'],
- 'argumentCount' => '3',
- ],
- 'CHISQ.DIST.RT' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\ChiSquared::class, 'distributionRightTail'],
- 'argumentCount' => '2',
- ],
- 'CHIINV' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\ChiSquared::class, 'inverseRightTail'],
- 'argumentCount' => '2',
- ],
- 'CHISQ.INV' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\ChiSquared::class, 'inverseLeftTail'],
- 'argumentCount' => '2',
- ],
- 'CHISQ.INV.RT' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\ChiSquared::class, 'inverseRightTail'],
- 'argumentCount' => '2',
- ],
- 'CHITEST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\ChiSquared::class, 'test'],
- 'argumentCount' => '2',
- ],
- 'CHISQ.TEST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\ChiSquared::class, 'test'],
- 'argumentCount' => '2',
- ],
- 'CHOOSE' => [
- 'category' => Category::CATEGORY_LOOKUP_AND_REFERENCE,
- 'functionCall' => [LookupRef\Selection::class, 'CHOOSE'],
- 'argumentCount' => '2+',
- ],
- 'CHOOSECOLS' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '2+',
- ],
- 'CHOOSEROWS' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '2+',
- ],
- 'CLEAN' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Trim::class, 'nonPrintable'],
- 'argumentCount' => '1',
- ],
- 'CODE' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\CharacterConvert::class, 'code'],
- 'argumentCount' => '1',
- ],
- 'COLUMN' => [
- 'category' => Category::CATEGORY_LOOKUP_AND_REFERENCE,
- 'functionCall' => [LookupRef\RowColumnInformation::class, 'COLUMN'],
- 'argumentCount' => '-1',
- 'passCellReference' => true,
- 'passByReference' => [true],
- ],
- 'COLUMNS' => [
- 'category' => Category::CATEGORY_LOOKUP_AND_REFERENCE,
- 'functionCall' => [LookupRef\RowColumnInformation::class, 'COLUMNS'],
- 'argumentCount' => '1',
- ],
- 'COMBIN' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Combinations::class, 'withoutRepetition'],
- 'argumentCount' => '2',
- ],
- 'COMBINA' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Combinations::class, 'withRepetition'],
- 'argumentCount' => '2',
- ],
- 'COMPLEX' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\Complex::class, 'COMPLEX'],
- 'argumentCount' => '2,3',
- ],
- 'CONCAT' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Concatenate::class, 'CONCATENATE'],
- 'argumentCount' => '1+',
- ],
- 'CONCATENATE' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Concatenate::class, 'CONCATENATE'],
- 'argumentCount' => '1+',
- ],
- 'CONFIDENCE' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Confidence::class, 'CONFIDENCE'],
- 'argumentCount' => '3',
- ],
- 'CONFIDENCE.NORM' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Confidence::class, 'CONFIDENCE'],
- 'argumentCount' => '3',
- ],
- 'CONFIDENCE.T' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '3',
- ],
- 'CONVERT' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ConvertUOM::class, 'CONVERT'],
- 'argumentCount' => '3',
- ],
- 'CORREL' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Trends::class, 'CORREL'],
- 'argumentCount' => '2',
- ],
- 'COS' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Trig\Cosine::class, 'cos'],
- 'argumentCount' => '1',
- ],
- 'COSH' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Trig\Cosine::class, 'cosh'],
- 'argumentCount' => '1',
- ],
- 'COT' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Trig\Cotangent::class, 'cot'],
- 'argumentCount' => '1',
- ],
- 'COTH' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Trig\Cotangent::class, 'coth'],
- 'argumentCount' => '1',
- ],
- 'COUNT' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Counts::class, 'COUNT'],
- 'argumentCount' => '1+',
- ],
- 'COUNTA' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Counts::class, 'COUNTA'],
- 'argumentCount' => '1+',
- ],
- 'COUNTBLANK' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Counts::class, 'COUNTBLANK'],
- 'argumentCount' => '1',
- ],
- 'COUNTIF' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Conditional::class, 'COUNTIF'],
- 'argumentCount' => '2',
- ],
- 'COUNTIFS' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Conditional::class, 'COUNTIFS'],
- 'argumentCount' => '2+',
- ],
- 'COUPDAYBS' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\Coupons::class, 'COUPDAYBS'],
- 'argumentCount' => '3,4',
- ],
- 'COUPDAYS' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\Coupons::class, 'COUPDAYS'],
- 'argumentCount' => '3,4',
- ],
- 'COUPDAYSNC' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\Coupons::class, 'COUPDAYSNC'],
- 'argumentCount' => '3,4',
- ],
- 'COUPNCD' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\Coupons::class, 'COUPNCD'],
- 'argumentCount' => '3,4',
- ],
- 'COUPNUM' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\Coupons::class, 'COUPNUM'],
- 'argumentCount' => '3,4',
- ],
- 'COUPPCD' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\Coupons::class, 'COUPPCD'],
- 'argumentCount' => '3,4',
- ],
- 'COVAR' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Trends::class, 'COVAR'],
- 'argumentCount' => '2',
- ],
- 'COVARIANCE.P' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Trends::class, 'COVAR'],
- 'argumentCount' => '2',
- ],
- 'COVARIANCE.S' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '2',
- ],
- 'CRITBINOM' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\Binomial::class, 'inverse'],
- 'argumentCount' => '3',
- ],
- 'CSC' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Trig\Cosecant::class, 'csc'],
- 'argumentCount' => '1',
- ],
- 'CSCH' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Trig\Cosecant::class, 'csch'],
- 'argumentCount' => '1',
- ],
- 'CUBEKPIMEMBER' => [
- 'category' => Category::CATEGORY_CUBE,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '?',
- ],
- 'CUBEMEMBER' => [
- 'category' => Category::CATEGORY_CUBE,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '?',
- ],
- 'CUBEMEMBERPROPERTY' => [
- 'category' => Category::CATEGORY_CUBE,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '?',
- ],
- 'CUBERANKEDMEMBER' => [
- 'category' => Category::CATEGORY_CUBE,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '?',
- ],
- 'CUBESET' => [
- 'category' => Category::CATEGORY_CUBE,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '?',
- ],
- 'CUBESETCOUNT' => [
- 'category' => Category::CATEGORY_CUBE,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '?',
- ],
- 'CUBEVALUE' => [
- 'category' => Category::CATEGORY_CUBE,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '?',
- ],
- 'CUMIPMT' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\CashFlow\Constant\Periodic\Cumulative::class, 'interest'],
- 'argumentCount' => '6',
- ],
- 'CUMPRINC' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\CashFlow\Constant\Periodic\Cumulative::class, 'principal'],
- 'argumentCount' => '6',
- ],
- 'DATE' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [DateTimeExcel\Date::class, 'fromYMD'],
- 'argumentCount' => '3',
- ],
- 'DATEDIF' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [DateTimeExcel\Difference::class, 'interval'],
- 'argumentCount' => '2,3',
- ],
- 'DATESTRING' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '?',
- ],
- 'DATEVALUE' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [DateTimeExcel\DateValue::class, 'fromString'],
- 'argumentCount' => '1',
- ],
- 'DAVERAGE' => [
- 'category' => Category::CATEGORY_DATABASE,
- 'functionCall' => [Database\DAverage::class, 'evaluate'],
- 'argumentCount' => '3',
- ],
- 'DAY' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [DateTimeExcel\DateParts::class, 'day'],
- 'argumentCount' => '1',
- ],
- 'DAYS' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [DateTimeExcel\Days::class, 'between'],
- 'argumentCount' => '2',
- ],
- 'DAYS360' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [DateTimeExcel\Days360::class, 'between'],
- 'argumentCount' => '2,3',
- ],
- 'DB' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\Depreciation::class, 'DB'],
- 'argumentCount' => '4,5',
- ],
- 'DBCS' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '1',
- ],
- 'DCOUNT' => [
- 'category' => Category::CATEGORY_DATABASE,
- 'functionCall' => [Database\DCount::class, 'evaluate'],
- 'argumentCount' => '3',
- ],
- 'DCOUNTA' => [
- 'category' => Category::CATEGORY_DATABASE,
- 'functionCall' => [Database\DCountA::class, 'evaluate'],
- 'argumentCount' => '3',
- ],
- 'DDB' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\Depreciation::class, 'DDB'],
- 'argumentCount' => '4,5',
- ],
- 'DEC2BIN' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ConvertDecimal::class, 'toBinary'],
- 'argumentCount' => '1,2',
- ],
- 'DEC2HEX' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ConvertDecimal::class, 'toHex'],
- 'argumentCount' => '1,2',
- ],
- 'DEC2OCT' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ConvertDecimal::class, 'toOctal'],
- 'argumentCount' => '1,2',
- ],
- 'DECIMAL' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '2',
- ],
- 'DEGREES' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Angle::class, 'toDegrees'],
- 'argumentCount' => '1',
- ],
- 'DELTA' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\Compare::class, 'DELTA'],
- 'argumentCount' => '1,2',
- ],
- 'DEVSQ' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Deviations::class, 'sumSquares'],
- 'argumentCount' => '1+',
- ],
- 'DGET' => [
- 'category' => Category::CATEGORY_DATABASE,
- 'functionCall' => [Database\DGet::class, 'evaluate'],
- 'argumentCount' => '3',
- ],
- 'DISC' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\Securities\Rates::class, 'discount'],
- 'argumentCount' => '4,5',
- ],
- 'DMAX' => [
- 'category' => Category::CATEGORY_DATABASE,
- 'functionCall' => [Database\DMax::class, 'evaluate'],
- 'argumentCount' => '3',
- ],
- 'DMIN' => [
- 'category' => Category::CATEGORY_DATABASE,
- 'functionCall' => [Database\DMin::class, 'evaluate'],
- 'argumentCount' => '3',
- ],
- 'DOLLAR' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Format::class, 'DOLLAR'],
- 'argumentCount' => '1,2',
- ],
- 'DOLLARDE' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\Dollar::class, 'decimal'],
- 'argumentCount' => '2',
- ],
- 'DOLLARFR' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\Dollar::class, 'fractional'],
- 'argumentCount' => '2',
- ],
- 'DPRODUCT' => [
- 'category' => Category::CATEGORY_DATABASE,
- 'functionCall' => [Database\DProduct::class, 'evaluate'],
- 'argumentCount' => '3',
- ],
- 'DROP' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '2-3',
- ],
- 'DSTDEV' => [
- 'category' => Category::CATEGORY_DATABASE,
- 'functionCall' => [Database\DStDev::class, 'evaluate'],
- 'argumentCount' => '3',
- ],
- 'DSTDEVP' => [
- 'category' => Category::CATEGORY_DATABASE,
- 'functionCall' => [Database\DStDevP::class, 'evaluate'],
- 'argumentCount' => '3',
- ],
- 'DSUM' => [
- 'category' => Category::CATEGORY_DATABASE,
- 'functionCall' => [Database\DSum::class, 'evaluate'],
- 'argumentCount' => '3',
- ],
- 'DURATION' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '5,6',
- ],
- 'DVAR' => [
- 'category' => Category::CATEGORY_DATABASE,
- 'functionCall' => [Database\DVar::class, 'evaluate'],
- 'argumentCount' => '3',
- ],
- 'DVARP' => [
- 'category' => Category::CATEGORY_DATABASE,
- 'functionCall' => [Database\DVarP::class, 'evaluate'],
- 'argumentCount' => '3',
- ],
- 'ECMA.CEILING' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '1,2',
- ],
- 'EDATE' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [DateTimeExcel\Month::class, 'adjust'],
- 'argumentCount' => '2',
- ],
- 'EFFECT' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\InterestRate::class, 'effective'],
- 'argumentCount' => '2',
- ],
- 'ENCODEURL' => [
- 'category' => Category::CATEGORY_WEB,
- 'functionCall' => [Web\Service::class, 'urlEncode'],
- 'argumentCount' => '1',
- ],
- 'EOMONTH' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [DateTimeExcel\Month::class, 'lastDay'],
- 'argumentCount' => '2',
- ],
- 'ERF' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\Erf::class, 'ERF'],
- 'argumentCount' => '1,2',
- ],
- 'ERF.PRECISE' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\Erf::class, 'ERFPRECISE'],
- 'argumentCount' => '1',
- ],
- 'ERFC' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ErfC::class, 'ERFC'],
- 'argumentCount' => '1',
- ],
- 'ERFC.PRECISE' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ErfC::class, 'ERFC'],
- 'argumentCount' => '1',
- ],
- 'ERROR.TYPE' => [
- 'category' => Category::CATEGORY_INFORMATION,
- 'functionCall' => [ExcelError::class, 'type'],
- 'argumentCount' => '1',
- ],
- 'EVEN' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Round::class, 'even'],
- 'argumentCount' => '1',
- ],
- 'EXACT' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Text::class, 'exact'],
- 'argumentCount' => '2',
- ],
- 'EXP' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Exp::class, 'evaluate'],
- 'argumentCount' => '1',
- ],
- 'EXPAND' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '2-4',
- ],
- 'EXPONDIST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\Exponential::class, 'distribution'],
- 'argumentCount' => '3',
- ],
- 'EXPON.DIST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\Exponential::class, 'distribution'],
- 'argumentCount' => '3',
- ],
- 'FACT' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Factorial::class, 'fact'],
- 'argumentCount' => '1',
- ],
- 'FACTDOUBLE' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Factorial::class, 'factDouble'],
- 'argumentCount' => '1',
- ],
- 'FALSE' => [
- 'category' => Category::CATEGORY_LOGICAL,
- 'functionCall' => [Logical\Boolean::class, 'FALSE'],
- 'argumentCount' => '0',
- ],
- 'FDIST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '3',
- ],
- 'F.DIST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\F::class, 'distribution'],
- 'argumentCount' => '4',
- ],
- 'F.DIST.RT' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '3',
- ],
- 'FILTER' => [
- 'category' => Category::CATEGORY_LOOKUP_AND_REFERENCE,
- 'functionCall' => [LookupRef\Filter::class, 'filter'],
- 'argumentCount' => '2-3',
- ],
- 'FILTERXML' => [
- 'category' => Category::CATEGORY_WEB,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '2',
- ],
- 'FIND' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Search::class, 'sensitive'],
- 'argumentCount' => '2,3',
- ],
- 'FINDB' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Search::class, 'sensitive'],
- 'argumentCount' => '2,3',
- ],
- 'FINV' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '3',
- ],
- 'F.INV' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '3',
- ],
- 'F.INV.RT' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '3',
- ],
- 'FISHER' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\Fisher::class, 'distribution'],
- 'argumentCount' => '1',
- ],
- 'FISHERINV' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\Fisher::class, 'inverse'],
- 'argumentCount' => '1',
- ],
- 'FIXED' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Format::class, 'FIXEDFORMAT'],
- 'argumentCount' => '1-3',
- ],
- 'FLOOR' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Floor::class, 'floor'],
- 'argumentCount' => '1-2', // Excel requries 2, Ods/Gnumeric 1-2
- ],
- 'FLOOR.MATH' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Floor::class, 'math'],
- 'argumentCount' => '1-3',
- ],
- 'FLOOR.PRECISE' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Floor::class, 'precise'],
- 'argumentCount' => '1-2',
- ],
- 'FORECAST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Trends::class, 'FORECAST'],
- 'argumentCount' => '3',
- ],
- 'FORECAST.ETS' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '3-6',
- ],
- 'FORECAST.ETS.CONFINT' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '3-6',
- ],
- 'FORECAST.ETS.SEASONALITY' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '2-4',
- ],
- 'FORECAST.ETS.STAT' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '3-6',
- ],
- 'FORECAST.LINEAR' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Trends::class, 'FORECAST'],
- 'argumentCount' => '3',
- ],
- 'FORMULATEXT' => [
- 'category' => Category::CATEGORY_LOOKUP_AND_REFERENCE,
- 'functionCall' => [LookupRef\Formula::class, 'text'],
- 'argumentCount' => '1',
- 'passCellReference' => true,
- 'passByReference' => [true],
- ],
- 'FREQUENCY' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '2',
- ],
- 'FTEST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '2',
- ],
- 'F.TEST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '2',
- ],
- 'FV' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\CashFlow\Constant\Periodic::class, 'futureValue'],
- 'argumentCount' => '3-5',
- ],
- 'FVSCHEDULE' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\CashFlow\Single::class, 'futureValue'],
- 'argumentCount' => '2',
- ],
- 'GAMMA' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\Gamma::class, 'gamma'],
- 'argumentCount' => '1',
- ],
- 'GAMMADIST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\Gamma::class, 'distribution'],
- 'argumentCount' => '4',
- ],
- 'GAMMA.DIST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\Gamma::class, 'distribution'],
- 'argumentCount' => '4',
- ],
- 'GAMMAINV' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\Gamma::class, 'inverse'],
- 'argumentCount' => '3',
- ],
- 'GAMMA.INV' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\Gamma::class, 'inverse'],
- 'argumentCount' => '3',
- ],
- 'GAMMALN' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\Gamma::class, 'ln'],
- 'argumentCount' => '1',
- ],
- 'GAMMALN.PRECISE' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\Gamma::class, 'ln'],
- 'argumentCount' => '1',
- ],
- 'GAUSS' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\StandardNormal::class, 'gauss'],
- 'argumentCount' => '1',
- ],
- 'GCD' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Gcd::class, 'evaluate'],
- 'argumentCount' => '1+',
- ],
- 'GEOMEAN' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Averages\Mean::class, 'geometric'],
- 'argumentCount' => '1+',
- ],
- 'GESTEP' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\Compare::class, 'GESTEP'],
- 'argumentCount' => '1,2',
- ],
- 'GETPIVOTDATA' => [
- 'category' => Category::CATEGORY_LOOKUP_AND_REFERENCE,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '2+',
- ],
- 'GROWTH' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Trends::class, 'GROWTH'],
- 'argumentCount' => '1-4',
- ],
- 'HARMEAN' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Averages\Mean::class, 'harmonic'],
- 'argumentCount' => '1+',
- ],
- 'HEX2BIN' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ConvertHex::class, 'toBinary'],
- 'argumentCount' => '1,2',
- ],
- 'HEX2DEC' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ConvertHex::class, 'toDecimal'],
- 'argumentCount' => '1',
- ],
- 'HEX2OCT' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ConvertHex::class, 'toOctal'],
- 'argumentCount' => '1,2',
- ],
- 'HLOOKUP' => [
- 'category' => Category::CATEGORY_LOOKUP_AND_REFERENCE,
- 'functionCall' => [LookupRef\HLookup::class, 'lookup'],
- 'argumentCount' => '3,4',
- ],
- 'HOUR' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [DateTimeExcel\TimeParts::class, 'hour'],
- 'argumentCount' => '1',
- ],
- 'HSTACK' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '1+',
- ],
- 'HYPERLINK' => [
- 'category' => Category::CATEGORY_LOOKUP_AND_REFERENCE,
- 'functionCall' => [LookupRef\Hyperlink::class, 'set'],
- 'argumentCount' => '1,2',
- 'passCellReference' => true,
- ],
- 'HYPGEOMDIST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\HyperGeometric::class, 'distribution'],
- 'argumentCount' => '4',
- ],
- 'HYPGEOM.DIST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '5',
- ],
- 'IF' => [
- 'category' => Category::CATEGORY_LOGICAL,
- 'functionCall' => [Logical\Conditional::class, 'statementIf'],
- 'argumentCount' => '2-3',
- ],
- 'IFERROR' => [
- 'category' => Category::CATEGORY_LOGICAL,
- 'functionCall' => [Logical\Conditional::class, 'IFERROR'],
- 'argumentCount' => '2',
- ],
- 'IFNA' => [
- 'category' => Category::CATEGORY_LOGICAL,
- 'functionCall' => [Logical\Conditional::class, 'IFNA'],
- 'argumentCount' => '2',
- ],
- 'IFS' => [
- 'category' => Category::CATEGORY_LOGICAL,
- 'functionCall' => [Logical\Conditional::class, 'IFS'],
- 'argumentCount' => '2+',
- ],
- 'IMABS' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ComplexFunctions::class, 'IMABS'],
- 'argumentCount' => '1',
- ],
- 'IMAGINARY' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\Complex::class, 'IMAGINARY'],
- 'argumentCount' => '1',
- ],
- 'IMARGUMENT' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ComplexFunctions::class, 'IMARGUMENT'],
- 'argumentCount' => '1',
- ],
- 'IMCONJUGATE' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ComplexFunctions::class, 'IMCONJUGATE'],
- 'argumentCount' => '1',
- ],
- 'IMCOS' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ComplexFunctions::class, 'IMCOS'],
- 'argumentCount' => '1',
- ],
- 'IMCOSH' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ComplexFunctions::class, 'IMCOSH'],
- 'argumentCount' => '1',
- ],
- 'IMCOT' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ComplexFunctions::class, 'IMCOT'],
- 'argumentCount' => '1',
- ],
- 'IMCSC' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ComplexFunctions::class, 'IMCSC'],
- 'argumentCount' => '1',
- ],
- 'IMCSCH' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ComplexFunctions::class, 'IMCSCH'],
- 'argumentCount' => '1',
- ],
- 'IMDIV' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ComplexOperations::class, 'IMDIV'],
- 'argumentCount' => '2',
- ],
- 'IMEXP' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ComplexFunctions::class, 'IMEXP'],
- 'argumentCount' => '1',
- ],
- 'IMLN' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ComplexFunctions::class, 'IMLN'],
- 'argumentCount' => '1',
- ],
- 'IMLOG10' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ComplexFunctions::class, 'IMLOG10'],
- 'argumentCount' => '1',
- ],
- 'IMLOG2' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ComplexFunctions::class, 'IMLOG2'],
- 'argumentCount' => '1',
- ],
- 'IMPOWER' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ComplexFunctions::class, 'IMPOWER'],
- 'argumentCount' => '2',
- ],
- 'IMPRODUCT' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ComplexOperations::class, 'IMPRODUCT'],
- 'argumentCount' => '1+',
- ],
- 'IMREAL' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\Complex::class, 'IMREAL'],
- 'argumentCount' => '1',
- ],
- 'IMSEC' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ComplexFunctions::class, 'IMSEC'],
- 'argumentCount' => '1',
- ],
- 'IMSECH' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ComplexFunctions::class, 'IMSECH'],
- 'argumentCount' => '1',
- ],
- 'IMSIN' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ComplexFunctions::class, 'IMSIN'],
- 'argumentCount' => '1',
- ],
- 'IMSINH' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ComplexFunctions::class, 'IMSINH'],
- 'argumentCount' => '1',
- ],
- 'IMSQRT' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ComplexFunctions::class, 'IMSQRT'],
- 'argumentCount' => '1',
- ],
- 'IMSUB' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ComplexOperations::class, 'IMSUB'],
- 'argumentCount' => '2',
- ],
- 'IMSUM' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ComplexOperations::class, 'IMSUM'],
- 'argumentCount' => '1+',
- ],
- 'IMTAN' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ComplexFunctions::class, 'IMTAN'],
- 'argumentCount' => '1',
- ],
- 'INDEX' => [
- 'category' => Category::CATEGORY_LOOKUP_AND_REFERENCE,
- 'functionCall' => [LookupRef\Matrix::class, 'index'],
- 'argumentCount' => '2-4',
- ],
- 'INDIRECT' => [
- 'category' => Category::CATEGORY_LOOKUP_AND_REFERENCE,
- 'functionCall' => [LookupRef\Indirect::class, 'INDIRECT'],
- 'argumentCount' => '1,2',
- 'passCellReference' => true,
- ],
- 'INFO' => [
- 'category' => Category::CATEGORY_INFORMATION,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '1',
- ],
- 'INT' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\IntClass::class, 'evaluate'],
- 'argumentCount' => '1',
- ],
- 'INTERCEPT' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Trends::class, 'INTERCEPT'],
- 'argumentCount' => '2',
- ],
- 'INTRATE' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\Securities\Rates::class, 'interest'],
- 'argumentCount' => '4,5',
- ],
- 'IPMT' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\CashFlow\Constant\Periodic\Interest::class, 'payment'],
- 'argumentCount' => '4-6',
- ],
- 'IRR' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\CashFlow\Variable\Periodic::class, 'rate'],
- 'argumentCount' => '1,2',
- ],
- 'ISBLANK' => [
- 'category' => Category::CATEGORY_INFORMATION,
- 'functionCall' => [Information\Value::class, 'isBlank'],
- 'argumentCount' => '1',
- ],
- 'ISERR' => [
- 'category' => Category::CATEGORY_INFORMATION,
- 'functionCall' => [Information\ErrorValue::class, 'isErr'],
- 'argumentCount' => '1',
- ],
- 'ISERROR' => [
- 'category' => Category::CATEGORY_INFORMATION,
- 'functionCall' => [Information\ErrorValue::class, 'isError'],
- 'argumentCount' => '1',
- ],
- 'ISEVEN' => [
- 'category' => Category::CATEGORY_INFORMATION,
- 'functionCall' => [Information\Value::class, 'isEven'],
- 'argumentCount' => '1',
- ],
- 'ISFORMULA' => [
- 'category' => Category::CATEGORY_INFORMATION,
- 'functionCall' => [Information\Value::class, 'isFormula'],
- 'argumentCount' => '1',
- 'passCellReference' => true,
- 'passByReference' => [true],
- ],
- 'ISLOGICAL' => [
- 'category' => Category::CATEGORY_INFORMATION,
- 'functionCall' => [Information\Value::class, 'isLogical'],
- 'argumentCount' => '1',
- ],
- 'ISNA' => [
- 'category' => Category::CATEGORY_INFORMATION,
- 'functionCall' => [Information\ErrorValue::class, 'isNa'],
- 'argumentCount' => '1',
- ],
- 'ISNONTEXT' => [
- 'category' => Category::CATEGORY_INFORMATION,
- 'functionCall' => [Information\Value::class, 'isNonText'],
- 'argumentCount' => '1',
- ],
- 'ISNUMBER' => [
- 'category' => Category::CATEGORY_INFORMATION,
- 'functionCall' => [Information\Value::class, 'isNumber'],
- 'argumentCount' => '1',
- ],
- 'ISO.CEILING' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '1,2',
- ],
- 'ISODD' => [
- 'category' => Category::CATEGORY_INFORMATION,
- 'functionCall' => [Information\Value::class, 'isOdd'],
- 'argumentCount' => '1',
- ],
- 'ISOMITTED' => [
- 'category' => Category::CATEGORY_INFORMATION,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '*',
- ],
- 'ISOWEEKNUM' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [DateTimeExcel\Week::class, 'isoWeekNumber'],
- 'argumentCount' => '1',
- ],
- 'ISPMT' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\CashFlow\Constant\Periodic\Interest::class, 'schedulePayment'],
- 'argumentCount' => '4',
- ],
- 'ISREF' => [
- 'category' => Category::CATEGORY_INFORMATION,
- 'functionCall' => [Information\Value::class, 'isRef'],
- 'argumentCount' => '1',
- 'passCellReference' => true,
- 'passByReference' => [true],
- ],
- 'ISTEXT' => [
- 'category' => Category::CATEGORY_INFORMATION,
- 'functionCall' => [Information\Value::class, 'isText'],
- 'argumentCount' => '1',
- ],
- 'ISTHAIDIGIT' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '?',
- ],
- 'JIS' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '1',
- ],
- 'KURT' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Deviations::class, 'kurtosis'],
- 'argumentCount' => '1+',
- ],
- 'LAMBDA' => [
- 'category' => Category::CATEGORY_LOGICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '*',
- ],
- 'LARGE' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Size::class, 'large'],
- 'argumentCount' => '2',
- ],
- 'LCM' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Lcm::class, 'evaluate'],
- 'argumentCount' => '1+',
- ],
- 'LEFT' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Extract::class, 'left'],
- 'argumentCount' => '1,2',
- ],
- 'LEFTB' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Extract::class, 'left'],
- 'argumentCount' => '1,2',
- ],
- 'LEN' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Text::class, 'length'],
- 'argumentCount' => '1',
- ],
- 'LENB' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Text::class, 'length'],
- 'argumentCount' => '1',
- ],
- 'LET' => [
- 'category' => Category::CATEGORY_LOGICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '*',
- ],
- 'LINEST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Trends::class, 'LINEST'],
- 'argumentCount' => '1-4',
- ],
- 'LN' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Logarithms::class, 'natural'],
- 'argumentCount' => '1',
- ],
- 'LOG' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Logarithms::class, 'withBase'],
- 'argumentCount' => '1,2',
- ],
- 'LOG10' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Logarithms::class, 'base10'],
- 'argumentCount' => '1',
- ],
- 'LOGEST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Trends::class, 'LOGEST'],
- 'argumentCount' => '1-4',
- ],
- 'LOGINV' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\LogNormal::class, 'inverse'],
- 'argumentCount' => '3',
- ],
- 'LOGNORMDIST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\LogNormal::class, 'cumulative'],
- 'argumentCount' => '3',
- ],
- 'LOGNORM.DIST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\LogNormal::class, 'distribution'],
- 'argumentCount' => '4',
- ],
- 'LOGNORM.INV' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\LogNormal::class, 'inverse'],
- 'argumentCount' => '3',
- ],
- 'LOOKUP' => [
- 'category' => Category::CATEGORY_LOOKUP_AND_REFERENCE,
- 'functionCall' => [LookupRef\Lookup::class, 'lookup'],
- 'argumentCount' => '2,3',
- ],
- 'LOWER' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\CaseConvert::class, 'lower'],
- 'argumentCount' => '1',
- ],
- 'MAKEARRAY' => [
- 'category' => Category::CATEGORY_LOGICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '*',
- ],
- 'MAP' => [
- 'category' => Category::CATEGORY_LOGICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '*',
- ],
- 'MATCH' => [
- 'category' => Category::CATEGORY_LOOKUP_AND_REFERENCE,
- 'functionCall' => [LookupRef\ExcelMatch::class, 'MATCH'],
- 'argumentCount' => '2,3',
- ],
- 'MAX' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Maximum::class, 'max'],
- 'argumentCount' => '1+',
- ],
- 'MAXA' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Maximum::class, 'maxA'],
- 'argumentCount' => '1+',
- ],
- 'MAXIFS' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Conditional::class, 'MAXIFS'],
- 'argumentCount' => '3+',
- ],
- 'MDETERM' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\MatrixFunctions::class, 'determinant'],
- 'argumentCount' => '1',
- ],
- 'MDURATION' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '5,6',
- ],
- 'MEDIAN' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Averages::class, 'median'],
- 'argumentCount' => '1+',
- ],
- 'MEDIANIF' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '2+',
- ],
- 'MID' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Extract::class, 'mid'],
- 'argumentCount' => '3',
- ],
- 'MIDB' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Extract::class, 'mid'],
- 'argumentCount' => '3',
- ],
- 'MIN' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Minimum::class, 'min'],
- 'argumentCount' => '1+',
- ],
- 'MINA' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Minimum::class, 'minA'],
- 'argumentCount' => '1+',
- ],
- 'MINIFS' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Conditional::class, 'MINIFS'],
- 'argumentCount' => '3+',
- ],
- 'MINUTE' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [DateTimeExcel\TimeParts::class, 'minute'],
- 'argumentCount' => '1',
- ],
- 'MINVERSE' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\MatrixFunctions::class, 'inverse'],
- 'argumentCount' => '1',
- ],
- 'MIRR' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\CashFlow\Variable\Periodic::class, 'modifiedRate'],
- 'argumentCount' => '3',
- ],
- 'MMULT' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\MatrixFunctions::class, 'multiply'],
- 'argumentCount' => '2',
- ],
- 'MOD' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Operations::class, 'mod'],
- 'argumentCount' => '2',
- ],
- 'MODE' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Averages::class, 'mode'],
- 'argumentCount' => '1+',
- ],
- 'MODE.MULT' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '1+',
- ],
- 'MODE.SNGL' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Averages::class, 'mode'],
- 'argumentCount' => '1+',
- ],
- 'MONTH' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [DateTimeExcel\DateParts::class, 'month'],
- 'argumentCount' => '1',
- ],
- 'MROUND' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Round::class, 'multiple'],
- 'argumentCount' => '2',
- ],
- 'MULTINOMIAL' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Factorial::class, 'multinomial'],
- 'argumentCount' => '1+',
- ],
- 'MUNIT' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\MatrixFunctions::class, 'identity'],
- 'argumentCount' => '1',
- ],
- 'N' => [
- 'category' => Category::CATEGORY_INFORMATION,
- 'functionCall' => [Information\Value::class, 'asNumber'],
- 'argumentCount' => '1',
- ],
- 'NA' => [
- 'category' => Category::CATEGORY_INFORMATION,
- 'functionCall' => [ExcelError::class, 'NA'],
- 'argumentCount' => '0',
- ],
- 'NEGBINOMDIST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\Binomial::class, 'negative'],
- 'argumentCount' => '3',
- ],
- 'NEGBINOM.DIST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '4',
- ],
- 'NETWORKDAYS' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [DateTimeExcel\NetworkDays::class, 'count'],
- 'argumentCount' => '2-3',
- ],
- 'NETWORKDAYS.INTL' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '2-4',
- ],
- 'NOMINAL' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\InterestRate::class, 'nominal'],
- 'argumentCount' => '2',
- ],
- 'NORMDIST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\Normal::class, 'distribution'],
- 'argumentCount' => '4',
- ],
- 'NORM.DIST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\Normal::class, 'distribution'],
- 'argumentCount' => '4',
- ],
- 'NORMINV' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\Normal::class, 'inverse'],
- 'argumentCount' => '3',
- ],
- 'NORM.INV' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\Normal::class, 'inverse'],
- 'argumentCount' => '3',
- ],
- 'NORMSDIST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\StandardNormal::class, 'cumulative'],
- 'argumentCount' => '1',
- ],
- 'NORM.S.DIST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\StandardNormal::class, 'distribution'],
- 'argumentCount' => '1,2',
- ],
- 'NORMSINV' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\StandardNormal::class, 'inverse'],
- 'argumentCount' => '1',
- ],
- 'NORM.S.INV' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\StandardNormal::class, 'inverse'],
- 'argumentCount' => '1',
- ],
- 'NOT' => [
- 'category' => Category::CATEGORY_LOGICAL,
- 'functionCall' => [Logical\Operations::class, 'NOT'],
- 'argumentCount' => '1',
- ],
- 'NOW' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [DateTimeExcel\Current::class, 'now'],
- 'argumentCount' => '0',
- ],
- 'NPER' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\CashFlow\Constant\Periodic::class, 'periods'],
- 'argumentCount' => '3-5',
- ],
- 'NPV' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\CashFlow\Variable\Periodic::class, 'presentValue'],
- 'argumentCount' => '2+',
- ],
- 'NUMBERSTRING' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '?',
- ],
- 'NUMBERVALUE' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Format::class, 'NUMBERVALUE'],
- 'argumentCount' => '1+',
- ],
- 'OCT2BIN' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ConvertOctal::class, 'toBinary'],
- 'argumentCount' => '1,2',
- ],
- 'OCT2DEC' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ConvertOctal::class, 'toDecimal'],
- 'argumentCount' => '1',
- ],
- 'OCT2HEX' => [
- 'category' => Category::CATEGORY_ENGINEERING,
- 'functionCall' => [Engineering\ConvertOctal::class, 'toHex'],
- 'argumentCount' => '1,2',
- ],
- 'ODD' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Round::class, 'odd'],
- 'argumentCount' => '1',
- ],
- 'ODDFPRICE' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '8,9',
- ],
- 'ODDFYIELD' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '8,9',
- ],
- 'ODDLPRICE' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '7,8',
- ],
- 'ODDLYIELD' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '7,8',
- ],
- 'OFFSET' => [
- 'category' => Category::CATEGORY_LOOKUP_AND_REFERENCE,
- 'functionCall' => [LookupRef\Offset::class, 'OFFSET'],
- 'argumentCount' => '3-5',
- 'passCellReference' => true,
- 'passByReference' => [true],
- ],
- 'OR' => [
- 'category' => Category::CATEGORY_LOGICAL,
- 'functionCall' => [Logical\Operations::class, 'logicalOr'],
- 'argumentCount' => '1+',
- ],
- 'PDURATION' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\CashFlow\Single::class, 'periods'],
- 'argumentCount' => '3',
- ],
- 'PEARSON' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Trends::class, 'CORREL'],
- 'argumentCount' => '2',
- ],
- 'PERCENTILE' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Percentiles::class, 'PERCENTILE'],
- 'argumentCount' => '2',
- ],
- 'PERCENTILE.EXC' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '2',
- ],
- 'PERCENTILE.INC' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Percentiles::class, 'PERCENTILE'],
- 'argumentCount' => '2',
- ],
- 'PERCENTRANK' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Percentiles::class, 'PERCENTRANK'],
- 'argumentCount' => '2,3',
- ],
- 'PERCENTRANK.EXC' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '2,3',
- ],
- 'PERCENTRANK.INC' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Percentiles::class, 'PERCENTRANK'],
- 'argumentCount' => '2,3',
- ],
- 'PERMUT' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Permutations::class, 'PERMUT'],
- 'argumentCount' => '2',
- ],
- 'PERMUTATIONA' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Permutations::class, 'PERMUTATIONA'],
- 'argumentCount' => '2',
- ],
- 'PHONETIC' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '1',
- ],
- 'PHI' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '1',
- ],
- 'PI' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => 'pi',
- 'argumentCount' => '0',
- ],
- 'PMT' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\CashFlow\Constant\Periodic\Payments::class, 'annuity'],
- 'argumentCount' => '3-5',
- ],
- 'POISSON' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\Poisson::class, 'distribution'],
- 'argumentCount' => '3',
- ],
- 'POISSON.DIST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\Poisson::class, 'distribution'],
- 'argumentCount' => '3',
- ],
- 'POWER' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Operations::class, 'power'],
- 'argumentCount' => '2',
- ],
- 'PPMT' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\CashFlow\Constant\Periodic\Payments::class, 'interestPayment'],
- 'argumentCount' => '4-6',
- ],
- 'PRICE' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\Securities\Price::class, 'price'],
- 'argumentCount' => '6,7',
- ],
- 'PRICEDISC' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\Securities\Price::class, 'priceDiscounted'],
- 'argumentCount' => '4,5',
- ],
- 'PRICEMAT' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\Securities\Price::class, 'priceAtMaturity'],
- 'argumentCount' => '5,6',
- ],
- 'PROB' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '3,4',
- ],
- 'PRODUCT' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Operations::class, 'product'],
- 'argumentCount' => '1+',
- ],
- 'PROPER' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\CaseConvert::class, 'proper'],
- 'argumentCount' => '1',
- ],
- 'PV' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\CashFlow\Constant\Periodic::class, 'presentValue'],
- 'argumentCount' => '3-5',
- ],
- 'QUARTILE' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Percentiles::class, 'QUARTILE'],
- 'argumentCount' => '2',
- ],
- 'QUARTILE.EXC' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '2',
- ],
- 'QUARTILE.INC' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Percentiles::class, 'QUARTILE'],
- 'argumentCount' => '2',
- ],
- 'QUOTIENT' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Operations::class, 'quotient'],
- 'argumentCount' => '2',
- ],
- 'RADIANS' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Angle::class, 'toRadians'],
- 'argumentCount' => '1',
- ],
- 'RAND' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Random::class, 'rand'],
- 'argumentCount' => '0',
- ],
- 'RANDARRAY' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Random::class, 'randArray'],
- 'argumentCount' => '0-5',
- ],
- 'RANDBETWEEN' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Random::class, 'randBetween'],
- 'argumentCount' => '2',
- ],
- 'RANK' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Percentiles::class, 'RANK'],
- 'argumentCount' => '2,3',
- ],
- 'RANK.AVG' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '2,3',
- ],
- 'RANK.EQ' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Percentiles::class, 'RANK'],
- 'argumentCount' => '2,3',
- ],
- 'RATE' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\CashFlow\Constant\Periodic\Interest::class, 'rate'],
- 'argumentCount' => '3-6',
- ],
- 'RECEIVED' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\Securities\Price::class, 'received'],
- 'argumentCount' => '4-5',
- ],
- 'REDUCE' => [
- 'category' => Category::CATEGORY_LOGICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '*',
- ],
- 'REPLACE' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Replace::class, 'replace'],
- 'argumentCount' => '4',
- ],
- 'REPLACEB' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Replace::class, 'replace'],
- 'argumentCount' => '4',
- ],
- 'REPT' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Concatenate::class, 'builtinREPT'],
- 'argumentCount' => '2',
- ],
- 'RIGHT' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Extract::class, 'right'],
- 'argumentCount' => '1,2',
- ],
- 'RIGHTB' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Extract::class, 'right'],
- 'argumentCount' => '1,2',
- ],
- 'ROMAN' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Roman::class, 'evaluate'],
- 'argumentCount' => '1,2',
- ],
- 'ROUND' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Round::class, 'round'],
- 'argumentCount' => '2',
- ],
- 'ROUNDBAHTDOWN' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '?',
- ],
- 'ROUNDBAHTUP' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '?',
- ],
- 'ROUNDDOWN' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Round::class, 'down'],
- 'argumentCount' => '2',
- ],
- 'ROUNDUP' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Round::class, 'up'],
- 'argumentCount' => '2',
- ],
- 'ROW' => [
- 'category' => Category::CATEGORY_LOOKUP_AND_REFERENCE,
- 'functionCall' => [LookupRef\RowColumnInformation::class, 'ROW'],
- 'argumentCount' => '-1',
- 'passCellReference' => true,
- 'passByReference' => [true],
- ],
- 'ROWS' => [
- 'category' => Category::CATEGORY_LOOKUP_AND_REFERENCE,
- 'functionCall' => [LookupRef\RowColumnInformation::class, 'ROWS'],
- 'argumentCount' => '1',
- ],
- 'RRI' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\CashFlow\Single::class, 'interestRate'],
- 'argumentCount' => '3',
- ],
- 'RSQ' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Trends::class, 'RSQ'],
- 'argumentCount' => '2',
- ],
- 'RTD' => [
- 'category' => Category::CATEGORY_LOOKUP_AND_REFERENCE,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '1+',
- ],
- 'SEARCH' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Search::class, 'insensitive'],
- 'argumentCount' => '2,3',
- ],
- 'SCAN' => [
- 'category' => Category::CATEGORY_LOGICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '*',
- ],
- 'SEARCHB' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Search::class, 'insensitive'],
- 'argumentCount' => '2,3',
- ],
- 'SEC' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Trig\Secant::class, 'sec'],
- 'argumentCount' => '1',
- ],
- 'SECH' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Trig\Secant::class, 'sech'],
- 'argumentCount' => '1',
- ],
- 'SECOND' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [DateTimeExcel\TimeParts::class, 'second'],
- 'argumentCount' => '1',
- ],
- 'SEQUENCE' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\MatrixFunctions::class, 'sequence'],
- 'argumentCount' => '1-4',
- ],
- 'SERIESSUM' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\SeriesSum::class, 'evaluate'],
- 'argumentCount' => '4',
- ],
- 'SHEET' => [
- 'category' => Category::CATEGORY_INFORMATION,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '0,1',
- ],
- 'SHEETS' => [
- 'category' => Category::CATEGORY_INFORMATION,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '0,1',
- ],
- 'SIGN' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Sign::class, 'evaluate'],
- 'argumentCount' => '1',
- ],
- 'SIN' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Trig\Sine::class, 'sin'],
- 'argumentCount' => '1',
- ],
- 'SINGLE' => [
- 'category' => Category::CATEGORY_UNCATEGORISED,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '*',
- ],
- 'SINH' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Trig\Sine::class, 'sinh'],
- 'argumentCount' => '1',
- ],
- 'SKEW' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Deviations::class, 'skew'],
- 'argumentCount' => '1+',
- ],
- 'SKEW.P' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '1+',
- ],
- 'SLN' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\Depreciation::class, 'SLN'],
- 'argumentCount' => '3',
- ],
- 'SLOPE' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Trends::class, 'SLOPE'],
- 'argumentCount' => '2',
- ],
- 'SMALL' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Size::class, 'small'],
- 'argumentCount' => '2',
- ],
- 'SORT' => [
- 'category' => Category::CATEGORY_LOOKUP_AND_REFERENCE,
- 'functionCall' => [LookupRef\Sort::class, 'sort'],
- 'argumentCount' => '1-4',
- ],
- 'SORTBY' => [
- 'category' => Category::CATEGORY_LOOKUP_AND_REFERENCE,
- 'functionCall' => [LookupRef\Sort::class, 'sortBy'],
- 'argumentCount' => '2+',
- ],
- 'SQRT' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Sqrt::class, 'sqrt'],
- 'argumentCount' => '1',
- ],
- 'SQRTPI' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Sqrt::class, 'pi'],
- 'argumentCount' => '1',
- ],
- 'STANDARDIZE' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Standardize::class, 'execute'],
- 'argumentCount' => '3',
- ],
- 'STDEV' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\StandardDeviations::class, 'STDEV'],
- 'argumentCount' => '1+',
- ],
- 'STDEV.S' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\StandardDeviations::class, 'STDEV'],
- 'argumentCount' => '1+',
- ],
- 'STDEV.P' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\StandardDeviations::class, 'STDEVP'],
- 'argumentCount' => '1+',
- ],
- 'STDEVA' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\StandardDeviations::class, 'STDEVA'],
- 'argumentCount' => '1+',
- ],
- 'STDEVP' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\StandardDeviations::class, 'STDEVP'],
- 'argumentCount' => '1+',
- ],
- 'STDEVPA' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\StandardDeviations::class, 'STDEVPA'],
- 'argumentCount' => '1+',
- ],
- 'STEYX' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Trends::class, 'STEYX'],
- 'argumentCount' => '2',
- ],
- 'SUBSTITUTE' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Replace::class, 'substitute'],
- 'argumentCount' => '3,4',
- ],
- 'SUBTOTAL' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Subtotal::class, 'evaluate'],
- 'argumentCount' => '2+',
- 'passCellReference' => true,
- ],
- 'SUM' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Sum::class, 'sumErroringStrings'],
- 'argumentCount' => '1+',
- ],
- 'SUMIF' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [Statistical\Conditional::class, 'SUMIF'],
- 'argumentCount' => '2,3',
- ],
- 'SUMIFS' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [Statistical\Conditional::class, 'SUMIFS'],
- 'argumentCount' => '3+',
- ],
- 'SUMPRODUCT' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Sum::class, 'product'],
- 'argumentCount' => '1+',
- ],
- 'SUMSQ' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\SumSquares::class, 'sumSquare'],
- 'argumentCount' => '1+',
- ],
- 'SUMX2MY2' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\SumSquares::class, 'sumXSquaredMinusYSquared'],
- 'argumentCount' => '2',
- ],
- 'SUMX2PY2' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\SumSquares::class, 'sumXSquaredPlusYSquared'],
- 'argumentCount' => '2',
- ],
- 'SUMXMY2' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\SumSquares::class, 'sumXMinusYSquared'],
- 'argumentCount' => '2',
- ],
- 'SWITCH' => [
- 'category' => Category::CATEGORY_LOGICAL,
- 'functionCall' => [Logical\Conditional::class, 'statementSwitch'],
- 'argumentCount' => '3+',
- ],
- 'SYD' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\Depreciation::class, 'SYD'],
- 'argumentCount' => '4',
- ],
- 'T' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Text::class, 'test'],
- 'argumentCount' => '1',
- ],
- 'TAKE' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '2-3',
- ],
- 'TAN' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Trig\Tangent::class, 'tan'],
- 'argumentCount' => '1',
- ],
- 'TANH' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Trig\Tangent::class, 'tanh'],
- 'argumentCount' => '1',
- ],
- 'TBILLEQ' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\TreasuryBill::class, 'bondEquivalentYield'],
- 'argumentCount' => '3',
- ],
- 'TBILLPRICE' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\TreasuryBill::class, 'price'],
- 'argumentCount' => '3',
- ],
- 'TBILLYIELD' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\TreasuryBill::class, 'yield'],
- 'argumentCount' => '3',
- ],
- 'TDIST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\StudentT::class, 'distribution'],
- 'argumentCount' => '3',
- ],
- 'T.DIST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '3',
- ],
- 'T.DIST.2T' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '2',
- ],
- 'T.DIST.RT' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '2',
- ],
- 'TEXT' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Format::class, 'TEXTFORMAT'],
- 'argumentCount' => '2',
- ],
- 'TEXTAFTER' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Extract::class, 'after'],
- 'argumentCount' => '2-6',
- ],
- 'TEXTBEFORE' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Extract::class, 'before'],
- 'argumentCount' => '2-6',
- ],
- 'TEXTJOIN' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Concatenate::class, 'TEXTJOIN'],
- 'argumentCount' => '3+',
- ],
- 'TEXTSPLIT' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Text::class, 'split'],
- 'argumentCount' => '2-6',
- ],
- 'THAIDAYOFWEEK' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '?',
- ],
- 'THAIDIGIT' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '?',
- ],
- 'THAIMONTHOFYEAR' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '?',
- ],
- 'THAINUMSOUND' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '?',
- ],
- 'THAINUMSTRING' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '?',
- ],
- 'THAISTRINGLENGTH' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '?',
- ],
- 'THAIYEAR' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '?',
- ],
- 'TIME' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [DateTimeExcel\Time::class, 'fromHMS'],
- 'argumentCount' => '3',
- ],
- 'TIMEVALUE' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [DateTimeExcel\TimeValue::class, 'fromString'],
- 'argumentCount' => '1',
- ],
- 'TINV' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\StudentT::class, 'inverse'],
- 'argumentCount' => '2',
- ],
- 'T.INV' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\StudentT::class, 'inverse'],
- 'argumentCount' => '2',
- ],
- 'T.INV.2T' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '2',
- ],
- 'TODAY' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [DateTimeExcel\Current::class, 'today'],
- 'argumentCount' => '0',
- ],
- 'TOCOL' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '1-3',
- ],
- 'TOROW' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '1-3',
- ],
- 'TRANSPOSE' => [
- 'category' => Category::CATEGORY_LOOKUP_AND_REFERENCE,
- 'functionCall' => [LookupRef\Matrix::class, 'transpose'],
- 'argumentCount' => '1',
- ],
- 'TREND' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Trends::class, 'TREND'],
- 'argumentCount' => '1-4',
- ],
- 'TRIM' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Trim::class, 'spaces'],
- 'argumentCount' => '1',
- ],
- 'TRIMMEAN' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Averages\Mean::class, 'trim'],
- 'argumentCount' => '2',
- ],
- 'TRUE' => [
- 'category' => Category::CATEGORY_LOGICAL,
- 'functionCall' => [Logical\Boolean::class, 'TRUE'],
- 'argumentCount' => '0',
- ],
- 'TRUNC' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [MathTrig\Trunc::class, 'evaluate'],
- 'argumentCount' => '1,2',
- ],
- 'TTEST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '4',
- ],
- 'T.TEST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '4',
- ],
- 'TYPE' => [
- 'category' => Category::CATEGORY_INFORMATION,
- 'functionCall' => [Information\Value::class, 'type'],
- 'argumentCount' => '1',
- ],
- 'UNICHAR' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\CharacterConvert::class, 'character'],
- 'argumentCount' => '1',
- ],
- 'UNICODE' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\CharacterConvert::class, 'code'],
- 'argumentCount' => '1',
- ],
- 'UNIQUE' => [
- 'category' => Category::CATEGORY_LOOKUP_AND_REFERENCE,
- 'functionCall' => [LookupRef\Unique::class, 'unique'],
- 'argumentCount' => '1+',
- ],
- 'UPPER' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\CaseConvert::class, 'upper'],
- 'argumentCount' => '1',
- ],
- 'USDOLLAR' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\Dollar::class, 'format'],
- 'argumentCount' => '2',
- ],
- 'VALUE' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Format::class, 'VALUE'],
- 'argumentCount' => '1',
- ],
- 'VALUETOTEXT' => [
- 'category' => Category::CATEGORY_TEXT_AND_DATA,
- 'functionCall' => [TextData\Format::class, 'valueToText'],
- 'argumentCount' => '1,2',
- ],
- 'VAR' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Variances::class, 'VAR'],
- 'argumentCount' => '1+',
- ],
- 'VAR.P' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Variances::class, 'VARP'],
- 'argumentCount' => '1+',
- ],
- 'VAR.S' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Variances::class, 'VAR'],
- 'argumentCount' => '1+',
- ],
- 'VARA' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Variances::class, 'VARA'],
- 'argumentCount' => '1+',
- ],
- 'VARP' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Variances::class, 'VARP'],
- 'argumentCount' => '1+',
- ],
- 'VARPA' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Variances::class, 'VARPA'],
- 'argumentCount' => '1+',
- ],
- 'VDB' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '5-7',
- ],
- 'VLOOKUP' => [
- 'category' => Category::CATEGORY_LOOKUP_AND_REFERENCE,
- 'functionCall' => [LookupRef\VLookup::class, 'lookup'],
- 'argumentCount' => '3,4',
- ],
- 'VSTACK' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '1+',
- ],
- 'WEBSERVICE' => [
- 'category' => Category::CATEGORY_WEB,
- 'functionCall' => [Web\Service::class, 'webService'],
- 'argumentCount' => '1',
- ],
- 'WEEKDAY' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [DateTimeExcel\Week::class, 'day'],
- 'argumentCount' => '1,2',
- ],
- 'WEEKNUM' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [DateTimeExcel\Week::class, 'number'],
- 'argumentCount' => '1,2',
- ],
- 'WEIBULL' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\Weibull::class, 'distribution'],
- 'argumentCount' => '4',
- ],
- 'WEIBULL.DIST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\Weibull::class, 'distribution'],
- 'argumentCount' => '4',
- ],
- 'WORKDAY' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [DateTimeExcel\WorkDay::class, 'date'],
- 'argumentCount' => '2-3',
- ],
- 'WORKDAY.INTL' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '2-4',
- ],
- 'WRAPCOLS' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '2-3',
- ],
- 'WRAPROWS' => [
- 'category' => Category::CATEGORY_MATH_AND_TRIG,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '2-3',
- ],
- 'XIRR' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\CashFlow\Variable\NonPeriodic::class, 'rate'],
- 'argumentCount' => '2,3',
- ],
- 'XLOOKUP' => [
- 'category' => Category::CATEGORY_LOOKUP_AND_REFERENCE,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '3-6',
- ],
- 'XNPV' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\CashFlow\Variable\NonPeriodic::class, 'presentValue'],
- 'argumentCount' => '3',
- ],
- 'XMATCH' => [
- 'category' => Category::CATEGORY_LOOKUP_AND_REFERENCE,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '2,3',
- ],
- 'XOR' => [
- 'category' => Category::CATEGORY_LOGICAL,
- 'functionCall' => [Logical\Operations::class, 'logicalXor'],
- 'argumentCount' => '1+',
- ],
- 'YEAR' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [DateTimeExcel\DateParts::class, 'year'],
- 'argumentCount' => '1',
- ],
- 'YEARFRAC' => [
- 'category' => Category::CATEGORY_DATE_AND_TIME,
- 'functionCall' => [DateTimeExcel\YearFrac::class, 'fraction'],
- 'argumentCount' => '2,3',
- ],
- 'YIELD' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Functions::class, 'DUMMY'],
- 'argumentCount' => '6,7',
- ],
- 'YIELDDISC' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\Securities\Yields::class, 'yieldDiscounted'],
- 'argumentCount' => '4,5',
- ],
- 'YIELDMAT' => [
- 'category' => Category::CATEGORY_FINANCIAL,
- 'functionCall' => [Financial\Securities\Yields::class, 'yieldAtMaturity'],
- 'argumentCount' => '5,6',
- ],
- 'ZTEST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\StandardNormal::class, 'zTest'],
- 'argumentCount' => '2-3',
- ],
- 'Z.TEST' => [
- 'category' => Category::CATEGORY_STATISTICAL,
- 'functionCall' => [Statistical\Distributions\StandardNormal::class, 'zTest'],
- 'argumentCount' => '2-3',
- ],
- ];
-
/**
* Internal functions used for special control purposes.
+ *
+ * @var array|string>>
*/
private static array $controlFunctions = [
'MKMATRIX' => [
@@ -2878,19 +196,6 @@ class Calculation
$this->cyclicReferenceStack = new CyclicReferenceStack();
$this->debugLog = new Logger($this->cyclicReferenceStack);
$this->branchPruner = new BranchPruner($this->branchPruningEnabled);
- self::$referenceHelper = ReferenceHelper::getInstance();
- }
-
- private static function loadLocales(): void
- {
- $localeFileDirectory = __DIR__ . '/locale/';
- $localeFileNames = glob($localeFileDirectory . '*', GLOB_ONLYDIR) ?: [];
- foreach ($localeFileNames as $filename) {
- $filename = substr($filename, strlen($localeFileDirectory));
- if ($filename != 'en') {
- self::$validLocaleLanguages[] = $filename;
- }
- }
}
/**
@@ -2902,10 +207,7 @@ class Calculation
public static function getInstance(?Spreadsheet $spreadsheet = null): self
{
if ($spreadsheet !== null) {
- $instance = $spreadsheet->getCalculationEngine();
- if (isset($instance)) {
- return $instance;
- }
+ return $spreadsheet->getCalculationEngine();
}
if (!self::$instance) {
@@ -2915,6 +217,20 @@ class Calculation
return self::$instance;
}
+ /**
+ * Intended for use only via a destructor.
+ *
+ * @internal
+ */
+ public static function getInstanceOrNull(?Spreadsheet $spreadsheet = null): ?self
+ {
+ if ($spreadsheet !== null) {
+ return $spreadsheet->getCalculationEngineOrNull();
+ }
+
+ return null;
+ }
+
/**
* Flush the calculation cache for any existing instance of this class
* but only if a Calculation instance exists.
@@ -2941,26 +257,6 @@ class Calculation
throw new Exception('Cloning the calculation engine is not allowed!');
}
- /**
- * Return the locale-specific translation of TRUE.
- *
- * @return string locale-specific translation of TRUE
- */
- public static function getTRUE(): string
- {
- return self::$localeBoolean['TRUE'];
- }
-
- /**
- * Return the locale-specific translation of FALSE.
- *
- * @return string locale-specific translation of FALSE
- */
- public static function getFALSE(): string
- {
- return self::$localeBoolean['FALSE'];
- }
-
/**
* Set the Array Return Type (Array or Value of first element in the array).
*
@@ -2993,6 +289,38 @@ class Calculation
return self::$returnArrayAsType;
}
+ /**
+ * Set the Instance Array Return Type (Array or Value of first element in the array).
+ *
+ * @param string $returnType Array return type
+ *
+ * @return bool Success or failure
+ */
+ public function setInstanceArrayReturnType(string $returnType): bool
+ {
+ if (
+ ($returnType == self::RETURN_ARRAY_AS_VALUE)
+ || ($returnType == self::RETURN_ARRAY_AS_ERROR)
+ || ($returnType == self::RETURN_ARRAY_AS_ARRAY)
+ ) {
+ $this->instanceArrayReturnType = $returnType;
+
+ return true;
+ }
+
+ return false;
+ }
+
+ /**
+ * Return the Array Return Type (Array or Value of first element in the array).
+ *
+ * @return string $returnType Array return type for instance if non-null, otherwise static property
+ */
+ public function getInstanceArrayReturnType(): string
+ {
+ return $this->instanceArrayReturnType ?? self::$returnArrayAsType;
+ }
+
/**
* Is calculation caching enabled?
*/
@@ -3004,10 +332,12 @@ class Calculation
/**
* Enable/disable calculation cache.
*/
- public function setCalculationCacheEnabled(bool $calculationCacheEnabled): void
+ public function setCalculationCacheEnabled(bool $calculationCacheEnabled): self
{
$this->calculationCacheEnabled = $calculationCacheEnabled;
$this->clearCalculationCache();
+
+ return $this;
}
/**
@@ -3055,13 +385,17 @@ class Calculation
}
}
- /**
- * Enable/disable calculation cache.
- */
- public function setBranchPruningEnabled(mixed $enabled): void
+ public function getBranchPruningEnabled(): bool
{
- $this->branchPruningEnabled = $enabled;
+ return $this->branchPruningEnabled;
+ }
+
+ public function setBranchPruningEnabled(mixed $enabled): self
+ {
+ $this->branchPruningEnabled = (bool) $enabled;
$this->branchPruner = new BranchPruner($this->branchPruningEnabled);
+
+ return $this;
}
public function enableBranchPruning(): void
@@ -3074,304 +408,6 @@ class Calculation
$this->setBranchPruningEnabled(false);
}
- /**
- * Get the currently defined locale code.
- */
- public function getLocale(): string
- {
- return self::$localeLanguage;
- }
-
- private function getLocaleFile(string $localeDir, string $locale, string $language, string $file): string
- {
- $localeFileName = $localeDir . str_replace('_', DIRECTORY_SEPARATOR, $locale)
- . DIRECTORY_SEPARATOR . $file;
- if (!file_exists($localeFileName)) {
- // If there isn't a locale specific file, look for a language specific file
- $localeFileName = $localeDir . $language . DIRECTORY_SEPARATOR . $file;
- if (!file_exists($localeFileName)) {
- throw new Exception('Locale file not found');
- }
- }
-
- return $localeFileName;
- }
-
- /**
- * Set the locale code.
- *
- * @param string $locale The locale to use for formula translation, eg: 'en_us'
- */
- public function setLocale(string $locale): bool
- {
- // Identify our locale and language
- $language = $locale = strtolower($locale);
- if (str_contains($locale, '_')) {
- [$language] = explode('_', $locale);
- }
- if (count(self::$validLocaleLanguages) == 1) {
- self::loadLocales();
- }
-
- // Test whether we have any language data for this language (any locale)
- if (in_array($language, self::$validLocaleLanguages, true)) {
- // initialise language/locale settings
- self::$localeFunctions = [];
- self::$localeArgumentSeparator = ',';
- self::$localeBoolean = ['TRUE' => 'TRUE', 'FALSE' => 'FALSE', 'NULL' => 'NULL'];
-
- // Default is US English, if user isn't requesting US english, then read the necessary data from the locale files
- if ($locale !== 'en_us') {
- $localeDir = implode(DIRECTORY_SEPARATOR, [__DIR__, 'locale', null]);
-
- // Search for a file with a list of function names for locale
- try {
- $functionNamesFile = $this->getLocaleFile($localeDir, $locale, $language, 'functions');
- } catch (Exception $e) {
- return false;
- }
-
- // Retrieve the list of locale or language specific function names
- $localeFunctions = file($functionNamesFile, FILE_IGNORE_NEW_LINES | FILE_SKIP_EMPTY_LINES) ?: [];
- foreach ($localeFunctions as $localeFunction) {
- [$localeFunction] = explode('##', $localeFunction); // Strip out comments
- if (str_contains($localeFunction, '=')) {
- [$fName, $lfName] = array_map('trim', explode('=', $localeFunction));
- if ((str_starts_with($fName, '*') || isset(self::$phpSpreadsheetFunctions[$fName])) && ($lfName != '') && ($fName != $lfName)) {
- self::$localeFunctions[$fName] = $lfName;
- }
- }
- }
- // Default the TRUE and FALSE constants to the locale names of the TRUE() and FALSE() functions
- if (isset(self::$localeFunctions['TRUE'])) {
- self::$localeBoolean['TRUE'] = self::$localeFunctions['TRUE'];
- }
- if (isset(self::$localeFunctions['FALSE'])) {
- self::$localeBoolean['FALSE'] = self::$localeFunctions['FALSE'];
- }
-
- try {
- $configFile = $this->getLocaleFile($localeDir, $locale, $language, 'config');
- } catch (Exception) {
- return false;
- }
-
- $localeSettings = file($configFile, FILE_IGNORE_NEW_LINES | FILE_SKIP_EMPTY_LINES) ?: [];
- foreach ($localeSettings as $localeSetting) {
- [$localeSetting] = explode('##', $localeSetting); // Strip out comments
- if (str_contains($localeSetting, '=')) {
- [$settingName, $settingValue] = array_map('trim', explode('=', $localeSetting));
- $settingName = strtoupper($settingName);
- if ($settingValue !== '') {
- switch ($settingName) {
- case 'ARGUMENTSEPARATOR':
- self::$localeArgumentSeparator = $settingValue;
-
- break;
- }
- }
- }
- }
- }
-
- self::$functionReplaceFromExcel = self::$functionReplaceToExcel
- = self::$functionReplaceFromLocale = self::$functionReplaceToLocale = null;
- self::$localeLanguage = $locale;
-
- return true;
- }
-
- return false;
- }
-
- public static function translateSeparator(
- string $fromSeparator,
- string $toSeparator,
- string $formula,
- int &$inBracesLevel,
- string $openBrace = self::FORMULA_OPEN_FUNCTION_BRACE,
- string $closeBrace = self::FORMULA_CLOSE_FUNCTION_BRACE
- ): string {
- $strlen = mb_strlen($formula);
- for ($i = 0; $i < $strlen; ++$i) {
- $chr = mb_substr($formula, $i, 1);
- switch ($chr) {
- case $openBrace:
- ++$inBracesLevel;
-
- break;
- case $closeBrace:
- --$inBracesLevel;
-
- break;
- case $fromSeparator:
- if ($inBracesLevel > 0) {
- $formula = mb_substr($formula, 0, $i) . $toSeparator . mb_substr($formula, $i + 1);
- }
- }
- }
-
- return $formula;
- }
-
- private static function translateFormulaBlock(
- array $from,
- array $to,
- string $formula,
- int &$inFunctionBracesLevel,
- int &$inMatrixBracesLevel,
- string $fromSeparator,
- string $toSeparator
- ): string {
- // Function Names
- $formula = (string) preg_replace($from, $to, $formula);
-
- // Temporarily adjust matrix separators so that they won't be confused with function arguments
- $formula = self::translateSeparator(';', '|', $formula, $inMatrixBracesLevel, self::FORMULA_OPEN_MATRIX_BRACE, self::FORMULA_CLOSE_MATRIX_BRACE);
- $formula = self::translateSeparator(',', '!', $formula, $inMatrixBracesLevel, self::FORMULA_OPEN_MATRIX_BRACE, self::FORMULA_CLOSE_MATRIX_BRACE);
- // Function Argument Separators
- $formula = self::translateSeparator($fromSeparator, $toSeparator, $formula, $inFunctionBracesLevel);
- // Restore matrix separators
- $formula = self::translateSeparator('|', ';', $formula, $inMatrixBracesLevel, self::FORMULA_OPEN_MATRIX_BRACE, self::FORMULA_CLOSE_MATRIX_BRACE);
- $formula = self::translateSeparator('!', ',', $formula, $inMatrixBracesLevel, self::FORMULA_OPEN_MATRIX_BRACE, self::FORMULA_CLOSE_MATRIX_BRACE);
-
- return $formula;
- }
-
- private static function translateFormula(array $from, array $to, string $formula, string $fromSeparator, string $toSeparator): string
- {
- // Convert any Excel function names and constant names to the required language;
- // and adjust function argument separators
- if (self::$localeLanguage !== 'en_us') {
- $inFunctionBracesLevel = 0;
- $inMatrixBracesLevel = 0;
- // If there is the possibility of separators within a quoted string, then we treat them as literals
- if (str_contains($formula, self::FORMULA_STRING_QUOTE)) {
- // So instead we skip replacing in any quoted strings by only replacing in every other array element
- // after we've exploded the formula
- $temp = explode(self::FORMULA_STRING_QUOTE, $formula);
- $notWithinQuotes = false;
- foreach ($temp as &$value) {
- // Only adjust in alternating array entries
- $notWithinQuotes = $notWithinQuotes === false;
- if ($notWithinQuotes === true) {
- $value = self::translateFormulaBlock($from, $to, $value, $inFunctionBracesLevel, $inMatrixBracesLevel, $fromSeparator, $toSeparator);
- }
- }
- unset($value);
- // Then rebuild the formula string
- $formula = implode(self::FORMULA_STRING_QUOTE, $temp);
- } else {
- // If there's no quoted strings, then we do a simple count/replace
- $formula = self::translateFormulaBlock($from, $to, $formula, $inFunctionBracesLevel, $inMatrixBracesLevel, $fromSeparator, $toSeparator);
- }
- }
-
- return $formula;
- }
-
- private static ?array $functionReplaceFromExcel;
-
- private static ?array $functionReplaceToLocale;
-
- /**
- * @deprecated 1.30.0 use translateFormulaToLocale() instead
- *
- * @codeCoverageIgnore
- */
- public function _translateFormulaToLocale(string $formula): string
- {
- return $this->translateFormulaToLocale($formula);
- }
-
- public function translateFormulaToLocale(string $formula): string
- {
- $formula = preg_replace(self::CALCULATION_REGEXP_STRIP_XLFN_XLWS, '', $formula) ?? '';
- // Build list of function names and constants for translation
- if (self::$functionReplaceFromExcel === null) {
- self::$functionReplaceFromExcel = [];
- foreach (array_keys(self::$localeFunctions) as $excelFunctionName) {
- self::$functionReplaceFromExcel[] = '/(@?[^\w\.])' . preg_quote($excelFunctionName, '/') . '([\s]*\()/ui';
- }
- foreach (array_keys(self::$localeBoolean) as $excelBoolean) {
- self::$functionReplaceFromExcel[] = '/(@?[^\w\.])' . preg_quote($excelBoolean, '/') . '([^\w\.])/ui';
- }
- }
-
- if (self::$functionReplaceToLocale === null) {
- self::$functionReplaceToLocale = [];
- foreach (self::$localeFunctions as $localeFunctionName) {
- self::$functionReplaceToLocale[] = '$1' . trim($localeFunctionName) . '$2';
- }
- foreach (self::$localeBoolean as $localeBoolean) {
- self::$functionReplaceToLocale[] = '$1' . trim($localeBoolean) . '$2';
- }
- }
-
- return self::translateFormula(
- self::$functionReplaceFromExcel,
- self::$functionReplaceToLocale,
- $formula,
- ',',
- self::$localeArgumentSeparator
- );
- }
-
- private static ?array $functionReplaceFromLocale;
-
- private static ?array $functionReplaceToExcel;
-
- /**
- * @deprecated 1.30.0 use translateFormulaToEnglish() instead
- *
- * @codeCoverageIgnore
- */
- public function _translateFormulaToEnglish(string $formula): string
- {
- return $this->translateFormulaToEnglish($formula);
- }
-
- public function translateFormulaToEnglish(string $formula): string
- {
- if (self::$functionReplaceFromLocale === null) {
- self::$functionReplaceFromLocale = [];
- foreach (self::$localeFunctions as $localeFunctionName) {
- self::$functionReplaceFromLocale[] = '/(@?[^\w\.])' . preg_quote($localeFunctionName, '/') . '([\s]*\()/ui';
- }
- foreach (self::$localeBoolean as $excelBoolean) {
- self::$functionReplaceFromLocale[] = '/(@?[^\w\.])' . preg_quote($excelBoolean, '/') . '([^\w\.])/ui';
- }
- }
-
- if (self::$functionReplaceToExcel === null) {
- self::$functionReplaceToExcel = [];
- foreach (array_keys(self::$localeFunctions) as $excelFunctionName) {
- self::$functionReplaceToExcel[] = '$1' . trim($excelFunctionName) . '$2';
- }
- foreach (array_keys(self::$localeBoolean) as $excelBoolean) {
- self::$functionReplaceToExcel[] = '$1' . trim($excelBoolean) . '$2';
- }
- }
-
- return self::translateFormula(self::$functionReplaceFromLocale, self::$functionReplaceToExcel, $formula, self::$localeArgumentSeparator, ',');
- }
-
- public static function localeFunc(string $function): string
- {
- if (self::$localeLanguage !== 'en_us') {
- $functionName = trim($function, '(');
- if (isset(self::$localeFunctions[$functionName])) {
- $brace = ($functionName != $function);
- $function = self::$localeFunctions[$functionName];
- if ($brace) {
- $function .= '(';
- }
- }
- }
-
- return $function;
- }
-
/**
* Wrap string values in quotes.
*/
@@ -3438,15 +474,12 @@ class Calculation
return null;
}
- $returnArrayAsType = self::$returnArrayAsType;
if ($resetLog) {
// Initialise the logging settings if requested
$this->formulaError = null;
$this->debugLog->clearLog();
$this->cyclicReferenceStack->clear();
$this->cyclicFormulaCounter = 1;
-
- self::$returnArrayAsType = self::RETURN_ARRAY_AS_ARRAY;
}
// Execute the calculation for the cell formula
@@ -3459,7 +492,15 @@ class Calculation
$cellAddress = null;
try {
- $result = self::unwrapResult($this->_calculateFormulaValue($cell->getValue(), $cell->getCoordinate(), $cell));
+ $value = $cell->getValue();
+ if (is_string($value) && $cell->getDataType() === DataType::TYPE_FORMULA) {
+ $value = preg_replace_callback(
+ self::CALCULATION_REGEXP_CELLREF_SPILL,
+ fn (array $matches) => 'ANCHORARRAY(' . substr($matches[0], 0, -1) . ')',
+ $value
+ );
+ }
+ $result = self::unwrapResult($this->_calculateFormulaValue($value, $cell->getCoordinate(), $cell)); //* @phpstan-ignore-line
if ($this->spreadsheet === null) {
throw new Exception('null spreadsheet in calculateCellValue');
}
@@ -3468,6 +509,7 @@ class Calculation
if ($cellAddress === null) {
throw new Exception('null cellAddress in calculateCellValue');
}
+ /** @var array{sheet: string, cell: string} $cellAddress */
$testSheet = $this->spreadsheet->getSheetByName($cellAddress['sheet']);
if ($testSheet === null) {
throw new Exception('worksheet not found in calculateCellValue');
@@ -3478,8 +520,10 @@ class Calculation
$cellAddress = array_pop($this->cellStack);
}
if ($this->spreadsheet !== null && is_array($cellAddress) && array_key_exists('sheet', $cellAddress)) {
- $testSheet = $this->spreadsheet->getSheetByName($cellAddress['sheet']);
+ $sheetName = $cellAddress['sheet'] ?? null;
+ $testSheet = is_string($sheetName) ? $this->spreadsheet->getSheetByName($sheetName) : null;
if ($testSheet !== null && array_key_exists('cell', $cellAddress)) {
+ /** @var array{cell: string} $cellAddress */
$testSheet->getCell($cellAddress['cell']);
}
}
@@ -3487,31 +531,13 @@ class Calculation
throw new Exception($e->getMessage(), $e->getCode(), $e);
}
- if ((is_array($result)) && (self::$returnArrayAsType != self::RETURN_ARRAY_AS_ARRAY)) {
- self::$returnArrayAsType = $returnArrayAsType;
+ if (is_array($result) && $this->getInstanceArrayReturnType() !== self::RETURN_ARRAY_AS_ARRAY) {
$testResult = Functions::flattenArray($result);
- if (self::$returnArrayAsType == self::RETURN_ARRAY_AS_ERROR) {
+ if ($this->getInstanceArrayReturnType() == self::RETURN_ARRAY_AS_ERROR) {
return ExcelError::VALUE();
}
- // If there's only a single cell in the array, then we allow it
- if (count($testResult) != 1) {
- // If keys are numeric, then it's a matrix result rather than a cell range result, so we permit it
- $r = array_keys($result);
- $r = array_shift($r);
- if (!is_numeric($r)) {
- return ExcelError::VALUE();
- }
- if (is_array($result[$r])) {
- $c = array_keys($result[$r]);
- $c = array_shift($c);
- if (!is_numeric($c)) {
- return ExcelError::VALUE();
- }
- }
- }
$result = array_shift($testResult);
}
- self::$returnArrayAsType = $returnArrayAsType;
if ($result === null && $cell->getWorksheet()->getSheetView()->getShowZeros()) {
return 0;
@@ -3526,9 +552,16 @@ class Calculation
* Validate and parse a formula string.
*
* @param string $formula Formula to parse
+ *
+ * @return array|bool
*/
public function parseFormula(string $formula): array|bool
{
+ $formula = preg_replace_callback(
+ self::CALCULATION_REGEXP_CELLREF_SPILL,
+ fn (array $matches) => 'ANCHORARRAY(' . substr($matches[0], 0, -1) . ')',
+ $formula
+ ) ?? $formula;
// Basic validation that this is indeed a formula
// We return an empty array if not
$formula = trim($formula);
@@ -3631,7 +664,7 @@ class Calculation
// Basic validation that this is indeed a formula
// We simply return the cell value if not
$formula = trim($formula);
- if ($formula[0] != '=') {
+ if ($formula === '' || $formula[0] !== '=') {
return self::wrapResult($formula);
}
$formula = ltrim(substr($formula, 1));
@@ -3688,22 +721,35 @@ class Calculation
* Ensure that paired matrix operands are both matrices and of the same size.
*
* @param mixed $operand1 First matrix operand
+ *
+ * @param-out mixed[] $operand1
+ *
* @param mixed $operand2 Second matrix operand
+ *
+ * @param-out mixed[] $operand2
+ *
* @param int $resize Flag indicating whether the matrices should be resized to match
* and (if so), whether the smaller dimension should grow or the
* larger should shrink.
* 0 = no resize
* 1 = shrink to fit
* 2 = extend to fit
+ *
+ * @return mixed[]
*/
- private static function checkMatrixOperands(mixed &$operand1, mixed &$operand2, int $resize = 1): array
+ public static function checkMatrixOperands(mixed &$operand1, mixed &$operand2, int $resize = 1): array
{
// Examine each of the two operands, and turn them into an array if they aren't one already
// Note that this function should only be called if one or both of the operand is already an array
if (!is_array($operand1)) {
- [$matrixRows, $matrixColumns] = self::getMatrixDimensions($operand2);
- $operand1 = array_fill(0, $matrixRows, array_fill(0, $matrixColumns, $operand1));
- $resize = 0;
+ if (is_array($operand2)) {
+ [$matrixRows, $matrixColumns] = self::getMatrixDimensions($operand2);
+ $operand1 = array_fill(0, $matrixRows, array_fill(0, $matrixColumns, $operand1));
+ $resize = 0;
+ } else {
+ $operand1 = [$operand1];
+ $operand2 = [$operand2];
+ }
} elseif (!is_array($operand2)) {
[$matrixRows, $matrixColumns] = self::getMatrixDimensions($operand1);
$operand2 = array_fill(0, $matrixRows, array_fill(0, $matrixColumns, $operand2));
@@ -3712,7 +758,9 @@ class Calculation
[$matrix1Rows, $matrix1Columns] = self::getMatrixDimensions($operand1);
[$matrix2Rows, $matrix2Columns] = self::getMatrixDimensions($operand2);
- if (($matrix1Rows == $matrix2Columns) && ($matrix2Rows == $matrix1Columns)) {
+ if ($resize === 3) {
+ $resize = 2;
+ } elseif (($matrix1Rows == $matrix2Columns) && ($matrix2Rows == $matrix1Columns)) {
$resize = 1;
}
@@ -3721,6 +769,8 @@ class Calculation
self::resizeMatricesExtend($operand1, $operand2, $matrix1Rows, $matrix1Columns, $matrix2Rows, $matrix2Columns);
} elseif ($resize == 1) {
// Given two matrices of (potentially) unequal size, convert the larger in each dimension to match the smaller
+ /** @var mixed[][] $operand1 */
+ /** @var mixed[][] $operand2 */
self::resizeMatricesShrink($operand1, $operand2, $matrix1Rows, $matrix1Columns, $matrix2Rows, $matrix2Columns);
}
[$matrix1Rows, $matrix1Columns] = self::getMatrixDimensions($operand1);
@@ -3732,7 +782,7 @@ class Calculation
/**
* Read the dimensions of a matrix, and re-index it with straight numeric keys starting from row 0, column 0.
*
- * @param array $matrix matrix operand
+ * @param mixed[] $matrix matrix operand
*
* @return int[] An array comprising the number of rows, and number of columns
*/
@@ -3757,8 +807,8 @@ class Calculation
/**
* Ensure that paired matrix operands are both matrices of the same size.
*
- * @param array $matrix1 First matrix operand
- * @param array $matrix2 Second matrix operand
+ * @param mixed[][] $matrix1 First matrix operand
+ * @param mixed[][] $matrix2 Second matrix operand
* @param int $matrix1Rows Row size of first matrix operand
* @param int $matrix1Columns Column size of first matrix operand
* @param int $matrix2Rows Row size of second matrix operand
@@ -3800,8 +850,8 @@ class Calculation
/**
* Ensure that paired matrix operands are both matrices of the same size.
*
- * @param array $matrix1 First matrix operand
- * @param array $matrix2 Second matrix operand
+ * @param mixed[] $matrix1 First matrix operand
+ * @param mixed[] $matrix2 Second matrix operand
* @param int $matrix1Rows Row size of first matrix operand
* @param int $matrix1Columns Column size of first matrix operand
* @param int $matrix2Rows Row size of second matrix operand
@@ -3812,15 +862,16 @@ class Calculation
if (($matrix2Columns < $matrix1Columns) || ($matrix2Rows < $matrix1Rows)) {
if ($matrix2Columns < $matrix1Columns) {
for ($i = 0; $i < $matrix2Rows; ++$i) {
- $x = $matrix2[$i][$matrix2Columns - 1];
+ /** @var mixed[][] $matrix2 */
+ $x = ($matrix2Columns === 1) ? $matrix2[$i][0] : null;
for ($j = $matrix2Columns; $j < $matrix1Columns; ++$j) {
$matrix2[$i][$j] = $x;
}
}
}
if ($matrix2Rows < $matrix1Rows) {
- $x = $matrix2[$matrix2Rows - 1];
- for ($i = 0; $i < $matrix1Rows; ++$i) {
+ $x = ($matrix2Rows === 1) ? $matrix2[0] : array_fill(0, $matrix2Columns, null);
+ for ($i = $matrix2Rows; $i < $matrix1Rows; ++$i) {
$matrix2[$i] = $x;
}
}
@@ -3829,15 +880,16 @@ class Calculation
if (($matrix1Columns < $matrix2Columns) || ($matrix1Rows < $matrix2Rows)) {
if ($matrix1Columns < $matrix2Columns) {
for ($i = 0; $i < $matrix1Rows; ++$i) {
- $x = $matrix1[$i][$matrix1Columns - 1];
+ /** @var mixed[][] $matrix1 */
+ $x = ($matrix1Columns === 1) ? $matrix1[$i][0] : null;
for ($j = $matrix1Columns; $j < $matrix2Columns; ++$j) {
$matrix1[$i][$j] = $x;
}
}
}
if ($matrix1Rows < $matrix2Rows) {
- $x = $matrix1[$matrix1Rows - 1];
- for ($i = 0; $i < $matrix2Rows; ++$i) {
+ $x = ($matrix1Rows === 1) ? $matrix1[0] : array_fill(0, $matrix1Columns, null);
+ for ($i = $matrix1Rows; $i < $matrix2Rows; ++$i) {
$matrix1[$i] = $x;
}
}
@@ -3906,6 +958,7 @@ class Calculation
} elseif (is_array($value)) {
$typeString = 'a matrix';
} else {
+ /** @var string $value */
if ($value == '') {
return 'an empty string';
} elseif ($value[0] == '#') {
@@ -3914,20 +967,20 @@ class Calculation
$typeString = 'a string';
}
- return $typeString . ' with a value of ' . $this->showValue($value);
+ return $typeString . ' with a value of ' . StringHelper::convertToString($this->showValue($value));
}
return null;
}
+ private const MATRIX_REPLACE_FROM = [self::FORMULA_OPEN_MATRIX_BRACE, ';', self::FORMULA_CLOSE_MATRIX_BRACE];
+ private const MATRIX_REPLACE_TO = ['MKMATRIX(MKMATRIX(', '),MKMATRIX(', '))'];
+
/**
* @return false|string False indicates an error
*/
private function convertMatrixReferences(string $formula): false|string
{
- static $matrixReplaceFrom = [self::FORMULA_OPEN_MATRIX_BRACE, ';', self::FORMULA_CLOSE_MATRIX_BRACE];
- static $matrixReplaceTo = ['MKMATRIX(MKMATRIX(', '),MKMATRIX(', '))'];
-
// Convert any Excel matrix references to the MKMATRIX() function
if (str_contains($formula, self::FORMULA_OPEN_MATRIX_BRACE)) {
// If there is the possibility of braces within a quoted string, then we don't treat those as matrix indicators
@@ -3944,7 +997,7 @@ class Calculation
if ($notWithinQuotes === true) {
$openCount += substr_count($value, self::FORMULA_OPEN_MATRIX_BRACE);
$closeCount += substr_count($value, self::FORMULA_CLOSE_MATRIX_BRACE);
- $value = str_replace($matrixReplaceFrom, $matrixReplaceTo, $value);
+ $value = str_replace(self::MATRIX_REPLACE_FROM, self::MATRIX_REPLACE_TO, $value);
}
}
unset($value);
@@ -3954,7 +1007,7 @@ class Calculation
// If there's no quoted strings, then we do a simple count/replace
$openCount = substr_count($formula, self::FORMULA_OPEN_MATRIX_BRACE);
$closeCount = substr_count($formula, self::FORMULA_CLOSE_MATRIX_BRACE);
- $formula = str_replace($matrixReplaceFrom, $matrixReplaceTo, $formula);
+ $formula = str_replace(self::MATRIX_REPLACE_FROM, self::MATRIX_REPLACE_TO, $formula);
}
// Trap for mismatched braces and trigger an appropriate error
if ($openCount < $closeCount) {
@@ -3975,32 +1028,18 @@ class Calculation
return $formula;
}
- /**
- * Binary Operators.
- * These operators always work on two values.
- * Array key is the operator, the value indicates whether this is a left or right associative operator.
- */
- private static array $operatorAssociativity = [
- '^' => 0, // Exponentiation
- '*' => 0, '/' => 0, // Multiplication and Division
- '+' => 0, '-' => 0, // Addition and Subtraction
- '&' => 0, // Concatenation
- '∪' => 0, '∩' => 0, ':' => 0, // Union, Intersect and Range
- '>' => 0, '<' => 0, '=' => 0, '>=' => 0, '<=' => 0, '<>' => 0, // Comparison
- ];
-
/**
* Comparison (Boolean) Operators.
* These operators work on two values, but always return a boolean result.
*/
- private static array $comparisonOperators = ['>' => true, '<' => true, '=' => true, '>=' => true, '<=' => true, '<>' => true];
+ private const COMPARISON_OPERATORS = ['>' => true, '<' => true, '=' => true, '>=' => true, '<=' => true, '<>' => true];
/**
* Operator Precedence.
* This list includes all valid operators, whether binary (including boolean) or unary (such as %).
* Array key is the operator, the value is its precedence.
*/
- private static array $operatorPrecedence = [
+ private const OPERATOR_PRECEDENCE = [
':' => 9, // Range
'∩' => 8, // Intersect
'∪' => 7, // Union
@@ -4013,8 +1052,6 @@ class Calculation
'>' => 0, '<' => 0, '=' => 0, '>=' => 0, '<=' => 0, '<>' => 0, // Comparison
];
- // Convert infix to postfix notation
-
/**
* @return array|false
*/
@@ -4023,6 +1060,7 @@ class Calculation
if (($formula = $this->convertMatrixReferences(trim($formula))) === false) {
return false;
}
+ $phpSpreadsheetFunctions = &self::getFunctionsAddress();
// If we're using cell caching, then $pCell may well be flushed back to the cache (which detaches the parent worksheet),
// so we store the parent worksheet so that we can re-attach it when necessary
@@ -4059,7 +1097,7 @@ class Calculation
$opCharacter = $formula[$index]; // Get the first character of the value at the current index position
// Check for two-character operators (e.g. >=, <=, <>)
- if ((isset(self::$comparisonOperators[$opCharacter])) && (strlen($formula) > $index) && isset($formula[$index + 1], self::$comparisonOperators[$formula[$index + 1]])) {
+ if ((isset(self::COMPARISON_OPERATORS[$opCharacter])) && (strlen($formula) > $index) && isset($formula[$index + 1], self::COMPARISON_OPERATORS[$formula[$index + 1]])) {
$opCharacter .= $formula[++$index];
}
// Find out if we're currently at the beginning of a number, variable, cell/row/column reference,
@@ -4081,12 +1119,7 @@ class Calculation
// We have to explicitly deny a tilde, union or intersect because they are legal
return $this->raiseFormulaError("Formula Error: Illegal character '~'"); // on the stack but not in the input expression
} elseif ((isset(self::CALCULATION_OPERATORS[$opCharacter]) || $isOperandOrFunction) && $expectingOperator) { // Are we putting an operator on the stack?
- while (
- $stack->count() > 0
- && ($o2 = $stack->last())
- && isset(self::CALCULATION_OPERATORS[$o2['value']])
- && @(self::$operatorAssociativity[$opCharacter] ? self::$operatorPrecedence[$opCharacter] < self::$operatorPrecedence[$o2['value']] : self::$operatorPrecedence[$opCharacter] <= self::$operatorPrecedence[$o2['value']])
- ) {
+ while (self::swapOperands($stack, $opCharacter)) {
$output[] = $stack->pop(); // Swap operands and higher precedence operators from the stack to the output
}
@@ -4106,7 +1139,7 @@ class Calculation
// call or a parenthesis
$this->branchPruner->decrementDepth();
- if (is_array($d) && preg_match('/^' . self::CALCULATION_REGEXP_FUNCTION . '$/miu', $d['value'], $matches)) {
+ if (is_array($d) && preg_match('/^' . self::CALCULATION_REGEXP_FUNCTION . '$/miu', StringHelper::convertToString($d['value']), $matches)) {
// Did this parenthesis just close a function?
try {
$this->branchPruner->closingBrace($d['value']);
@@ -4121,8 +1154,8 @@ class Calculation
$output[] = $stack->pop(); // Pop the function and push onto the output
if (isset(self::$controlFunctions[$functionName])) {
$expectedArgumentCount = self::$controlFunctions[$functionName]['argumentCount'];
- } elseif (isset(self::$phpSpreadsheetFunctions[$functionName])) {
- $expectedArgumentCount = self::$phpSpreadsheetFunctions[$functionName]['argumentCount'];
+ } elseif (isset($phpSpreadsheetFunctions[$functionName])) {
+ $expectedArgumentCount = $phpSpreadsheetFunctions[$functionName]['argumentCount'];
} else { // did we somehow push a non-function on the stack? this should never happen
return $this->raiseFormulaError('Formula Error: Internal error, non-function on stack');
}
@@ -4131,9 +1164,9 @@ class Calculation
$expectedArgumentCountString = null;
if (is_numeric($expectedArgumentCount)) {
if ($expectedArgumentCount < 0) {
- if ($argumentCount > abs($expectedArgumentCount)) {
+ if ($argumentCount > abs($expectedArgumentCount + 0)) {
$argumentCountError = true;
- $expectedArgumentCountString = 'no more than ' . abs($expectedArgumentCount);
+ $expectedArgumentCountString = 'no more than ' . abs($expectedArgumentCount + 0);
}
} else {
if ($argumentCount != $expectedArgumentCount) {
@@ -4141,9 +1174,11 @@ class Calculation
$expectedArgumentCountString = $expectedArgumentCount;
}
}
- } elseif ($expectedArgumentCount != '*') {
- preg_match('/(\d*)([-+,])(\d*)/', $expectedArgumentCount, $argMatch);
- switch ($argMatch[2] ?? '') {
+ } elseif (is_string($expectedArgumentCount) && $expectedArgumentCount !== '*') {
+ if (1 !== preg_match('/(\d*)([-+,])(\d*)/', $expectedArgumentCount, $argMatch)) {
+ $argMatch = ['', '', '', ''];
+ }
+ switch ($argMatch[2]) {
case '+':
if ($argumentCount < $argMatch[1]) {
$argumentCountError = true;
@@ -4168,6 +1203,7 @@ class Calculation
}
}
if ($argumentCountError) {
+ /** @var int $argumentCount */
return $this->raiseFormulaError("Formula Error: Wrong number of arguments for $functionName() function: $argumentCount given, " . $expectedArgumentCountString . ' expected');
}
}
@@ -4189,7 +1225,9 @@ class Calculation
}
// make sure there was a function
$d = $stack->last(2);
- if (!preg_match('/^' . self::CALCULATION_REGEXP_FUNCTION . '$/miu', $d['value'] ?? '', $matches)) {
+ /** @var string */
+ $temp = $d['value'] ?? '';
+ if (!preg_match('/^' . self::CALCULATION_REGEXP_FUNCTION . '$/miu', $temp, $matches)) {
// Can we inject a dummy function at this point so that the braces at least have some context
// because at least the braces are paired up (at this stage in the formula)
// MS Excel allows this if the content is cell references; but doesn't allow actual values,
@@ -4197,7 +1235,7 @@ class Calculation
return $this->raiseFormulaError('Formula Error: Unexpected ,');
}
- /** @var array $d */
+ /** @var array $d */
$d = $stack->pop();
++$d['value']; // increment the argument count
@@ -4216,12 +1254,12 @@ class Calculation
// do we now have a function/variable/number?
$expectingOperator = true;
$expectingOperand = false;
- $val = $match[1];
+ $val = $match[1] ?? ''; //* @phpstan-ignore-line
$length = strlen($val);
if (preg_match('/^' . self::CALCULATION_REGEXP_FUNCTION . '$/miu', $val, $matches)) {
$val = (string) preg_replace('/\s/u', '', $val);
- if (isset(self::$phpSpreadsheetFunctions[strtoupper($matches[1])]) || isset(self::$controlFunctions[strtoupper($matches[1])])) { // it's a function
+ if (isset($phpSpreadsheetFunctions[strtoupper($matches[1])]) || isset(self::$controlFunctions[strtoupper($matches[1])])) { // it's a function
$valToUpper = strtoupper($val);
} else {
$valToUpper = 'NAME.ERROR(';
@@ -4257,6 +1295,7 @@ class Calculation
// Do we have chained range operators?
$rangeStartCellRef = $output[count($output) - 2]['value'] ?? '';
}
+ /** @var string $rangeStartCellRef */
preg_match('/^' . self::CALCULATION_REGEXP_CELLREF . '$/miu', $rangeStartCellRef, $rangeStartMatches);
if (array_key_exists(2, $rangeStartMatches)) {
if ($rangeStartMatches[2] > '') {
@@ -4271,8 +1310,9 @@ class Calculation
// Do we have chained range operators?
$rangeStartCellRef = $output[count($output) - 2]['value'] ?? '';
}
+ /** @var string $rangeStartCellRef */
preg_match('/^' . self::CALCULATION_REGEXP_CELLREF . '$/miu', $rangeStartCellRef, $rangeStartMatches);
- if ($rangeStartMatches[2] !== $matches[2]) {
+ if (isset($rangeStartMatches[2]) && $rangeStartMatches[2] !== $matches[2]) {
return $this->raiseFormulaError('3D Range references are not yet supported');
}
}
@@ -4342,7 +1382,9 @@ class Calculation
if ($rangeWS1 !== '') {
$rangeWS1 .= '!';
}
- $rangeSheetRef = trim($rangeSheetRef, "'");
+ if (str_starts_with($rangeSheetRef, "'")) {
+ $rangeSheetRef = Worksheet::unApostrophizeTitle($rangeSheetRef);
+ }
[$rangeWS2, $val] = Worksheet::extractSheetTitle($val, true);
if ($rangeWS2 !== '') {
$rangeWS2 .= '!';
@@ -4358,11 +1400,10 @@ class Calculation
if (ctype_digit($val) && $val <= 1048576) {
// Row range
$stackItemType = 'Row Reference';
- /** @var int $valx */
$valx = $val;
$endRowColRef = ($refSheet !== null) ? $refSheet->getHighestDataColumn($valx) : AddressRange::MAX_COLUMN; // Max 16,384 columns for Excel2007
$val = "{$rangeWS2}{$endRowColRef}{$val}";
- } elseif (ctype_alpha($val) && strlen($val ?? '') <= 3) {
+ } elseif (ctype_alpha($val) && strlen($val) <= 3) {
// Column range
$stackItemType = 'Column Reference';
$endRowColRef = ($refSheet !== null) ? $refSheet->getHighestDataRow($val) : AddressRange::MAX_ROW; // Max 1,048,576 rows for Excel2007
@@ -4372,15 +1413,15 @@ class Calculation
}
} elseif ($opCharacter === self::FORMULA_STRING_QUOTE) {
// UnEscape any quotes within the string
- $val = self::wrapResult(str_replace('""', self::FORMULA_STRING_QUOTE, self::unwrapResult($val)));
- } elseif (isset(self::$excelConstants[trim(strtoupper($val))])) {
+ $val = self::wrapResult(str_replace('""', self::FORMULA_STRING_QUOTE, StringHelper::convertToString(self::unwrapResult($val))));
+ } elseif (isset(self::EXCEL_CONSTANTS[trim(strtoupper($val))])) {
$stackItemType = 'Constant';
$excelConstant = trim(strtoupper($val));
- $val = self::$excelConstants[$excelConstant];
+ $val = self::EXCEL_CONSTANTS[$excelConstant];
$stackItemReference = $excelConstant;
} elseif (($localeConstant = array_search(trim(strtoupper($val)), self::$localeBoolean)) !== false) {
$stackItemType = 'Constant';
- $val = self::$excelConstants[$localeConstant];
+ $val = self::EXCEL_CONSTANTS[$localeConstant];
$stackItemReference = $localeConstant;
} elseif (
preg_match('/^' . self::CALCULATION_REGEXP_ROW_RANGE . '/miu', substr($formula, $index), $rowRangeReference)
@@ -4480,12 +1521,7 @@ class Calculation
&& ($output[$countOutputMinus1]['type'] === Operands\StructuredReference::NAME || $output[$countOutputMinus1]['type'] === 'Value')
)
) {
- while (
- $stack->count() > 0
- && ($o2 = $stack->last())
- && isset(self::CALCULATION_OPERATORS[$o2['value']])
- && @(self::$operatorAssociativity[$opCharacter] ? self::$operatorPrecedence[$opCharacter] < self::$operatorPrecedence[$o2['value']] : self::$operatorPrecedence[$opCharacter] <= self::$operatorPrecedence[$o2['value']])
- ) {
+ while (self::swapOperands($stack, $opCharacter)) {
$output[] = $stack->pop(); // Swap operands and higher precedence operators from the stack to the output
}
$stack->push('Binary Operator', '∩'); // Put an Intersect Operator on the stack
@@ -4496,7 +1532,7 @@ class Calculation
while (($op = $stack->pop()) !== null) {
// pop everything off the stack and push onto output
- if ((is_array($op) && $op['value'] == '(')) {
+ if ($op['value'] == '(') {
return $this->raiseFormulaError("Formula Error: Expecting ')'"); // if there are any opening braces on the stack, then braces were unbalanced
}
$output[] = $op;
@@ -4505,6 +1541,7 @@ class Calculation
return $output;
}
+ /** @param mixed[] $operandData */
private static function dataTestReference(array &$operandData): mixed
{
$operand = $operandData['value'];
@@ -4527,18 +1564,28 @@ class Calculation
return $operand;
}
+ private static int $matchIndex8 = 8;
+
+ private static int $matchIndex9 = 9;
+
+ private static int $matchIndex10 = 10;
+
/**
- * @return array|false
+ * @param array|false $tokens
+ *
+ * @return array|false|string
*/
- private function processTokenStack(mixed $tokens, ?string $cellID = null, ?Cell $cell = null)
+ private function processTokenStack(false|array $tokens, ?string $cellID = null, ?Cell $cell = null)
{
if ($tokens === false) {
return false;
}
+ $phpSpreadsheetFunctions = &self::getFunctionsAddress();
// If we're using cell caching, then $pCell may well be flushed back to the cache (which detaches the parent cell collection),
// so we store the parent cell collection so that we can re-attach it when necessary
$pCellWorksheet = ($cell !== null) ? $cell->getWorksheet() : null;
+ $originalCoordinate = $cell?->getCoordinate();
$pCellParent = ($cell !== null) ? $cell->getParent() : null;
$stack = new Stack($this->branchPruner);
@@ -4547,11 +1594,18 @@ class Calculation
// help us to know when pruning ['branchTestId' => true/false]
$branchStore = [];
// Loop through each token in turn
- foreach ($tokens as $tokenData) {
+ foreach ($tokens as $tokenIdx => $tokenData) {
+ /** @var mixed[] $tokenData */
+ $this->processingAnchorArray = false;
+ if ($tokenData['type'] === 'Cell Reference' && isset($tokens[$tokenIdx + 1]) && $tokens[$tokenIdx + 1]['type'] === 'Operand Count for Function ANCHORARRAY()') { //* @phpstan-ignore-line
+ $this->processingAnchorArray = true;
+ }
$token = $tokenData['value'];
// Branch pruning: skip useless resolutions
+ /** @var ?string */
$storeKey = $tokenData['storeKey'] ?? null;
if ($this->branchPruningEnabled && isset($tokenData['onlyIf'])) {
+ /** @var string */
$onlyIfStoreKey = $tokenData['onlyIf'];
$storeValue = $branchStore[$onlyIfStoreKey] ?? null;
$storeValueAsBool = ($storeValue === null)
@@ -4566,7 +1620,9 @@ class Calculation
&& (!$storeValueAsBool || Information\ErrorValue::isError($storeValue) || ($storeValue === 'Pruned branch'))
) {
// If branching value is not true, we don't need to compute
+ /** @var string $onlyIfStoreKey */
if (!isset($fakedForBranchPruning['onlyIf-' . $onlyIfStoreKey])) {
+ /** @var string $token */
$stack->push('Value', 'Pruned branch (only if ' . $onlyIfStoreKey . ') ' . $token);
$fakedForBranchPruning['onlyIf-' . $onlyIfStoreKey] = true;
}
@@ -4584,6 +1640,7 @@ class Calculation
}
if ($this->branchPruningEnabled && isset($tokenData['onlyIfNot'])) {
+ /** @var string */
$onlyIfNotStoreKey = $tokenData['onlyIfNot'];
$storeValue = $branchStore[$onlyIfNotStoreKey] ?? null;
$storeValueAsBool = ($storeValue === null)
@@ -4599,6 +1656,7 @@ class Calculation
) {
// If branching value is true, we don't need to compute
if (!isset($fakedForBranchPruning['onlyIfNot-' . $onlyIfNotStoreKey])) {
+ /** @var string $token */
$stack->push('Value', 'Pruned branch (only if not ' . $onlyIfNotStoreKey . ') ' . $token);
$fakedForBranchPruning['onlyIfNot-' . $onlyIfNotStoreKey] = true;
}
@@ -4680,27 +1738,43 @@ class Calculation
break;
// Binary Operators
case ':': // Range
+ if ($operand1Data['type'] === 'Error') {
+ $stack->push($operand1Data['type'], $operand1Data['value'], null);
+
+ break;
+ }
+ if ($operand2Data['type'] === 'Error') {
+ $stack->push($operand2Data['type'], $operand2Data['value'], null);
+
+ break;
+ }
if ($operand1Data['type'] === 'Defined Name') {
+ /** @var array{reference: string} $operand1Data */
if (preg_match('/$' . self::CALCULATION_REGEXP_DEFINEDNAME . '^/mui', $operand1Data['reference']) !== false && $this->spreadsheet !== null) {
+ /** @var string[] $operand1Data */
$definedName = $this->spreadsheet->getNamedRange($operand1Data['reference']);
if ($definedName !== null) {
$operand1Data['reference'] = $operand1Data['value'] = str_replace('$', '', $definedName->getValue());
}
}
}
+ /** @var array{reference?: ?string} $operand1Data */
if (str_contains($operand1Data['reference'] ?? '', '!')) {
- [$sheet1, $operand1Data['reference']] = Worksheet::extractSheetTitle($operand1Data['reference'], true);
+ [$sheet1, $operand1Data['reference']] = Worksheet::extractSheetTitle($operand1Data['reference'], true, true);
} else {
$sheet1 = ($pCellWorksheet !== null) ? $pCellWorksheet->getTitle() : '';
}
- $sheet1 ??= '';
+ //$sheet1 ??= ''; // phpstan level 10 says this is unneeded
- [$sheet2, $operand2Data['reference']] = Worksheet::extractSheetTitle($operand2Data['reference'], true);
+ /** @var string */
+ $op2ref = $operand2Data['reference'];
+ [$sheet2, $operand2Data['reference']] = Worksheet::extractSheetTitle($op2ref, true, true);
if (empty($sheet2)) {
$sheet2 = $sheet1;
}
- if (trim($sheet1, "'") === trim($sheet2, "'")) {
+ if ($sheet1 === $sheet2) {
+ /** @var array{reference: ?string, value: string|string[]} $operand1Data */
if ($operand1Data['reference'] === null && $cell !== null) {
if (is_array($operand1Data['value'])) {
$operand1Data['reference'] = $cell->getCoordinate();
@@ -4712,6 +1786,7 @@ class Calculation
$operand1Data['reference'] = $operand1Data['value'] . $cell->getRow();
}
}
+ /** @var array{reference: ?string, value: string|string[]} $operand2Data */
if ($operand2Data['reference'] === null && $cell !== null) {
if (is_array($operand2Data['value'])) {
$operand2Data['reference'] = $cell->getCoordinate();
@@ -4742,7 +1817,7 @@ class Calculation
if ($breakNeeded) {
break;
}
- $cellRef = Coordinate::stringFromColumnIndex(min($oCol) + 1) . min($oRow) . ':' . Coordinate::stringFromColumnIndex(max($oCol) + 1) . max($oRow);
+ $cellRef = Coordinate::stringFromColumnIndex(min($oCol) + 1) . min($oRow) . ':' . Coordinate::stringFromColumnIndex(max($oCol) + 1) . max($oRow); // @phpstan-ignore-line
if ($pCellParent !== null && $this->spreadsheet !== null) {
$cellValue = $this->extractCellRange($cellRef, $this->spreadsheet->getSheetByName($sheet1), false);
} else {
@@ -4786,15 +1861,19 @@ class Calculation
for ($row = 0; $row < $rows; ++$row) {
for ($column = 0; $column < $columns; ++$column) {
+ /** @var mixed[][] $operand1 */
$op1x = self::boolToString($operand1[$row][$column]);
+ /** @var mixed[][] $operand2 */
$op2x = self::boolToString($operand2[$row][$column]);
if (Information\ErrorValue::isError($op1x)) {
// no need to do anything
} elseif (Information\ErrorValue::isError($op2x)) {
$operand1[$row][$column] = $op2x;
} else {
+ /** @var string $op1x */
+ /** @var string $op2x */
$operand1[$row][$column]
- = Shared\StringHelper::substring(
+ = StringHelper::substring(
$op1x . $op2x,
0,
DataType::MAX_STRING_LENGTH
@@ -4804,17 +1883,18 @@ class Calculation
}
$result = $operand1;
} else {
- // In theory, we should truncate here.
- // But I can't figure out a formula
- // using the concatenation operator
- // with literals that fits in 32K,
- // so I don't think we can overflow here.
if (Information\ErrorValue::isError($operand1)) {
$result = $operand1;
} elseif (Information\ErrorValue::isError($operand2)) {
$result = $operand2;
} else {
- $result = self::FORMULA_STRING_QUOTE . str_replace('""', self::FORMULA_STRING_QUOTE, self::unwrapResult($operand1) . self::unwrapResult($operand2)) . self::FORMULA_STRING_QUOTE;
+ $result = str_replace('""', self::FORMULA_STRING_QUOTE, self::unwrapResult($operand1) . self::unwrapResult($operand2)); //* @phpstan-ignore-line
+ $result = StringHelper::substring(
+ $result,
+ 0,
+ DataType::MAX_STRING_LENGTH
+ );
+ $result = self::FORMULA_STRING_QUOTE . $result . self::FORMULA_STRING_QUOTE;
}
}
$this->debugLog->writeDebugLog('Evaluation Result is %s', $this->showTypeDetails($result));
@@ -4826,6 +1906,8 @@ class Calculation
break;
case '∩': // Intersect
+ /** @var mixed[][] $operand1 */
+ /** @var mixed[][] $operand2 */
$rowIntersect = array_intersect_key($operand1, $operand2);
$cellIntersect = $oCol = $oRow = [];
foreach (array_keys($rowIntersect) as $row) {
@@ -4839,8 +1921,8 @@ class Calculation
$this->debugLog->writeDebugLog('Evaluation Result is %s', $this->showTypeDetails($cellIntersect));
$stack->push('Error', ExcelError::null(), null);
} else {
- $cellRef = Coordinate::stringFromColumnIndex(min($oCol) + 1) . min($oRow) . ':'
- . Coordinate::stringFromColumnIndex(max($oCol) + 1) . max($oRow);
+ $cellRef = Coordinate::stringFromColumnIndex(min($oCol) + 1) . min($oRow) . ':' // @phpstan-ignore-line
+ . Coordinate::stringFromColumnIndex(max($oCol) + 1) . max($oRow); // @phpstan-ignore-line
$this->debugLog->writeDebugLog('Evaluation Result is %s', $this->showTypeDetails($cellIntersect));
$stack->push('Value', $cellIntersect, $cellRef);
}
@@ -4866,8 +1948,11 @@ class Calculation
[$rows, $columns] = self::checkMatrixOperands($result, $operand2, 0);
for ($row = 0; $row < $rows; ++$row) {
for ($column = 0; $column < $columns; ++$column) {
+ /** @var mixed[][] $result */
if (self::isNumericOrBool($result[$row][$column])) {
- $result[$row][$column] *= $multiplier;
+ /** @var float|int|numeric-string */
+ $temp = $result[$row][$column];
+ $result[$row][$column] = $temp * $multiplier;
} else {
$result[$row][$column] = self::makeError($result[$row][$column]);
}
@@ -4882,15 +1967,20 @@ class Calculation
} else {
$this->executeNumericBinaryOperation($multiplier, $arg, '*', $stack);
}
- } elseif (preg_match('/^' . self::CALCULATION_REGEXP_CELLREF . '$/i', $token ?? '', $matches)) {
+ } elseif (preg_match('/^' . self::CALCULATION_REGEXP_CELLREF . '$/i', StringHelper::convertToString($token ?? ''), $matches)) {
$cellRef = null;
- if (isset($matches[8])) {
+ /* Phpstan says matches[8/9/10] is never set,
+ and code coverage report seems to confirm.
+ Appease PhpStan for now;
+ probably delete this block later.
+ */
+ if (isset($matches[self::$matchIndex8])) {
if ($cell === null) {
// We can't access the range, so return a REF error
$cellValue = ExcelError::REF();
} else {
- $cellRef = $matches[6] . $matches[7] . ':' . $matches[9] . $matches[10];
+ $cellRef = $matches[6] . $matches[7] . ':' . $matches[self::$matchIndex9] . $matches[self::$matchIndex10];
if ($matches[2] > '') {
$matches[2] = trim($matches[2], "\"'");
if ((str_contains($matches[2], '[')) || (str_contains($matches[2], ']'))) {
@@ -4930,6 +2020,9 @@ class Calculation
$this->debugLog->writeDebugLog('Evaluating Cell %s in worksheet %s', $cellRef, $matches[2]);
if ($pCellParent !== null && $this->spreadsheet !== null) {
$cellSheet = $this->spreadsheet->getSheetByName($matches[2]);
+ if ($cellSheet && !$cellSheet->cellExists($cellRef)) {
+ $cellSheet->setCellValue($cellRef, null);
+ }
if ($cellSheet && $cellSheet->cellExists($cellRef)) {
$cellValue = $this->extractCellRange($cellRef, $this->spreadsheet->getSheetByName($matches[2]), false);
$cell->attach($pCellParent);
@@ -4954,30 +2047,41 @@ class Calculation
}
}
+ if ($this->getInstanceArrayReturnType() === self::RETURN_ARRAY_AS_ARRAY && !$this->processingAnchorArray && is_array($cellValue)) {
+ while (is_array($cellValue)) {
+ $cellValue = array_shift($cellValue);
+ }
+ if (is_string($cellValue)) {
+ $cellValue = preg_replace('/"/', '""', $cellValue);
+ }
+ $this->debugLog->writeDebugLog('Scalar Result for cell %s is %s', $cellRef, $this->showTypeDetails($cellValue));
+ }
+ $this->processingAnchorArray = false;
$stack->push('Cell Value', $cellValue, $cellRef);
if (isset($storeKey)) {
$branchStore[$storeKey] = $cellValue;
}
- } elseif (preg_match('/^' . self::CALCULATION_REGEXP_FUNCTION . '$/miu', $token ?? '', $matches)) {
+ } elseif (preg_match('/^' . self::CALCULATION_REGEXP_FUNCTION . '$/miu', StringHelper::convertToString($token ?? ''), $matches)) {
// if the token is a function, pop arguments off the stack, hand them to the function, and push the result back on
if ($cell !== null && $pCellParent !== null) {
$cell->attach($pCellParent);
}
$functionName = $matches[1];
+ /** @var array $argCount */
$argCount = $stack->pop();
$argCount = $argCount['value'];
if ($functionName !== 'MKMATRIX') {
$this->debugLog->writeDebugLog('Evaluating Function %s() with %s argument%s', self::localeFunc($functionName), (($argCount == 0) ? 'no' : $argCount), (($argCount == 1) ? '' : 's'));
}
- if ((isset(self::$phpSpreadsheetFunctions[$functionName])) || (isset(self::$controlFunctions[$functionName]))) { // function
+ if ((isset($phpSpreadsheetFunctions[$functionName])) || (isset(self::$controlFunctions[$functionName]))) { // function
$passByReference = false;
$passCellReference = false;
$functionCall = null;
- if (isset(self::$phpSpreadsheetFunctions[$functionName])) {
- $functionCall = self::$phpSpreadsheetFunctions[$functionName]['functionCall'];
- $passByReference = isset(self::$phpSpreadsheetFunctions[$functionName]['passByReference']);
- $passCellReference = isset(self::$phpSpreadsheetFunctions[$functionName]['passCellReference']);
+ if (isset($phpSpreadsheetFunctions[$functionName])) {
+ $functionCall = $phpSpreadsheetFunctions[$functionName]['functionCall'];
+ $passByReference = isset($phpSpreadsheetFunctions[$functionName]['passByReference']);
+ $passCellReference = isset($phpSpreadsheetFunctions[$functionName]['passCellReference']);
} elseif (isset(self::$controlFunctions[$functionName])) {
$functionCall = self::$controlFunctions[$functionName]['functionCall'];
$passByReference = isset(self::$controlFunctions[$functionName]['passByReference']);
@@ -4992,11 +2096,30 @@ class Calculation
$a = $argCount - $i - 1;
if (
($passByReference)
- && (isset(self::$phpSpreadsheetFunctions[$functionName]['passByReference'][$a]))
- && (self::$phpSpreadsheetFunctions[$functionName]['passByReference'][$a])
+ && (isset($phpSpreadsheetFunctions[$functionName]['passByReference'][$a])) //* @phpstan-ignore-line
+ && ($phpSpreadsheetFunctions[$functionName]['passByReference'][$a])
) {
+ /** @var mixed[] $arg */
if ($arg['reference'] === null) {
- $args[] = $cellID;
+ $nextArg = $cellID;
+ if ($functionName === 'ISREF' && ($arg['type'] ?? '') === 'Value') {
+ if (array_key_exists('value', $arg)) {
+ $argValue = $arg['value'];
+ if (is_scalar($argValue)) {
+ $nextArg = $argValue;
+ } elseif (empty($argValue)) {
+ $nextArg = '';
+ }
+ }
+ } elseif (($arg['type'] ?? '') === 'Error') {
+ $argValue = $arg['value'];
+ if (is_scalar($argValue)) {
+ $nextArg = $argValue;
+ } elseif (empty($argValue)) {
+ $nextArg = '';
+ }
+ }
+ $args[] = $nextArg;
if ($functionName !== 'MKMATRIX') {
$argArrayVals[] = $this->showValue($cellID);
}
@@ -5007,6 +2130,7 @@ class Calculation
}
}
} else {
+ /** @var mixed[] $arg */
if ($arg['type'] === 'Empty Argument' && in_array($functionName, ['MIN', 'MINA', 'MAX', 'MAXA', 'IF'], true)) {
$emptyArguments[] = false;
$args[] = $arg['value'] = 0;
@@ -5026,7 +2150,9 @@ class Calculation
krsort($emptyArguments);
if ($argCount > 0 && is_array($functionCall)) {
- $args = $this->addDefaultArgumentValues($functionCall, $args, $emptyArguments);
+ /** @var string[] */
+ $functionCallCopy = $functionCall;
+ $args = $this->addDefaultArgumentValues($functionCallCopy, $args, $emptyArguments);
}
if (($passByReference) && ($argCount == 0)) {
@@ -5042,6 +2168,10 @@ class Calculation
}
// Process the argument with the appropriate function call
+ if ($pCellWorksheet !== null && $originalCoordinate !== null) {
+ $pCellWorksheet->getCell($originalCoordinate);
+ }
+ /** @var array|string $functionCall */
$args = $this->addCellReference($args, $passCellReference, $functionCall, $cell);
if (!is_array($functionCall)) {
@@ -5051,8 +2181,15 @@ class Calculation
unset($arg);
}
- $result = call_user_func_array($functionCall, $args);
-
+ /** @var callable $functionCall */
+ try {
+ $result = call_user_func_array($functionCall, $args);
+ } catch (TypeError $e) {
+ if (!$this->suppressFormulaErrors) {
+ throw $e;
+ }
+ $result = false;
+ }
if ($functionName !== 'MKMATRIX') {
$this->debugLog->writeDebugLog('Evaluation Result for %s() function call is %s', self::localeFunc($functionName), $this->showTypeDetails($result));
}
@@ -5063,14 +2200,16 @@ class Calculation
}
} else {
// if the token is a number, boolean, string or an Excel error, push it onto the stack
- if (isset(self::$excelConstants[strtoupper($token ?? '')])) {
- $excelConstant = strtoupper($token);
- $stack->push('Constant Value', self::$excelConstants[$excelConstant]);
+ /** @var ?string $token */
+ if (isset(self::EXCEL_CONSTANTS[strtoupper($token ?? '')])) {
+ $excelConstant = strtoupper("$token");
+ $stack->push('Constant Value', self::EXCEL_CONSTANTS[$excelConstant]);
if (isset($storeKey)) {
- $branchStore[$storeKey] = self::$excelConstants[$excelConstant];
+ $branchStore[$storeKey] = self::EXCEL_CONSTANTS[$excelConstant];
}
- $this->debugLog->writeDebugLog('Evaluating Constant %s as %s', $excelConstant, $this->showTypeDetails(self::$excelConstants[$excelConstant]));
- } elseif ((is_numeric($token)) || ($token === null) || (is_bool($token)) || ($token == '') || ($token[0] == self::FORMULA_STRING_QUOTE) || ($token[0] == '#')) {
+ $this->debugLog->writeDebugLog('Evaluating Constant %s as %s', $excelConstant, $this->showTypeDetails(self::EXCEL_CONSTANTS[$excelConstant]));
+ } elseif ((is_numeric($token)) || ($token === null) || (is_bool($token)) || ($token == '') || ($token[0] == self::FORMULA_STRING_QUOTE) || ($token[0] == '#')) { //* @phpstan-ignore-line
+ /** @var array{type: string, reference: ?string} $tokenData */
$stack->push($tokenData['type'], $token, $tokenData['reference']);
if (isset($storeKey)) {
$branchStore[$storeKey] = $token;
@@ -5078,6 +2217,9 @@ class Calculation
} elseif (preg_match('/^' . self::CALCULATION_REGEXP_DEFINEDNAME . '$/miu', $token, $matches)) {
// if the token is a named range or formula, evaluate it and push the result onto the stack
$definedName = $matches[6];
+ if (str_starts_with($definedName, '_xleta')) {
+ return Functions::NOT_YET_IMPLEMENTED;
+ }
if ($cell === null || $pCellWorksheet === null) {
return $this->raiseFormulaError("undefined name '$token'");
}
@@ -5106,10 +2248,13 @@ class Calculation
}
}
if ($namedRange === null) {
- return $this->raiseFormulaError("undefined name '$definedName'");
+ $result = ExcelError::NAME();
+ $stack->push('Error', $result, null);
+ $this->debugLog->writeDebugLog("Error $result");
+ } else {
+ $result = $this->evaluateDefinedName($cell, $namedRange, $pCellWorksheet, $stack, $specifiedWorksheet !== '');
}
- $result = $this->evaluateDefinedName($cell, $namedRange, $pCellWorksheet, $stack, $specifiedWorksheet !== '');
if (isset($storeKey)) {
$branchStore[$storeKey] = $result;
}
@@ -5122,13 +2267,14 @@ class Calculation
if ($stack->count() != 1) {
return $this->raiseFormulaError('internal error');
}
+ /** @var array|false|string> */
$output = $stack->pop();
$output = $output['value'];
return $output;
}
- private function validateBinaryOperand(mixed &$operand, mixed &$stack): bool
+ private function validateBinaryOperand(mixed &$operand, Stack &$stack): bool
{
if (is_array($operand)) {
if ((count($operand, COUNT_RECURSIVE) - count($operand)) == 1) {
@@ -5142,7 +2288,7 @@ class Calculation
// We only need special validations for the operand if it is a string
// Start by stripping off the quotation marks we use to identify true excel string values internally
if ($operand > '' && $operand[0] == self::FORMULA_STRING_QUOTE) {
- $operand = self::unwrapResult($operand);
+ $operand = StringHelper::convertToString(self::unwrapResult($operand));
}
// If the string is a numeric value, we treat it as a numeric, so no further testing
if (!is_numeric($operand)) {
@@ -5166,26 +2312,29 @@ class Calculation
return true;
}
+ /** @return mixed[] */
private function executeArrayComparison(mixed $operand1, mixed $operand2, string $operation, Stack &$stack, bool $recursingArrays): array
{
$result = [];
- if (!is_array($operand2)) {
+ if (!is_array($operand2) && is_array($operand1)) {
// Operand 1 is an array, Operand 2 is a scalar
foreach ($operand1 as $x => $operandData) {
$this->debugLog->writeDebugLog('Evaluating Comparison %s %s %s', $this->showValue($operandData), $operation, $this->showValue($operand2));
$this->executeBinaryComparisonOperation($operandData, $operand2, $operation, $stack);
+ /** @var array $r */
$r = $stack->pop();
$result[$x] = $r['value'];
}
- } elseif (!is_array($operand1)) {
+ } elseif (is_array($operand2) && !is_array($operand1)) {
// Operand 1 is a scalar, Operand 2 is an array
foreach ($operand2 as $x => $operandData) {
$this->debugLog->writeDebugLog('Evaluating Comparison %s %s %s', $this->showValue($operand1), $operation, $this->showValue($operandData));
$this->executeBinaryComparisonOperation($operand1, $operandData, $operation, $stack);
+ /** @var array $r */
$r = $stack->pop();
$result[$x] = $r['value'];
}
- } else {
+ } elseif (is_array($operand2) && is_array($operand1)) {
// Operand 1 and Operand 2 are both arrays
if (!$recursingArrays) {
self::checkMatrixOperands($operand1, $operand2, 2);
@@ -5193,9 +2342,12 @@ class Calculation
foreach ($operand1 as $x => $operandData) {
$this->debugLog->writeDebugLog('Evaluating Comparison %s %s %s', $this->showValue($operandData), $operation, $this->showValue($operand2[$x]));
$this->executeBinaryComparisonOperation($operandData, $operand2[$x], $operation, $stack, true);
+ /** @var array $r */
$r = $stack->pop();
$result[$x] = $r['value'];
}
+ } else {
+ throw new Exception('Neither operand is an arra');
}
// Log the result details
$this->debugLog->writeDebugLog('Comparison Evaluation Result is %s', $this->showTypeDetails($result));
@@ -5205,6 +2357,7 @@ class Calculation
return $result;
}
+ /** @return bool|mixed[] */
private function executeBinaryComparisonOperation(mixed $operand1, mixed $operand2, string $operation, Stack &$stack, bool $recursingArrays = false): array|bool
{
// If we're dealing with matrix operations, we want a matrix result
@@ -5250,47 +2403,53 @@ class Calculation
$operand2[$key] = Functions::flattenArray($value);
}
}
- [$rows, $columns] = self::checkMatrixOperands($operand1, $operand2, 2);
+ [$rows, $columns] = self::checkMatrixOperands($operand1, $operand2, 3);
for ($row = 0; $row < $rows; ++$row) {
for ($column = 0; $column < $columns; ++$column) {
- if ($operand1[$row][$column] === null) {
+ /** @var mixed[][] $operand1 */
+ if (($operand1[$row][$column] ?? null) === null) {
$operand1[$row][$column] = 0;
} elseif (!self::isNumericOrBool($operand1[$row][$column])) {
$operand1[$row][$column] = self::makeError($operand1[$row][$column]);
continue;
}
- if ($operand2[$row][$column] === null) {
+ /** @var mixed[][] $operand2 */
+ if (($operand2[$row][$column] ?? null) === null) {
$operand2[$row][$column] = 0;
} elseif (!self::isNumericOrBool($operand2[$row][$column])) {
$operand1[$row][$column] = self::makeError($operand2[$row][$column]);
continue;
}
+ /** @var float|int */
+ $operand1Val = $operand1[$row][$column];
+ /** @var float|int */
+ $operand2Val = $operand2[$row][$column];
switch ($operation) {
case '+':
- $operand1[$row][$column] += $operand2[$row][$column];
+ $operand1[$row][$column] = $operand1Val + $operand2Val;
break;
case '-':
- $operand1[$row][$column] -= $operand2[$row][$column];
+ $operand1[$row][$column] = $operand1Val - $operand2Val;
break;
case '*':
- $operand1[$row][$column] *= $operand2[$row][$column];
+ $operand1[$row][$column] = $operand1Val * $operand2Val;
break;
case '/':
- if ($operand2[$row][$column] == 0) {
+ if ($operand2Val == 0) {
$operand1[$row][$column] = ExcelError::DIV0();
} else {
- $operand1[$row][$column] /= $operand2[$row][$column];
+ $operand1[$row][$column] = $operand1Val / $operand2Val;
}
break;
case '^':
- $operand1[$row][$column] = $operand1[$row][$column] ** $operand2[$row][$column];
+ $operand1[$row][$column] = $operand1Val ** $operand2Val;
break;
@@ -5302,6 +2461,8 @@ class Calculation
$result = $operand1;
} else {
// If we're dealing with non-matrix operations, execute the necessary operation
+ /** @var float|int $operand1 */
+ /** @var float|int $operand2 */
switch ($operation) {
// Addition
case '+':
@@ -5359,6 +2520,8 @@ class Calculation
$this->formulaError = $errorMessage;
$this->cyclicReferenceStack->clear();
$suppress = $this->suppressFormulaErrors;
+ $suppressed = $suppress ? ' $suppressed' : '';
+ $this->debugLog->writeDebugLog("Raise Error$suppressed $errorMessage");
if (!$suppress) {
throw new Exception($errorMessage, $code, $exception);
}
@@ -5373,18 +2536,19 @@ class Calculation
* @param ?Worksheet $worksheet Worksheet
* @param bool $resetLog Flag indicating whether calculation log should be reset or not
*
- * @return array Array of values in range if range contains more than one element. Otherwise, a single value is returned.
+ * @return mixed[] Array of values in range if range contains more than one element. Otherwise, a single value is returned.
*/
- public function extractCellRange(string &$range = 'A1', ?Worksheet $worksheet = null, bool $resetLog = true): array
+ public function extractCellRange(string &$range = 'A1', ?Worksheet $worksheet = null, bool $resetLog = true, bool $createCell = false): array
{
// Return value
+ /** @var mixed[][] */
$returnValue = [];
if ($worksheet !== null) {
$worksheetName = $worksheet->getTitle();
if (str_contains($range, '!')) {
- [$worksheetName, $range] = Worksheet::extractSheetTitle($range, true);
+ [$worksheetName, $range] = Worksheet::extractSheetTitle($range, true, true);
$worksheet = ($this->spreadsheet === null) ? null : $this->spreadsheet->getSheetByName($worksheetName);
}
@@ -5396,8 +2560,17 @@ class Calculation
if (!isset($aReferences[1])) {
// Single cell in range
sscanf($aReferences[0], '%[A-Z]%d', $currentCol, $currentRow);
+ if ($createCell && $worksheet !== null && !$worksheet->cellExists($aReferences[0])) {
+ $worksheet->setCellValue($aReferences[0], null);
+ }
if ($worksheet !== null && $worksheet->cellExists($aReferences[0])) {
- $returnValue[$currentRow][$currentCol] = $worksheet->getCell($aReferences[0])->getCalculatedValue($resetLog);
+ $temp = $worksheet->getCell($aReferences[0])->getCalculatedValue($resetLog);
+ if ($this->getInstanceArrayReturnType() === self::RETURN_ARRAY_AS_ARRAY) {
+ while (is_array($temp)) {
+ $temp = array_shift($temp);
+ }
+ }
+ $returnValue[$currentRow][$currentCol] = $temp;
} else {
$returnValue[$currentRow][$currentCol] = null;
}
@@ -5406,8 +2579,17 @@ class Calculation
foreach ($aReferences as $reference) {
// Extract range
sscanf($reference, '%[A-Z]%d', $currentCol, $currentRow);
+ if ($createCell && $worksheet !== null && !$worksheet->cellExists($reference)) {
+ $worksheet->setCellValue($reference, null);
+ }
if ($worksheet !== null && $worksheet->cellExists($reference)) {
- $returnValue[$currentRow][$currentCol] = $worksheet->getCell($reference)->getCalculatedValue($resetLog);
+ $temp = $worksheet->getCell($reference)->getCalculatedValue($resetLog);
+ if ($this->getInstanceArrayReturnType() === self::RETURN_ARRAY_AS_ARRAY) {
+ while (is_array($temp)) {
+ $temp = array_shift($temp);
+ }
+ }
+ $returnValue[$currentRow][$currentCol] = $temp;
} else {
$returnValue[$currentRow][$currentCol] = null;
}
@@ -5425,7 +2607,7 @@ class Calculation
* @param null|Worksheet $worksheet Worksheet
* @param bool $resetLog Flag indicating whether calculation log should be reset or not
*
- * @return array|string Array of values in range if range contains more than one element. Otherwise, a single value is returned.
+ * @return mixed[]|string Array of values in range if range contains more than one element. Otherwise, a single value is returned.
*/
public function extractNamedRange(string &$range = 'A1', ?Worksheet $worksheet = null, bool $resetLog = true): string|array
{
@@ -5434,7 +2616,7 @@ class Calculation
if ($worksheet !== null) {
if (str_contains($range, '!')) {
- [$worksheetName, $range] = Worksheet::extractSheetTitle($range, true);
+ [$worksheetName, $range] = Worksheet::extractSheetTitle($range, true, true);
$worksheet = ($this->spreadsheet === null) ? null : $this->spreadsheet->getSheetByName($worksheetName);
}
@@ -5459,6 +2641,7 @@ class Calculation
if (!isset($aReferences[1])) {
// Single cell (or single column or row) in range
[$currentCol, $currentRow] = Coordinate::coordinateFromString($aReferences[0]);
+ /** @var mixed[][] $returnValue */
if ($worksheet !== null && $worksheet->cellExists($aReferences[0])) {
$returnValue[$currentRow][$currentCol] = $worksheet->getCell($aReferences[0])->getCalculatedValue($resetLog);
} else {
@@ -5489,26 +2672,22 @@ class Calculation
public function isImplemented(string $function): bool
{
$function = strtoupper($function);
- $notImplemented = !isset(self::$phpSpreadsheetFunctions[$function]) || (is_array(self::$phpSpreadsheetFunctions[$function]['functionCall']) && self::$phpSpreadsheetFunctions[$function]['functionCall'][1] === 'DUMMY');
+ $phpSpreadsheetFunctions = &self::getFunctionsAddress();
+ $notImplemented = !isset($phpSpreadsheetFunctions[$function]) || (is_array($phpSpreadsheetFunctions[$function]['functionCall']) && $phpSpreadsheetFunctions[$function]['functionCall'][1] === 'DUMMY');
return !$notImplemented;
}
- /**
- * Get a list of all implemented functions as an array of function objects.
- */
- public static function getFunctions(): array
- {
- return self::$phpSpreadsheetFunctions;
- }
-
/**
* Get a list of implemented Excel function names.
+ *
+ * @return string[]
*/
public function getImplementedFunctionNames(): array
{
$returnValue = [];
- foreach (self::$phpSpreadsheetFunctions as $functionName => $function) {
+ $phpSpreadsheetFunctions = &self::getFunctionsAddress();
+ foreach ($phpSpreadsheetFunctions as $functionName => $function) {
if ($this->isImplemented($functionName)) {
$returnValue[] = $functionName;
}
@@ -5517,6 +2696,13 @@ class Calculation
return $returnValue;
}
+ /**
+ * @param string[] $functionCall
+ * @param mixed[] $args
+ * @param mixed[] $emptyArguments
+ *
+ * @return mixed[]
+ */
private function addDefaultArgumentValues(array $functionCall, array $args, array $emptyArguments): array
{
$reflector = new ReflectionMethod($functionCall[0], $functionCall[1]);
@@ -5567,6 +2753,11 @@ class Calculation
/**
* Add cell reference if needed while making sure that it is the last argument.
+ *
+ * @param mixed[] $args
+ * @param string|string[] $functionCall
+ *
+ * @return mixed[]
*/
private function addCellReference(array $args, bool $passCellReference, array|string $functionCall, ?Cell $cell = null): array
{
@@ -5616,11 +2807,14 @@ class Calculation
$recursiveCalculationCellAddress = $recursiveCalculationCell->getCoordinate();
// Adjust relative references in ranges and formulae so that we execute the calculation for the correct rows and columns
- $definedNameValue = self::$referenceHelper->updateFormulaReferencesAnyWorksheet(
- $definedNameValue,
- Coordinate::columnIndexFromString($cell->getColumn()) - 1,
- $cell->getRow() - 1
- );
+ $definedNameValue = ReferenceHelper::getInstance()
+ ->updateFormulaReferencesAnyWorksheet(
+ $definedNameValue,
+ Coordinate::columnIndexFromString(
+ $cell->getColumn()
+ ) - 1,
+ $cell->getRow() - 1
+ );
$this->debugLog->writeDebugLog('Value adjusted for relative references is %s', $definedNameValue);
@@ -5635,14 +2829,22 @@ class Calculation
$this->debugLog->writeDebugLog('Evaluation Result for Named %s %s is %s', $definedNameType, $namedRange->getName(), $this->showTypeDetails($result));
}
- $stack->push('Defined Name', $result, $namedRange->getName());
+ $y = $namedRange->getWorksheet()?->getTitle();
+ $x = $namedRange->getLocalOnly();
+ if ($x && $y !== null) {
+ $stack->push('Defined Name', $result, "'$y'!" . $namedRange->getName());
+ } else {
+ $stack->push('Defined Name', $result, $namedRange->getName());
+ }
return $result;
}
- public function setSuppressFormulaErrors(bool $suppressFormulaErrors): void
+ public function setSuppressFormulaErrors(bool $suppressFormulaErrors): self
{
$this->suppressFormulaErrors = $suppressFormulaErrors;
+
+ return $this;
}
public function getSuppressFormulaErrors(): bool
@@ -5650,7 +2852,7 @@ class Calculation
return $this->suppressFormulaErrors;
}
- private static function boolToString(mixed $operand1): mixed
+ public static function boolToString(mixed $operand1): mixed
{
if (is_bool($operand1)) {
$operand1 = ($operand1) ? self::$localeBoolean['TRUE'] : self::$localeBoolean['FALSE'];
@@ -5668,6 +2870,26 @@ class Calculation
private static function makeError(mixed $operand = ''): string
{
- return Information\ErrorValue::isError($operand) ? $operand : ExcelError::VALUE();
+ return (is_string($operand) && Information\ErrorValue::isError($operand)) ? $operand : ExcelError::VALUE();
+ }
+
+ private static function swapOperands(Stack $stack, string $opCharacter): bool
+ {
+ $retVal = false;
+ if ($stack->count() > 0) {
+ $o2 = $stack->last();
+ if ($o2) {
+ if (isset(self::CALCULATION_OPERATORS[$o2['value']])) {
+ $retVal = (self::OPERATOR_PRECEDENCE[$opCharacter] ?? 0) <= self::OPERATOR_PRECEDENCE[$o2['value']];
+ }
+ }
+ }
+
+ return $retVal;
+ }
+
+ public function getSpreadsheet(): ?Spreadsheet
+ {
+ return $this->spreadsheet;
}
}
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Category.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Category.php
index b661fafe..38c19b30 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Category.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Category.php
@@ -18,4 +18,5 @@ abstract class Category
const CATEGORY_TEXT_AND_DATA = 'Text and Data';
const CATEGORY_WEB = 'Web';
const CATEGORY_UNCATEGORISED = 'Uncategorised';
+ const CATEGORY_MICROSOFT_INTERNAL = 'MS Internal';
}
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DAverage.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DAverage.php
index e54f1bb3..a85134ee 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DAverage.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DAverage.php
@@ -19,12 +19,12 @@ class DAverage extends DatabaseAbstract
* A database is a list of related data in which rows of related
* information are records, and columns of data are fields. The
* first row of the list contains labels for each column.
- * @param null|array|int|string $field Indicates which column is used in the function. Enter the
+ * @param null|array|int|string $field Indicates which column is used in the function. Enter the
* column label enclosed between double quotation marks, such as
* "Age" or "Yield," or a number (without quotation marks) that
* represents the position of the column within the list: 1 for
* the first column, 2 for the second column, and so on.
- * @param mixed[] $criteria The range of cells that contains the conditions you specify.
+ * @param mixed[][] $criteria The range of cells that contains the conditions you specify.
* You can use any range for the criteria argument, as long as it
* includes at least one column label and at least one cell below
* the column label in which you specify a condition for the
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DCount.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DCount.php
index fff7ab0d..75b21dd2 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DCount.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DCount.php
@@ -20,12 +20,12 @@ class DCount extends DatabaseAbstract
* A database is a list of related data in which rows of related
* information are records, and columns of data are fields. The
* first row of the list contains labels for each column.
- * @param null|array|int|string $field Indicates which column is used in the function. Enter the
+ * @param null|array|int|string $field Indicates which column is used in the function. Enter the
* column label enclosed between double quotation marks, such as
* "Age" or "Yield," or a number (without quotation marks) that
* represents the position of the column within the list: 1 for
* the first column, 2 for the second column, and so on.
- * @param mixed[] $criteria The range of cells that contains the conditions you specify.
+ * @param mixed[][] $criteria The range of cells that contains the conditions you specify.
* You can use any range for the criteria argument, as long as it
* includes at least one column label and at least one cell below
* the column label in which you specify a condition for the
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DCountA.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DCountA.php
index f1a68c1b..23676bff 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DCountA.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DCountA.php
@@ -19,12 +19,12 @@ class DCountA extends DatabaseAbstract
* A database is a list of related data in which rows of related
* information are records, and columns of data are fields. The
* first row of the list contains labels for each column.
- * @param null|array|int|string $field Indicates which column is used in the function. Enter the
+ * @param null|array|int|string $field Indicates which column is used in the function. Enter the
* column label enclosed between double quotation marks, such as
* "Age" or "Yield," or a number (without quotation marks) that
* represents the position of the column within the list: 1 for
* the first column, 2 for the second column, and so on.
- * @param mixed[] $criteria The range of cells that contains the conditions you specify.
+ * @param mixed[][] $criteria The range of cells that contains the conditions you specify.
* You can use any range for the criteria argument, as long as it
* includes at least one column label and at least one cell below
* the column label in which you specify a condition for the
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DGet.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DGet.php
index dd0f0061..719beaa7 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DGet.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DGet.php
@@ -19,12 +19,12 @@ class DGet extends DatabaseAbstract
* A database is a list of related data in which rows of related
* information are records, and columns of data are fields. The
* first row of the list contains labels for each column.
- * @param null|array|int|string $field Indicates which column is used in the function. Enter the
+ * @param null|array|int|string $field Indicates which column is used in the function. Enter the
* column label enclosed between double quotation marks, such as
* "Age" or "Yield," or a number (without quotation marks) that
* represents the position of the column within the list: 1 for
* the first column, 2 for the second column, and so on.
- * @param mixed[] $criteria The range of cells that contains the conditions you specify.
+ * @param mixed[][] $criteria The range of cells that contains the conditions you specify.
* You can use any range for the criteria argument, as long as it
* includes at least one column label and at least one cell below
* the column label in which you specify a condition for the
@@ -42,6 +42,7 @@ class DGet extends DatabaseAbstract
return ExcelError::NAN();
}
+ /** @var array */
$row = array_pop($columnData);
return array_pop($row);
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DMax.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DMax.php
index 23b95a7d..a1942884 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DMax.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DMax.php
@@ -20,12 +20,12 @@ class DMax extends DatabaseAbstract
* A database is a list of related data in which rows of related
* information are records, and columns of data are fields. The
* first row of the list contains labels for each column.
- * @param null|array|int|string $field Indicates which column is used in the function. Enter the
+ * @param null|array|int|string $field Indicates which column is used in the function. Enter the
* column label enclosed between double quotation marks, such as
* "Age" or "Yield," or a number (without quotation marks) that
* represents the position of the column within the list: 1 for
* the first column, 2 for the second column, and so on.
- * @param mixed[] $criteria The range of cells that contains the conditions you specify.
+ * @param mixed[][] $criteria The range of cells that contains the conditions you specify.
* You can use any range for the criteria argument, as long as it
* includes at least one column label and at least one cell below
* the column label in which you specify a condition for the
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DMin.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DMin.php
index 541803dc..f94e09f9 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DMin.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DMin.php
@@ -20,12 +20,12 @@ class DMin extends DatabaseAbstract
* A database is a list of related data in which rows of related
* information are records, and columns of data are fields. The
* first row of the list contains labels for each column.
- * @param null|array|int|string $field Indicates which column is used in the function. Enter the
+ * @param null|array|int|string $field Indicates which column is used in the function. Enter the
* column label enclosed between double quotation marks, such as
* "Age" or "Yield," or a number (without quotation marks) that
* represents the position of the column within the list: 1 for
* the first column, 2 for the second column, and so on.
- * @param mixed[] $criteria The range of cells that contains the conditions you specify.
+ * @param mixed[][] $criteria The range of cells that contains the conditions you specify.
* You can use any range for the criteria argument, as long as it
* includes at least one column label and at least one cell below
* the column label in which you specify a condition for the
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DProduct.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DProduct.php
index b60aa0dc..e7122d63 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DProduct.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DProduct.php
@@ -19,12 +19,12 @@ class DProduct extends DatabaseAbstract
* A database is a list of related data in which rows of related
* information are records, and columns of data are fields. The
* first row of the list contains labels for each column.
- * @param null|array|int|string $field Indicates which column is used in the function. Enter the
+ * @param null|array|int|string $field Indicates which column is used in the function. Enter the
* column label enclosed between double quotation marks, such as
* "Age" or "Yield," or a number (without quotation marks) that
* represents the position of the column within the list: 1 for
* the first column, 2 for the second column, and so on.
- * @param mixed[] $criteria The range of cells that contains the conditions you specify.
+ * @param mixed[][] $criteria The range of cells that contains the conditions you specify.
* You can use any range for the criteria argument, as long as it
* includes at least one column label and at least one cell below
* the column label in which you specify a condition for the
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DStDev.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DStDev.php
index dc354056..a03ee2a0 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DStDev.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DStDev.php
@@ -20,12 +20,12 @@ class DStDev extends DatabaseAbstract
* A database is a list of related data in which rows of related
* information are records, and columns of data are fields. The
* first row of the list contains labels for each column.
- * @param null|array|int|string $field Indicates which column is used in the function. Enter the
+ * @param null|array|int|string $field Indicates which column is used in the function. Enter the
* column label enclosed between double quotation marks, such as
* "Age" or "Yield," or a number (without quotation marks) that
* represents the position of the column within the list: 1 for
* the first column, 2 for the second column, and so on.
- * @param mixed[] $criteria The range of cells that contains the conditions you specify.
+ * @param mixed[][] $criteria The range of cells that contains the conditions you specify.
* You can use any range for the criteria argument, as long as it
* includes at least one column label and at least one cell below
* the column label in which you specify a condition for the
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DStDevP.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DStDevP.php
index a05d5968..54f11424 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DStDevP.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DStDevP.php
@@ -20,12 +20,12 @@ class DStDevP extends DatabaseAbstract
* A database is a list of related data in which rows of related
* information are records, and columns of data are fields. The
* first row of the list contains labels for each column.
- * @param null|array|int|string $field Indicates which column is used in the function. Enter the
+ * @param null|array|int|string $field Indicates which column is used in the function. Enter the
* column label enclosed between double quotation marks, such as
* "Age" or "Yield," or a number (without quotation marks) that
* represents the position of the column within the list: 1 for
* the first column, 2 for the second column, and so on.
- * @param mixed[] $criteria The range of cells that contains the conditions you specify.
+ * @param mixed[][] $criteria The range of cells that contains the conditions you specify.
* You can use any range for the criteria argument, as long as it
* includes at least one column label and at least one cell below
* the column label in which you specify a condition for the
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DSum.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DSum.php
index f9f926b0..7396a0f5 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DSum.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DSum.php
@@ -19,12 +19,12 @@ class DSum extends DatabaseAbstract
* A database is a list of related data in which rows of related
* information are records, and columns of data are fields. The
* first row of the list contains labels for each column.
- * @param null|array|int|string $field Indicates which column is used in the function. Enter the
+ * @param null|array|int|string $field Indicates which column is used in the function. Enter the
* column label enclosed between double quotation marks, such as
* "Age" or "Yield," or a number (without quotation marks) that
* represents the position of the column within the list: 1 for
* the first column, 2 for the second column, and so on.
- * @param mixed[] $criteria The range of cells that contains the conditions you specify.
+ * @param mixed[][] $criteria The range of cells that contains the conditions you specify.
* You can use any range for the criteria argument, as long as it
* includes at least one column label and at least one cell below
* the column label in which you specify a condition for the
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DVar.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DVar.php
index 33b5b56e..e056e9e5 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DVar.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DVar.php
@@ -20,12 +20,12 @@ class DVar extends DatabaseAbstract
* A database is a list of related data in which rows of related
* information are records, and columns of data are fields. The
* first row of the list contains labels for each column.
- * @param null|array|int|string $field Indicates which column is used in the function. Enter the
+ * @param null|array|int|string $field Indicates which column is used in the function. Enter the
* column label enclosed between double quotation marks, such as
* "Age" or "Yield," or a number (without quotation marks) that
* represents the position of the column within the list: 1 for
* the first column, 2 for the second column, and so on.
- * @param mixed[] $criteria The range of cells that contains the conditions you specify.
+ * @param mixed[][] $criteria The range of cells that contains the conditions you specify.
* You can use any range for the criteria argument, as long as it
* includes at least one column label and at least one cell below
* the column label in which you specify a condition for the
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DVarP.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DVarP.php
index 942a4a1f..3bd571fa 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DVarP.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DVarP.php
@@ -20,12 +20,12 @@ class DVarP extends DatabaseAbstract
* A database is a list of related data in which rows of related
* information are records, and columns of data are fields. The
* first row of the list contains labels for each column.
- * @param null|array|int|string $field Indicates which column is used in the function. Enter the
+ * @param null|array|int|string $field Indicates which column is used in the function. Enter the
* column label enclosed between double quotation marks, such as
* "Age" or "Yield," or a number (without quotation marks) that
* represents the position of the column within the list: 1 for
* the first column, 2 for the second column, and so on.
- * @param mixed[] $criteria The range of cells that contains the conditions you specify.
+ * @param mixed[][] $criteria The range of cells that contains the conditions you specify.
* You can use any range for the criteria argument, as long as it
* includes at least one column label and at least one cell below
* the column label in which you specify a condition for the
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DatabaseAbstract.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DatabaseAbstract.php
index 7d9885ee..5c72a199 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DatabaseAbstract.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Database/DatabaseAbstract.php
@@ -5,9 +5,26 @@ namespace PhpOffice\PhpSpreadsheet\Calculation\Database;
use PhpOffice\PhpSpreadsheet\Calculation\Calculation;
use PhpOffice\PhpSpreadsheet\Calculation\Functions;
use PhpOffice\PhpSpreadsheet\Calculation\Internal\WildcardMatch;
+use PhpOffice\PhpSpreadsheet\Shared\StringHelper;
abstract class DatabaseAbstract
{
+ /**
+ * @param mixed[] $database The range of cells that makes up the list or database.
+ * A database is a list of related data in which rows of related
+ * information are records, and columns of data are fields. The
+ * first row of the list contains labels for each column.
+ * @param null|array|int|string $field Indicates which column is used in the function. Enter the
+ * column label enclosed between double quotation marks, such as
+ * "Age" or "Yield," or a number (without quotation marks) that
+ * represents the position of the column within the list: 1 for
+ * the first column, 2 for the second column, and so on.
+ * @param mixed[] $criteria The range of cells that contains the conditions you specify.
+ * You can use any range for the criteria argument, as long as it
+ * includes at least one column label and at least one cell below
+ * the column label in which you specify a condition for the
+ * column.
+ */
abstract public static function evaluate(array $database, array|null|int|string $field, array $criteria): null|float|int|string;
/**
@@ -27,12 +44,16 @@ abstract class DatabaseAbstract
*/
protected static function fieldExtract(array $database, mixed $field): ?int
{
- $field = strtoupper(Functions::flattenSingleValue($field) ?? '');
+ /** @var ?string */
+ $single = Functions::flattenSingleValue($field);
+ $field = strtoupper($single ?? '');
if ($field === '') {
return null;
}
- $fieldNames = array_map('strtoupper', array_shift($database));
+ /** @var callable */
+ $callable = 'strtoupper';
+ $fieldNames = array_map($callable, array_shift($database)); //* @phpstan-ignore-line
if (is_numeric($field)) {
$field = (int) $field - 1;
if ($field < 0 || $field >= count($fieldNames)) {
@@ -56,7 +77,7 @@ abstract class DatabaseAbstract
* A database is a list of related data in which rows of related
* information are records, and columns of data are fields. The
* first row of the list contains labels for each column.
- * @param mixed[] $criteria The range of cells that contains the conditions you specify.
+ * @param mixed[][] $criteria The range of cells that contains the conditions you specify.
* You can use any range for the criteria argument, as long as it
* includes at least one column label and at least one cell below
* the column label in which you specify a condition for the
@@ -66,16 +87,25 @@ abstract class DatabaseAbstract
*/
protected static function filter(array $database, array $criteria): array
{
+ /** @var mixed[] */
$fieldNames = array_shift($database);
$criteriaNames = array_shift($criteria);
// Convert the criteria into a set of AND/OR conditions with [:placeholders]
+ /** @var string[] $criteriaNames */
$query = self::buildQuery($criteriaNames, $criteria);
// Loop through each row of the database
+ /** @var mixed[][] $criteriaNames */
return self::executeQuery($database, $query, $criteriaNames, $fieldNames);
}
+ /**
+ * @param mixed[] $database The range of cells that makes up the list or database
+ * @param mixed[][] $criteria
+ *
+ * @return mixed[]
+ */
protected static function getFilteredColumn(array $database, ?int $field, array $criteria): array
{
// reduce the database to a set of rows that match all the criteria
@@ -84,6 +114,7 @@ abstract class DatabaseAbstract
// extract an array of values for the requested column
$columnData = [];
+ /** @var mixed[] $row */
foreach ($database as $rowKey => $row) {
$keys = array_keys($row);
$key = $keys[$field] ?? null;
@@ -94,6 +125,10 @@ abstract class DatabaseAbstract
return $columnData;
}
+ /**
+ * @param string[] $criteriaNames
+ * @param mixed[][] $criteria
+ */
private static function buildQuery(array $criteriaNames, array $criteria): string
{
$baseQuery = [];
@@ -108,7 +143,7 @@ abstract class DatabaseAbstract
}
$rowQuery = array_map(
- fn ($rowValue): string => (count($rowValue) > 1) ? 'AND(' . implode(',', $rowValue) . ')' : ($rowValue[0] ?? ''),
+ fn ($rowValue): string => (count($rowValue) > 1) ? 'AND(' . implode(',', $rowValue) . ')' : ($rowValue[0] ?? ''), // @phpstan-ignore-line
$baseQuery
);
@@ -135,12 +170,21 @@ abstract class DatabaseAbstract
return $condition;
}
+ /**
+ * @param mixed[] $database
+ * @param mixed[][] $criteria
+ * @param array $fields
+ *
+ * @return mixed[]
+ */
private static function executeQuery(array $database, string $query, array $criteria, array $fields): array
{
foreach ($database as $dataRow => $dataValues) {
// Substitute actual values from the database row for our [:placeholders]
$conditions = $query;
foreach ($criteria as $criterion) {
+ /** @var string $criterion */
+ /** @var mixed[] $dataValues */
$conditions = self::processCondition($criterion, $fields, $dataValues, $conditions);
}
@@ -156,6 +200,10 @@ abstract class DatabaseAbstract
return $database;
}
+ /**
+ * @param array $fields
+ * @param array $dataValues
+ */
private static function processCondition(string $criterion, array $fields, array $dataValues, string $conditions): string
{
$key = array_search($criterion, $fields, true);
@@ -169,7 +217,10 @@ abstract class DatabaseAbstract
if (is_string($dataValue) && str_contains($dataValue, '"')) {
$dataValue = str_replace('"', '""', $dataValue);
}
- $dataValue = (is_string($dataValue)) ? Calculation::wrapResult(strtoupper($dataValue)) : $dataValue;
+ if (is_string($dataValue)) {
+ $dataValue = Calculation::wrapResult(strtoupper($dataValue));
+ }
+ $dataValue = StringHelper::convertToString($dataValue);
}
return str_replace('[:' . $criterion . ']', $dataValue, $conditions);
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Date.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Date.php
index e0e4b25f..9a6faf32 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Date.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Date.php
@@ -28,7 +28,7 @@ class Date
* A Month name or abbreviation (English only at this point) such as 'January' or 'Jan' will still be accepted,
* as will a day value with a suffix (e.g. '21st' rather than simply 21); again only English language.
*
- * @param array|float|int|string $year The value of the year argument can include one to four digits.
+ * @param array|float|int|string $year The value of the year argument can include one to four digits.
* Excel interprets the year argument according to the configured
* date system: 1900 or 1904.
* If year is between 0 (zero) and 1899 (inclusive), Excel adds that
@@ -39,7 +39,7 @@ class Date
* 2008.
* If year is less than 0 or is 10000 or greater, Excel returns the
* #NUM! error value.
- * @param array|float|int|string $month A positive or negative integer representing the month of the year
+ * @param array|float|int|string $month A positive or negative integer representing the month of the year
* from 1 to 12 (January to December).
* If month is greater than 12, month adds that number of months to
* the first month in the year specified. For example, DATE(2008,14,2)
@@ -48,7 +48,7 @@ class Date
* number of months, plus 1, from the first month in the year
* specified. For example, DATE(2008,-3,2) returns the serial number
* representing September 2, 2007.
- * @param array|float|int|string $day A positive or negative integer representing the day of the month
+ * @param array|float|int|string $day A positive or negative integer representing the day of the month
* from 1 to 31.
* If day is greater than the number of days in the month specified,
* day adds that number of days to the first day in the month. For
@@ -59,12 +59,12 @@ class Date
* example, DATE(2008,1,-15) returns the serial number representing
* December 16, 2007.
*
- * @return array|DateTime|float|int|string Excel date/time serial value, PHP date/time serial value or PHP date/time object,
+ * @return array|DateTime|float|int|string Excel date/time serial value, PHP date/time serial value or PHP date/time object,
* depending on the value of the ReturnDateType flag
* If an array of numbers is passed as the argument, then the returned result will also be an array
* with the same dimensions
*/
- public static function fromYMD(array|float|int|string $year, array|float|int|string $month, array|float|int|string $day): float|int|DateTime|string|array
+ public static function fromYMD(array|float|int|string $year, null|array|bool|float|int|string $month, array|float|int|string $day): float|int|DateTime|string|array
{
if (is_array($year) || is_array($month) || is_array($day)) {
return self::evaluateArrayArguments([self::class, __FUNCTION__], $year, $month, $day);
@@ -92,7 +92,11 @@ class Date
*/
private static function getYear(mixed $year, int $baseYear): int
{
- $year = ($year !== null) ? StringHelper::testStringAsNumeric((string) $year) : 0;
+ if ($year === null) {
+ $year = 0;
+ } elseif (is_scalar($year)) {
+ $year = StringHelper::testStringAsNumeric((string) $year);
+ }
if (!is_numeric($year)) {
throw new Exception(ExcelError::VALUE());
}
@@ -117,11 +121,15 @@ class Date
*/
private static function getMonth(mixed $month): int
{
- if (($month !== null) && (!is_numeric($month))) {
- $month = SharedDateHelper::monthStringToNumber($month);
+ if (is_string($month)) {
+ if (!is_numeric($month)) {
+ $month = SharedDateHelper::monthStringToNumber($month);
+ }
+ } elseif ($month === null) {
+ $month = 0;
+ } elseif (is_bool($month)) {
+ $month = (int) $month;
}
-
- $month = ($month !== null) ? StringHelper::testStringAsNumeric((string) $month) : 0;
if (!is_numeric($month)) {
throw new Exception(ExcelError::VALUE());
}
@@ -134,11 +142,15 @@ class Date
*/
private static function getDay(mixed $day): int
{
- if (($day !== null) && (!is_numeric($day))) {
+ if (is_string($day) && !is_numeric($day)) {
$day = SharedDateHelper::dayStringToNumber($day);
}
- $day = ($day !== null) ? StringHelper::testStringAsNumeric((string) $day) : 0;
+ if ($day === null) {
+ $day = 0;
+ } elseif (is_scalar($day)) {
+ $day = StringHelper::testStringAsNumeric((string) $day);
+ }
if (!is_numeric($day)) {
throw new Exception(ExcelError::VALUE());
}
@@ -151,11 +163,11 @@ class Date
if ($month < 1) {
// Handle year/month adjustment if month < 1
--$month;
- $year += ceil($month / 12) - 1;
+ $year += (int) (ceil($month / 12) - 1);
$month = 13 - abs($month % 12);
} elseif ($month > 12) {
// Handle year/month adjustment if month > 12
- $year += floor($month / 12);
+ $year += intdiv($month, 12);
$month = ($month % 12);
}
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/DateParts.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/DateParts.php
index 60e4de19..fc239041 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/DateParts.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/DateParts.php
@@ -24,7 +24,7 @@ class DateParts
* PHP DateTime object, or a standard date string
* Or can be an array of date values
*
- * @return array|int|string Day of the month
+ * @return array|int|string Day of the month
* If an array of numbers is passed as the argument, then the returned result will also be an array
* with the same dimensions
*/
@@ -65,7 +65,7 @@ class DateParts
* PHP DateTime object, or a standard date string
* Or can be an array of date values
*
- * @return array|int|string Month of the year
+ * @return array|int|string Month of the year
* If an array of numbers is passed as the argument, then the returned result will also be an array
* with the same dimensions
*/
@@ -104,7 +104,7 @@ class DateParts
* PHP DateTime object, or a standard date string
* Or can be an array of date values
*
- * @return array|int|string Year
+ * @return array|int|string Year
* If an array of numbers is passed as the argument, then the returned result will also be an array
* with the same dimensions
*/
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/DateValue.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/DateValue.php
index 8c5fa71c..d29ed6ba 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/DateValue.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/DateValue.php
@@ -25,7 +25,7 @@ class DateValue
* Excel Function:
* DATEVALUE(dateValue)
*
- * @param null|array|bool|float|int|string $dateValue Text that represents a date in a Microsoft Excel date format.
+ * @param null|array|bool|float|int|string $dateValue Text that represents a date in a Microsoft Excel date format.
* For example, "1/30/2008" or "30-Jan-2008" are text strings within
* quotation marks that represent dates. Using the default date
* system in Excel for Windows, date_text must represent a date from
@@ -35,7 +35,7 @@ class DateValue
* #VALUE! error value if date_text is out of this range.
* Or can be an array of date values
*
- * @return array|DateTime|float|int|string Excel date/time serial value, PHP date/time serial value or PHP date/time object,
+ * @return array|DateTime|float|int|string Excel date/time serial value, PHP date/time serial value or PHP date/time object,
* depending on the value of the ReturnDateType flag
* If an array of numbers is passed as the argument, then the returned result will also be an array
* with the same dimensions
@@ -47,7 +47,7 @@ class DateValue
}
// try to parse as date iff there is at least one digit
- if (is_string($dateValue) && preg_match('/\\d/', $dateValue) !== 1) {
+ if (is_string($dateValue) && preg_match('/\d/', $dateValue) !== 1) {
return ExcelError::VALUE();
}
@@ -86,6 +86,7 @@ class DateValue
return self::finalResults($PHPDateArray, $dti, $baseYear);
}
+ /** @param mixed[] $t1 */
private static function t1ToString(array $t1, DateTimeImmutable $dti, bool $yearFound): string
{
if (count($t1) == 2) {
@@ -108,6 +109,8 @@ class DateValue
/**
* Parse date.
+ *
+ * @return mixed[]
*/
private static function setUpArray(string $dateValue, DateTimeImmutable $dti): array
{
@@ -132,6 +135,8 @@ class DateValue
/**
* Final results.
*
+ * @param mixed[] $PHPDateArray
+ *
* @return DateTime|float|int|string Excel date/time serial value, PHP date/time serial value or PHP date/time object,
* depending on the value of the ReturnDateType flag
*/
@@ -139,6 +144,7 @@ class DateValue
{
$retValue = ExcelError::Value();
if (Helpers::dateParseSucceeded($PHPDateArray)) {
+ /** @var array{year: int, month: int, day: int, hour: int, minute: int, second: int} $PHPDateArray */
// Execute function
Helpers::replaceIfEmpty($PHPDateArray['year'], $dti->format('Y'));
if ($PHPDateArray['year'] < $baseYear) {
@@ -146,12 +152,13 @@ class DateValue
}
Helpers::replaceIfEmpty($PHPDateArray['month'], $dti->format('m'));
Helpers::replaceIfEmpty($PHPDateArray['day'], $dti->format('d'));
+ /** @var array{year: int, month: int, day: int, hour: int, minute: int, second: int} $PHPDateArray */
$PHPDateArray['hour'] = 0;
$PHPDateArray['minute'] = 0;
$PHPDateArray['second'] = 0;
- $month = (int) $PHPDateArray['month'];
- $day = (int) $PHPDateArray['day'];
- $year = (int) $PHPDateArray['year'];
+ $month = self::getInt($PHPDateArray, 'month');
+ $day = self::getInt($PHPDateArray, 'day');
+ $year = self::getInt($PHPDateArray, 'year');
if (!checkdate($month, $day, $year)) {
return ($year === 1900 && $month === 2 && $day === 29) ? Helpers::returnIn3FormatsFloat(60.0) : ExcelError::VALUE();
}
@@ -160,4 +167,10 @@ class DateValue
return $retValue;
}
+
+ /** @param mixed[] $array */
+ private static function getInt(array $array, string $index): int
+ {
+ return (array_key_exists($index, $array) && is_numeric($array[$index])) ? (int) $array[$index] : 0;
+ }
}
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Days.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Days.php
index 6c6fd3d7..a3e50588 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Days.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Days.php
@@ -20,14 +20,14 @@ class Days
* Excel Function:
* DAYS(endDate, startDate)
*
- * @param array|DateTimeInterface|float|int|string $endDate Excel date serial value (float),
+ * @param array|DateTimeInterface|float|int|string $endDate Excel date serial value (float),
* PHP date timestamp (integer), PHP DateTime object, or a standard date string
* Or can be an array of date values
- * @param array|DateTimeInterface|float|int|string $startDate Excel date serial value (float),
+ * @param array|DateTimeInterface|float|int|string $startDate Excel date serial value (float),
* PHP date timestamp (integer), PHP DateTime object, or a standard date string
* Or can be an array of date values
*
- * @return array|int|string Number of days between start date and end date or an error
+ * @return array|int|string Number of days between start date and end date or an error
* If an array of values is passed for the $startDate or $endDays,arguments, then the returned result
* will also be an array with matching dimensions
*/
@@ -50,7 +50,7 @@ class Days
$days = ExcelError::VALUE();
$diff = $PHPStartDateObject->diff($PHPEndDateObject);
- if ($diff !== false && !is_bool($diff->days)) {
+ if (!is_bool($diff->days)) {
$days = $diff->days;
if ($diff->invert) {
$days = -$days;
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Days360.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Days360.php
index c7e03fc0..2ccadf53 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Days360.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Days360.php
@@ -40,7 +40,7 @@ class Days360
* same month.
* Or can be an array of methods
*
- * @return array|int|string Number of days between start date and end date
+ * @return array|int|string Number of days between start date and end date
* If an array of values is passed for the $startDate or $endDays,arguments, then the returned result
* will also be an array with matching dimensions
*/
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Difference.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Difference.php
index 199d5d85..748586de 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Difference.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Difference.php
@@ -22,9 +22,9 @@ class Difference
* @param mixed $endDate Excel date serial value, PHP date/time stamp, PHP DateTime object
* or a standard date string
* Or can be an array of date values
- * @param array|string $unit Or can be an array of unit values
+ * @param array|string $unit Or can be an array of unit values
*
- * @return array|int|string Interval between the dates
+ * @return array|int|string Interval between the dates
* If an array of values is passed for the $startDate or $endDays,arguments, then the returned result
* will also be an array with matching dimensions
*/
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Helpers.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Helpers.php
index 1e9af6cb..ee1d4866 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Helpers.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Helpers.php
@@ -44,7 +44,9 @@ class Helpers
if (!is_numeric($dateValue)) {
$saveReturnDateType = Functions::getReturnDateType();
Functions::setReturnDateType(Functions::RETURNDATE_EXCEL);
- $dateValue = DateValue::fromString($dateValue);
+ if (is_string($dateValue)) {
+ $dateValue = DateValue::fromString($dateValue);
+ }
Functions::setReturnDateType($saveReturnDateType);
if (!is_numeric($dateValue)) {
throw new Exception(ExcelError::VALUE());
@@ -75,8 +77,10 @@ class Helpers
/**
* Adjust date by given months.
+ *
+ * @param float|int $dateValue date to be adjusted
*/
- public static function adjustDateByMonths(mixed $dateValue = 0, float $adjustmentMonths = 0): DateTime
+ public static function adjustDateByMonths($dateValue = 0, float $adjustmentMonths = 0): DateTime
{
// Execute function
$PHPDateObject = SharedDateHelper::excelToDateTimeObject($dateValue);
@@ -119,7 +123,7 @@ class Helpers
if (!is_numeric($testVal1) || $testVal1 < 31) {
if (!is_numeric($testVal2) || $testVal2 < 12) {
if (is_numeric($testVal3) && $testVal3 < 12) {
- $testVal3 += 2000;
+ $testVal3 = (string) ($testVal3 + 2000);
}
}
}
@@ -127,6 +131,8 @@ class Helpers
/**
* Return result in one of three formats.
+ *
+ * @param array{year: int, month: int, day: int, hour: int, minute: int, second: int} $dateArray
*/
public static function returnIn3FormatsArray(array $dateArray, bool $noFrac = false): DateTime|float|int
{
@@ -264,11 +270,16 @@ class Helpers
}
}
+ /** @return array{year: int, month: int, day: int, hour: int, minute: int, second: int} */
public static function dateParse(string $string): array
{
- return self::forceArray(date_parse($string));
+ /** @var array{year: int, month: int, day: int, hour: int, minute: int, second: int} */
+ $temp = self::forceArray(date_parse($string));
+
+ return $temp;
}
+ /** @param mixed[] $dateArray */
public static function dateParseSucceeded(array $dateArray): bool
{
return $dateArray['error_count'] === 0;
@@ -278,10 +289,19 @@ class Helpers
* Despite documentation, date_parse probably never returns false.
* Just in case, this routine helps guarantee it.
*
- * @param array|false $dateArray
+ * @param array|false $dateArray
+ *
+ * @return mixed[]
*/
private static function forceArray(array|bool $dateArray): array
{
return is_array($dateArray) ? $dateArray : ['error_count' => 1];
}
+
+ public static function floatOrInt(mixed $value): float|int
+ {
+ $result = Functions::scalar($value);
+
+ return is_numeric($result) ? ($result + 0) : 0;
+ }
}
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Month.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Month.php
index a90c0517..74fd50e3 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Month.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Month.php
@@ -24,12 +24,12 @@ class Month
* @param mixed $dateValue Excel date serial value (float), PHP date timestamp (integer),
* PHP DateTime object, or a standard date string
* Or can be an array of date values
- * @param array|int $adjustmentMonths The number of months before or after start_date.
+ * @param array|int $adjustmentMonths The number of months before or after start_date.
* A positive value for months yields a future date;
* a negative value yields a past date.
* Or can be an array of adjustment values
*
- * @return array|DateTime|float|int|string Excel date/time serial value, PHP date/time serial value or PHP date/time object,
+ * @return array|DateTime|float|int|string Excel date/time serial value, PHP date/time serial value or PHP date/time object,
* depending on the value of the ReturnDateType flag
* If an array of values is passed as the argument, then the returned result will also be an array
* with the same dimensions
@@ -68,12 +68,12 @@ class Month
* @param mixed $dateValue Excel date serial value (float), PHP date timestamp (integer),
* PHP DateTime object, or a standard date string
* Or can be an array of date values
- * @param array|int $adjustmentMonths The number of months before or after start_date.
+ * @param array|int $adjustmentMonths The number of months before or after start_date.
* A positive value for months yields a future date;
* a negative value yields a past date.
* Or can be an array of adjustment values
*
- * @return array|DateTime|float|int|string Excel date/time serial value, PHP date/time serial value or PHP date/time object,
+ * @return array|DateTime|float|int|string Excel date/time serial value, PHP date/time serial value or PHP date/time object,
* depending on the value of the ReturnDateType flag
* If an array of values is passed as the argument, then the returned result will also be an array
* with the same dimensions
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/NetworkDays.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/NetworkDays.php
index 503e30e8..2f1d96bd 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/NetworkDays.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/NetworkDays.php
@@ -29,7 +29,7 @@ class NetworkDays
* Or can be an array of date values
* @param mixed $dateArgs An array of dates (such as holidays) to exclude from the calculation
*
- * @return array|int|string Interval between the dates
+ * @return array|int|string Interval between the dates
* If an array of values is passed for the $startDate or $endDate arguments, then the returned result
* will also be an array with matching dimensions
*/
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Time.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Time.php
index 3f8f324c..65c4d790 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Time.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Time.php
@@ -24,21 +24,21 @@ class Time
* Excel Function:
* TIME(hour,minute,second)
*
- * @param null|array|bool|float|int|string $hour A number from 0 (zero) to 32767 representing the hour.
+ * @param null|array|bool|float|int|string $hour A number from 0 (zero) to 32767 representing the hour.
* Any value greater than 23 will be divided by 24 and the remainder
* will be treated as the hour value. For example, TIME(27,0,0) =
* TIME(3,0,0) = .125 or 3:00 AM.
- * @param null|array|bool|float|int|string $minute A number from 0 to 32767 representing the minute.
+ * @param null|array|bool|float|int|string $minute A number from 0 to 32767 representing the minute.
* Any value greater than 59 will be converted to hours and minutes.
* For example, TIME(0,750,0) = TIME(12,30,0) = .520833 or 12:30 PM.
- * @param null|array|bool|float|int|string $second A number from 0 to 32767 representing the second.
+ * @param null|array|bool|float|int|string $second A number from 0 to 32767 representing the second.
* Any value greater than 59 will be converted to hours, minutes,
* and seconds. For example, TIME(0,0,2000) = TIME(0,33,22) = .023148
* or 12:33:20 AM
* If an array of numbers is passed as the argument, then the returned result will also be an array
* with the same dimensions
*
- * @return array|DateTime|float|int|string Excel date/time serial value, PHP date/time serial value or PHP date/time object,
+ * @return array|DateTime|float|int|string Excel date/time serial value, PHP date/time serial value or PHP date/time object,
* depending on the value of the ReturnDateType flag
* If an array of numbers is passed as the argument, then the returned result will also be an array
* with the same dimensions
@@ -87,13 +87,13 @@ class Time
private static function adjustSecond(int &$second, int &$minute): void
{
if ($second < 0) {
- $minute += floor($second / 60);
+ $minute += (int) floor($second / 60);
$second = 60 - abs($second % 60);
if ($second == 60) {
$second = 0;
}
} elseif ($second >= 60) {
- $minute += floor($second / 60);
+ $minute += intdiv($second, 60);
$second = $second % 60;
}
}
@@ -101,13 +101,13 @@ class Time
private static function adjustMinute(int &$minute, int &$hour): void
{
if ($minute < 0) {
- $hour += floor($minute / 60);
+ $hour += (int) floor($minute / 60);
$minute = 60 - abs($minute % 60);
if ($minute == 60) {
$minute = 0;
}
} elseif ($minute >= 60) {
- $hour += floor($minute / 60);
+ $hour += intdiv($minute, 60);
$minute = $minute % 60;
}
}
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/TimeParts.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/TimeParts.php
index de522692..a7f415d3 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/TimeParts.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/TimeParts.php
@@ -23,7 +23,7 @@ class TimeParts
* PHP DateTime object, or a standard time string
* Or can be an array of date/time values
*
- * @return array|int|string Hour
+ * @return array|int|string Hour
* If an array of numbers is passed as the argument, then the returned result will also be an array
* with the same dimensions
*/
@@ -35,7 +35,7 @@ class TimeParts
try {
Helpers::nullFalseTrueToNumber($timeValue);
- if (!is_numeric($timeValue)) {
+ if (is_string($timeValue) && !is_numeric($timeValue)) {
$timeValue = Helpers::getTimeValue($timeValue);
}
Helpers::validateNotNegative($timeValue);
@@ -64,7 +64,7 @@ class TimeParts
* PHP DateTime object, or a standard time string
* Or can be an array of date/time values
*
- * @return array|int|string Minute
+ * @return array|int|string Minute
* If an array of numbers is passed as the argument, then the returned result will also be an array
* with the same dimensions
*/
@@ -76,7 +76,7 @@ class TimeParts
try {
Helpers::nullFalseTrueToNumber($timeValue);
- if (!is_numeric($timeValue)) {
+ if (is_string($timeValue) && !is_numeric($timeValue)) {
$timeValue = Helpers::getTimeValue($timeValue);
}
Helpers::validateNotNegative($timeValue);
@@ -105,7 +105,7 @@ class TimeParts
* PHP DateTime object, or a standard time string
* Or can be an array of date/time values
*
- * @return array|int|string Second
+ * @return array|int|string Second
* If an array of numbers is passed as the argument, then the returned result will also be an array
* with the same dimensions
*/
@@ -117,7 +117,7 @@ class TimeParts
try {
Helpers::nullFalseTrueToNumber($timeValue);
- if (!is_numeric($timeValue)) {
+ if (is_string($timeValue) && !is_numeric($timeValue)) {
$timeValue = Helpers::getTimeValue($timeValue);
}
Helpers::validateNotNegative($timeValue);
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/TimeValue.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/TimeValue.php
index d8c53b47..fc1f72c4 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/TimeValue.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/TimeValue.php
@@ -2,6 +2,7 @@
namespace PhpOffice\PhpSpreadsheet\Calculation\DateTimeExcel;
+use Composer\Pcre\Preg;
use Datetime;
use PhpOffice\PhpSpreadsheet\Calculation\ArrayEnabled;
use PhpOffice\PhpSpreadsheet\Calculation\Functions;
@@ -12,6 +13,19 @@ class TimeValue
{
use ArrayEnabled;
+ private const EXTRACT_TIME = '/\b'
+ . '(\d+)' // match[1] - hour
+ . '(:' // start of match[2] (rest of string) - colon
+ . '(\d+' // start of match[3] - minute
+ . '(:\d+' // start of match[4] - colon and seconds
+ . '([.]\d+)?' // match[5] - optional decimal point followed by fractional seconds
+ . ')?' // end of match[4], which is optional
+ . ')' // end of match 3
+ // Excel does not require 'm' to trail 'a' or 'p'; Php does
+ . '(\s*(a|p))?' // match[6] optional whitespace followed by optional match[7] a or p
+ . ')' // end of match[2]
+ . '/i';
+
/**
* TIMEVALUE.
*
@@ -25,13 +39,13 @@ class TimeValue
* Excel Function:
* TIMEVALUE(timeValue)
*
- * @param null|array|bool|float|int|string $timeValue A text string that represents a time in any one of the Microsoft
+ * @param null|array|bool|float|int|string $timeValue A text string that represents a time in any one of the Microsoft
* Excel time formats; for example, "6:45 PM" and "18:45" text strings
* within quotation marks that represent time.
* Date information in time_text is ignored.
* Or can be an array of date/time values
*
- * @return array|Datetime|float|int|string Excel date/time serial value, PHP date/time serial value or PHP date/time object,
+ * @return array|Datetime|float|int|string Excel date/time serial value, PHP date/time serial value or PHP date/time object,
* depending on the value of the ReturnDateType flag
* If an array of numbers is passed as the argument, then the returned result will also be an array
* with the same dimensions
@@ -43,17 +57,20 @@ class TimeValue
}
// try to parse as time iff there is at least one digit
- if (is_string($timeValue) && preg_match('/\\d/', $timeValue) !== 1) {
+ if (is_string($timeValue) && !Preg::isMatch('/\d/', $timeValue)) {
return ExcelError::VALUE();
}
$timeValue = trim((string) $timeValue, '"');
- $timeValue = str_replace(['/', '.'], '-', $timeValue);
-
- $arraySplit = preg_split('/[\/:\-\s]/', $timeValue) ?: [];
- if ((count($arraySplit) == 2 || count($arraySplit) == 3) && $arraySplit[0] > 24) {
- $arraySplit[0] = ((int) $arraySplit[0] % 24);
- $timeValue = implode(':', $arraySplit);
+ if (Preg::isMatch(self::EXTRACT_TIME, $timeValue, $matches)) {
+ if (empty($matches[6])) { // am/pm
+ $hour = (int) $matches[0];
+ $timeValue = ($hour % 24) . $matches[2];
+ } elseif ($matches[6] === $matches[7]) { // Excel wants space before am/pm
+ return ExcelError::VALUE();
+ } else {
+ $timeValue = $matches[0] . 'm';
+ }
}
$PHPDateArray = Helpers::dateParse($timeValue);
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Week.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Week.php
index e620b4ca..080a57b0 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Week.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/Week.php
@@ -28,7 +28,7 @@ class Week
* @param mixed $dateValue Excel date serial value (float), PHP date timestamp (integer),
* PHP DateTime object, or a standard date string
* Or can be an array of date values
- * @param array|int $method Week begins on Sunday or Monday
+ * @param array|int $method Week begins on Sunday or Monday
* 1 or omitted Week begins on Sunday.
* 2 Week begins on Monday.
* 11 Week begins on Monday.
@@ -41,7 +41,7 @@ class Week
* 21 ISO (Jan. 4 is week 1, begins on Monday).
* Or can be an array of methods
*
- * @return array|int|string Week Number
+ * @return array|int|string Week Number
* If an array of values is passed as the argument, then the returned result will also be an array
* with the same dimensions
*/
@@ -101,7 +101,7 @@ class Week
* PHP DateTime object, or a standard date string
* Or can be an array of date values
*
- * @return array|int|string Week Number
+ * @return array|int|string Week Number
* If an array of numbers is passed as the argument, then the returned result will also be an array
* with the same dimensions
*/
@@ -137,7 +137,7 @@ class Week
* Excel Function:
* WEEKDAY(dateValue[,style])
*
- * @param null|array|bool|float|int|string $dateValue Excel date serial value (float), PHP date timestamp (integer),
+ * @param null|array|bool|float|int|string $dateValue Excel date serial value (float), PHP date timestamp (integer),
* PHP DateTime object, or a standard date string
* Or can be an array of date values
* @param mixed $style A number that determines the type of return value
@@ -146,7 +146,7 @@ class Week
* 3 Numbers 0 (Monday) through 6 (Sunday).
* Or can be an array of styles
*
- * @return array|int|string Day of the week value
+ * @return array|int|string Day of the week value
* If an array of values is passed as the argument, then the returned result will also be an array
* with the same dimensions
*/
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/WorkDay.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/WorkDay.php
index 4e4ed3c8..84658647 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/WorkDay.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/WorkDay.php
@@ -22,16 +22,16 @@ class WorkDay
* Excel Function:
* WORKDAY(startDate,endDays[,holidays[,holiday[,...]]])
*
- * @param array|mixed $startDate Excel date serial value (float), PHP date timestamp (integer),
+ * @param array|mixed $startDate Excel date serial value (float), PHP date timestamp (integer),
* PHP DateTime object, or a standard date string
* Or can be an array of date values
- * @param array|int $endDays The number of nonweekend and nonholiday days before or after
+ * @param array|int $endDays The number of nonweekend and nonholiday days before or after
* startDate. A positive value for days yields a future date; a
* negative value yields a past date.
* Or can be an array of int values
* @param null|mixed $dateArgs An array of dates (such as holidays) to exclude from the calculation
*
- * @return array|DateTime|float|int|string Excel date/time serial value, PHP date/time serial value or PHP date/time object,
+ * @return array|DateTime|float|int|string Excel date/time serial value, PHP date/time serial value or PHP date/time object,
* depending on the value of the ReturnDateType flag
* If an array of values is passed for the $startDate or $endDays,arguments, then the returned result
* will also be an array with matching dimensions
@@ -72,6 +72,8 @@ class WorkDay
/**
* Use incrementing logic to determine Workday.
+ *
+ * @param array $holidayArray
*/
private static function incrementing(float $startDate, int $endDays, array $holidayArray): float|int|DateTime
{
@@ -103,10 +105,12 @@ class WorkDay
return Helpers::returnIn3FormatsFloat($endDate);
}
+ /** @param array $holidayArray */
private static function incrementingArray(float $startDate, float $endDate, array $holidayArray): float
{
$holidayCountedArray = $holidayDates = [];
foreach ($holidayArray as $holidayDate) {
+ /** @var float $holidayDate */
if (self::getWeekDay($holidayDate, 3) < 5) {
$holidayDates[] = $holidayDate;
}
@@ -131,6 +135,8 @@ class WorkDay
/**
* Use decrementing logic to determine Workday.
+ *
+ * @param array $holidayArray
*/
private static function decrementing(float $startDate, int $endDays, array $holidayArray): float|int|DateTime
{
@@ -162,10 +168,12 @@ class WorkDay
return Helpers::returnIn3FormatsFloat($endDate);
}
+ /** @param array $holidayArray */
private static function decrementingArray(float $startDate, float $endDate, array $holidayArray): float
{
$holidayCountedArray = $holidayDates = [];
foreach ($holidayArray as $holidayDate) {
+ /** @var float $holidayDate */
if (self::getWeekDay($holidayDate, 3) < 5) {
$holidayDates[] = $holidayDate;
}
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/YearFrac.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/YearFrac.php
index 2713754a..6b3d0f4b 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/YearFrac.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/DateTimeExcel/YearFrac.php
@@ -31,7 +31,7 @@ class YearFrac
* @param mixed $endDate Excel date serial value (float), PHP date timestamp (integer),
* PHP DateTime object, or a standard date string
* Or can be an array of methods
- * @param array|int $method Method used for the calculation
+ * @param array|int $method Method used for the calculation
* 0 or omitted US (NASD) 30/360
* 1 Actual/actual
* 2 Actual/360
@@ -39,7 +39,7 @@ class YearFrac
* 4 European 30/360
* Or can be an array of methods
*
- * @return array|float|int|string fraction of the year, or a string containing an error
+ * @return array|float|int|string fraction of the year, or a string containing an error
* If an array of values is passed for the $startDate or $endDays,arguments, then the returned result
* will also be an array with matching dimensions
*/
@@ -62,11 +62,11 @@ class YearFrac
}
return match ($method) {
- 0 => Functions::scalar(Days360::between($startDate, $endDate)) / 360,
+ 0 => Helpers::floatOrInt(Days360::between($startDate, $endDate)) / 360,
1 => self::method1($startDate, $endDate),
- 2 => Functions::scalar(Difference::interval($startDate, $endDate)) / 360,
- 3 => Functions::scalar(Difference::interval($startDate, $endDate)) / 365,
- 4 => Functions::scalar(Days360::between($startDate, $endDate, true)) / 360,
+ 2 => Helpers::floatOrInt(Difference::interval($startDate, $endDate)) / 360,
+ 3 => Helpers::floatOrInt(Difference::interval($startDate, $endDate)) / 365,
+ 4 => Helpers::floatOrInt(Days360::between($startDate, $endDate, true)) / 360,
default => ExcelError::NAN(),
};
}
@@ -91,7 +91,7 @@ class YearFrac
private static function method1(float $startDate, float $endDate): float
{
- $days = Functions::scalar(Difference::interval($startDate, $endDate));
+ $days = Helpers::floatOrInt(Difference::interval($startDate, $endDate));
$startYear = (int) DateParts::year($startDate);
$endYear = (int) DateParts::year($endDate);
$years = $endYear - $startYear + 1;
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engine/ArrayArgumentHelper.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engine/ArrayArgumentHelper.php
index 0107f404..9b4e6c02 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engine/ArrayArgumentHelper.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engine/ArrayArgumentHelper.php
@@ -8,14 +8,18 @@ class ArrayArgumentHelper
{
protected int $indexStart = 0;
+ /** @var mixed[] */
protected array $arguments;
protected int $argumentCount;
+ /** @var int[] */
protected array $rows;
+ /** @var int[] */
protected array $columns;
+ /** @param mixed[] $arguments */
public function initialise(array $arguments): void
{
$keys = array_keys($arguments);
@@ -34,6 +38,7 @@ class ArrayArgumentHelper
}
}
+ /** @return mixed[] */
public function arguments(): array
{
return $this->arguments;
@@ -65,6 +70,7 @@ class ArrayArgumentHelper
return count($rowVectors) === 1 ? array_pop($rowVectors) : null;
}
+ /** @return int[] */
private function getRowVectors(): array
{
$rowVectors = [];
@@ -84,6 +90,7 @@ class ArrayArgumentHelper
return count($columnVectors) === 1 ? array_pop($columnVectors) : null;
}
+ /** @return int[] */
private function getColumnVectors(): array
{
$columnVectors = [];
@@ -96,6 +103,7 @@ class ArrayArgumentHelper
return $columnVectors;
}
+ /** @return int[] */
public function getMatrixPair(): array
{
for ($i = $this->indexStart; $i < ($this->indexStart + $this->argumentCount - 1); ++$i) {
@@ -134,6 +142,11 @@ class ArrayArgumentHelper
return $this->columns[$argument];
}
+ /**
+ * @param mixed[] $arguments
+ *
+ * @return int[]
+ */
private function rows(array $arguments): array
{
return array_map(
@@ -142,14 +155,17 @@ class ArrayArgumentHelper
);
}
+ /**
+ * @param mixed[] $arguments
+ *
+ * @return int[]
+ */
private function columns(array $arguments): array
{
return array_map(
- function (mixed $argument): int {
- return is_array($argument) && is_array($argument[array_keys($argument)[0]])
+ fn (mixed $argument): int => is_array($argument) && is_array($argument[array_keys($argument)[0]])
? count($argument[array_keys($argument)[0]])
- : 1;
- },
+ : 1,
$arguments
);
}
@@ -166,6 +182,13 @@ class ArrayArgumentHelper
return $count;
}
+ /**
+ * @param mixed[] $arguments
+ * @param int[] $rows
+ * @param int[] $columns
+ *
+ * @return mixed[]
+ */
private function flattenSingleCellArrays(array $arguments, array $rows, array $columns): array
{
foreach ($arguments as $index => $argument) {
@@ -180,6 +203,11 @@ class ArrayArgumentHelper
return $arguments;
}
+ /**
+ * @param mixed[] $array
+ *
+ * @return mixed[]
+ */
private function filterArray(array $array): array
{
return array_filter(
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engine/ArrayArgumentProcessor.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engine/ArrayArgumentProcessor.php
index fb2c853b..7004d087 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engine/ArrayArgumentProcessor.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engine/ArrayArgumentProcessor.php
@@ -2,12 +2,14 @@
namespace PhpOffice\PhpSpreadsheet\Calculation\Engine;
+use PhpOffice\PhpSpreadsheet\Calculation\Exception;
use PhpOffice\PhpSpreadsheet\Calculation\Functions;
class ArrayArgumentProcessor
{
private static ArrayArgumentHelper $arrayArgumentHelper;
+ /** @return mixed[] */
public static function processArguments(
ArrayArgumentHelper $arrayArgumentHelper,
callable $method,
@@ -54,23 +56,30 @@ class ArrayArgumentProcessor
return ['#VALUE!'];
}
+ /**
+ * @param int[] $matrixIndexes
+ *
+ * @return mixed[]
+ */
private static function evaluateVectorMatrixPair(callable $method, array $matrixIndexes, mixed ...$arguments): array
{
- $matrix2 = array_pop($matrixIndexes);
- /** @var array $matrixValues2 */
+ $matrix2 = array_pop($matrixIndexes) ?? throw new Exception('empty array 2');
+ /** @var mixed[][] $matrixValues2 */
$matrixValues2 = $arguments[$matrix2];
- $matrix1 = array_pop($matrixIndexes);
- /** @var array $matrixValues1 */
+ $matrix1 = array_pop($matrixIndexes) ?? throw new Exception('empty array 1');
+ /** @var mixed[][] $matrixValues1 */
$matrixValues1 = $arguments[$matrix1];
- $rows = min(array_map([self::$arrayArgumentHelper, 'rowCount'], [$matrix1, $matrix2]));
- $columns = min(array_map([self::$arrayArgumentHelper, 'columnCount'], [$matrix1, $matrix2]));
+ /** @var non-empty-array */
+ $matrix12 = [$matrix1, $matrix2];
+ $rows = min(array_map(self::$arrayArgumentHelper->rowCount(...), $matrix12));
+ $columns = min(array_map(self::$arrayArgumentHelper->columnCount(...), $matrix12));
if ($rows === 1) {
- $rows = max(array_map([self::$arrayArgumentHelper, 'rowCount'], [$matrix1, $matrix2]));
+ $rows = max(array_map(self::$arrayArgumentHelper->rowCount(...), $matrix12));
}
if ($columns === 1) {
- $columns = max(array_map([self::$arrayArgumentHelper, 'columnCount'], [$matrix1, $matrix2]));
+ $columns = max(array_map(self::$arrayArgumentHelper->columnCount(...), $matrix12));
}
$result = [];
@@ -92,13 +101,18 @@ class ArrayArgumentProcessor
return $result;
}
+ /**
+ * @param mixed[] $matrixIndexes
+ *
+ * @return mixed[]
+ */
private static function evaluateMatrixPair(callable $method, array $matrixIndexes, mixed ...$arguments): array
{
$matrix2 = array_pop($matrixIndexes);
- /** @var array $matrixValues2 */
+ /** @var mixed[][] $matrixValues2 */
$matrixValues2 = $arguments[$matrix2];
$matrix1 = array_pop($matrixIndexes);
- /** @var array $matrixValues1 */
+ /** @var mixed[][] $matrixValues1 */
$matrixValues1 = $arguments[$matrix1];
$result = [];
@@ -119,6 +133,7 @@ class ArrayArgumentProcessor
return $result;
}
+ /** @return mixed[] */
private static function evaluateVectorPair(callable $method, int $rowIndex, int $columnIndex, mixed ...$arguments): array
{
$rowVector = Functions::flattenArray($arguments[$rowIndex]);
@@ -141,11 +156,13 @@ class ArrayArgumentProcessor
/**
* Note, offset is from 1 (for the first argument) rather than from 0.
+ *
+ * @return mixed[]
*/
private static function evaluateNthArgumentAsArray(callable $method, int $nthArgument, mixed ...$arguments): array
{
$values = array_slice($arguments, $nthArgument - 1, 1);
- /** @var array $values */
+ /** @var mixed[] $values */
$values = array_pop($values);
$result = [];
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engine/FormattedNumber.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engine/FormattedNumber.php
index 331fa448..c64c8d27 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engine/FormattedNumber.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engine/FormattedNumber.php
@@ -16,40 +16,30 @@ class FormattedNumber
// preg_quoted string for major currency symbols, with a %s for locale currency
private const CURRENCY_CONVERSION_LIST = '\$€£¥%s';
- private const STRING_CONVERSION_LIST = [
- [self::class, 'convertToNumberIfNumeric'],
- [self::class, 'convertToNumberIfFraction'],
- [self::class, 'convertToNumberIfPercent'],
- [self::class, 'convertToNumberIfCurrency'],
- ];
-
/**
* Identify whether a string contains a formatted numeric value,
* and convert it to a numeric if it is.
*
- * @param string $operand string value to test
+ * @param float|string $operand string value to test
*/
- public static function convertToNumberIfFormatted(string &$operand): bool
+ public static function convertToNumberIfFormatted(float|string &$operand): bool
{
- foreach (self::STRING_CONVERSION_LIST as $conversionMethod) {
- if ($conversionMethod($operand) === true) {
- return true;
- }
- }
-
- return false;
+ return self::convertToNumberIfNumeric($operand)
+ || self::convertToNumberIfFraction($operand)
+ || self::convertToNumberIfPercent($operand)
+ || self::convertToNumberIfCurrency($operand);
}
/**
* Identify whether a string contains a numeric value,
* and convert it to a numeric if it is.
*
- * @param string $operand string value to test
+ * @param float|string $operand string value to test
*/
- public static function convertToNumberIfNumeric(string &$operand): bool
+ public static function convertToNumberIfNumeric(float|string &$operand): bool
{
$thousandsSeparator = preg_quote(StringHelper::getThousandsSeparator(), '/');
- $value = preg_replace(['/(\d)' . $thousandsSeparator . '(\d)/u', '/([+-])\s+(\d)/u'], ['$1$2', '$1$2'], trim($operand));
+ $value = preg_replace(['/(\d)' . $thousandsSeparator . '(\d)/u', '/([+-])\s+(\d)/u'], ['$1$2', '$1$2'], trim("$operand"));
$decimalSeparator = preg_quote(StringHelper::getDecimalSeparator(), '/');
$value = preg_replace(['/(\d)' . $decimalSeparator . '(\d)/u', '/([+-])\s+(\d)/u'], ['$1.$2', '$1$2'], $value ?? '');
@@ -68,13 +58,15 @@ class FormattedNumber
*
* @param string $operand string value to test
*/
- public static function convertToNumberIfFraction(string &$operand): bool
+ public static function convertToNumberIfFraction(float|string &$operand): bool
{
- if (preg_match(self::STRING_REGEXP_FRACTION, $operand, $match)) {
+ if (is_string($operand) && preg_match(self::STRING_REGEXP_FRACTION, $operand, $match)) {
$sign = ($match[1] === '-') ? '-' : '+';
$wholePart = ($match[3] === '') ? '' : ($sign . $match[3]);
$fractionFormula = '=' . $wholePart . $sign . $match[4];
- $operand = Calculation::getInstance()->_calculateFormulaValue($fractionFormula);
+ /** @var string */
+ $operandx = Calculation::getInstance()->_calculateFormulaValue($fractionFormula);
+ $operand = $operandx;
return true;
}
@@ -86,12 +78,12 @@ class FormattedNumber
* Identify whether a string contains a percentage, and if so,
* convert it to a numeric.
*
- * @param string $operand string value to test
+ * @param float|string $operand string value to test
*/
- public static function convertToNumberIfPercent(string &$operand): bool
+ public static function convertToNumberIfPercent(float|string &$operand): bool
{
$thousandsSeparator = preg_quote(StringHelper::getThousandsSeparator(), '/');
- $value = preg_replace('/(\d)' . $thousandsSeparator . '(\d)/u', '$1$2', trim($operand));
+ $value = preg_replace('/(\d)' . $thousandsSeparator . '(\d)/u', '$1$2', trim("$operand"));
$decimalSeparator = preg_quote(StringHelper::getDecimalSeparator(), '/');
$value = preg_replace(['/(\d)' . $decimalSeparator . '(\d)/u', '/([+-])\s+(\d)/u'], ['$1.$2', '$1$2'], $value ?? '');
@@ -111,13 +103,13 @@ class FormattedNumber
* Identify whether a string contains a currency value, and if so,
* convert it to a numeric.
*
- * @param string $operand string value to test
+ * @param float|string $operand string value to test
*/
- public static function convertToNumberIfCurrency(string &$operand): bool
+ public static function convertToNumberIfCurrency(float|string &$operand): bool
{
$currencyRegexp = self::currencyMatcherRegexp();
$thousandsSeparator = preg_quote(StringHelper::getThousandsSeparator(), '/');
- $value = preg_replace('/(\d)' . $thousandsSeparator . '(\d)/u', '$1$2', $operand);
+ $value = preg_replace('/(\d)' . $thousandsSeparator . '(\d)/u', '$1$2', "$operand");
$match = [];
if ($value !== null && preg_match($currencyRegexp, $value, $match, PREG_UNMATCHED_AS_NULL)) {
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engine/Logger.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engine/Logger.php
index 9adcd559..e82faf5f 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engine/Logger.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engine/Logger.php
@@ -78,7 +78,7 @@ class Logger
{
// Only write the debug log if logging is enabled
if ($this->writeDebugLog) {
- $message = sprintf($message, ...$args);
+ $message = sprintf($message, ...$args); //* @phpstan-ignore-line
$cellReference = implode(' -> ', $this->cellStack->showStack());
if ($this->echoDebugLog) {
echo $cellReference,
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engine/Operands/Operand.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engine/Operands/Operand.php
index 05264c3f..87a733f9 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engine/Operands/Operand.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engine/Operands/Operand.php
@@ -4,6 +4,7 @@ namespace PhpOffice\PhpSpreadsheet\Calculation\Engine\Operands;
interface Operand
{
+ /** @param string[] $matches */
public static function fromParser(string $formula, int $index, array $matches): self;
public function value(): string;
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engine/Operands/StructuredReference.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engine/Operands/StructuredReference.php
index fc2b5ea4..9c4cf726 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engine/Operands/StructuredReference.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engine/Operands/StructuredReference.php
@@ -6,6 +6,7 @@ use PhpOffice\PhpSpreadsheet\Calculation\Calculation;
use PhpOffice\PhpSpreadsheet\Calculation\Exception;
use PhpOffice\PhpSpreadsheet\Cell\Cell;
use PhpOffice\PhpSpreadsheet\Cell\Coordinate;
+use PhpOffice\PhpSpreadsheet\Shared\StringHelper;
use PhpOffice\PhpSpreadsheet\Worksheet\Table;
use Stringable;
@@ -29,7 +30,7 @@ final class StructuredReference implements Operand, Stringable
self::ITEM_SPECIFIER_TOTALS,
];
- private const TABLE_REFERENCE = '/([\p{L}_\\\\][\p{L}\p{N}\._]+)?(\[(?:[^\]\[]+|(?R))*+\])/miu';
+ private const TABLE_REFERENCE = '/([\p{L}_\\\][\p{L}\p{N}\._]+)?(\[(?:[^\]\[]+|(?R))*+\])/miu';
private string $value;
@@ -47,6 +48,7 @@ final class StructuredReference implements Operand, Stringable
private ?int $totalsRow;
+ /** @var mixed[] */
private array $columns;
public function __construct(string $structuredReference)
@@ -54,6 +56,7 @@ final class StructuredReference implements Operand, Stringable
$this->value = $structuredReference;
}
+ /** @param string[] $matches */
public static function fromParser(string $formula, int $index, array $matches): self
{
$val = $matches[0];
@@ -171,14 +174,20 @@ final class StructuredReference implements Operand, Stringable
return $table;
}
+ /**
+ * @param array{array{string, int}, array{string, int}} $tableRange
+ *
+ * @return mixed[]
+ */
private function getColumns(Cell $cell, array $tableRange): array
{
$worksheet = $cell->getWorksheet();
$cellReference = $cell->getCoordinate();
$columns = [];
- $lastColumn = ++$tableRange[1][0];
- for ($column = $tableRange[0][0]; $column !== $lastColumn; ++$column) {
+ $lastColumn = StringHelper::stringIncrement($tableRange[1][0]);
+ for ($column = $tableRange[0][0]; $column !== $lastColumn; StringHelper::stringIncrement($column)) {
+ /** @var string $column */
$columns[$column] = $worksheet
->getCell($column . ($this->headersRow ?? ($this->firstDataRow - 1)))
->getCalculatedValue();
@@ -196,7 +205,7 @@ final class StructuredReference implements Operand, Stringable
$reference = str_replace('[' . self::ITEM_SPECIFIER_THIS_ROW . '],', '', $reference);
foreach ($this->columns as $columnId => $columnName) {
- $columnName = str_replace("\u{a0}", ' ', $columnName);
+ $columnName = str_replace("\u{a0}", ' ', $columnName); //* @phpstan-ignore-line
$reference = $this->adjustRowReference($columnName, $reference, $cell, $columnId);
}
@@ -330,7 +339,7 @@ final class StructuredReference implements Operand, Stringable
{
$columnsSelected = false;
foreach ($this->columns as $columnId => $columnName) {
- $columnName = str_replace("\u{a0}", ' ', $columnName ?? '');
+ $columnName = str_replace("\u{a0}", ' ', $columnName ?? ''); //* @phpstan-ignore-line
$cellFrom = "{$columnId}{$startRow}";
$cellTo = "{$columnId}{$endRow}";
$cellReference = ($cellFrom === $cellTo) ? $cellFrom : "{$cellFrom}:{$cellTo}";
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/BesselI.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/BesselI.php
index 5d564a05..7f67806c 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/BesselI.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/BesselI.php
@@ -31,7 +31,7 @@ class BesselI
* If $ord < 0, BESSELI returns the #NUM! error value.
* Or can be an array of values
*
- * @return array|float|string Result, or a string containing an error
+ * @return array|float|string Result, or a string containing an error
* If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/BesselJ.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/BesselJ.php
index 4a9d9ffd..58ae2584 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/BesselJ.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/BesselJ.php
@@ -30,7 +30,7 @@ class BesselJ
* If $ord < 0, BESSELJ returns the #NUM! error value.
* Or can be an array of values
*
- * @return array|float|string Result, or a string containing an error
+ * @return array|float|string Result, or a string containing an error
* If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/BesselK.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/BesselK.php
index 5a9bd54c..13e85cbe 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/BesselK.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/BesselK.php
@@ -28,7 +28,7 @@ class BesselK
* If $ord < 0, BESSELKI returns the #NUM! error value.
* Or can be an array of values
*
- * @return array|float|string Result, or a string containing an error
+ * @return array|float|string Result, or a string containing an error
* If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/BesselY.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/BesselY.php
index 5d99638a..c65a01bc 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/BesselY.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/BesselY.php
@@ -27,7 +27,7 @@ class BesselY
* If $ord < 0, BESSELY returns the #NUM! error value.
* Or can be an array of values
*
- * @return array|float|string Result, or a string containing an error
+ * @return array|float|string Result, or a string containing an error
* If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/BitWise.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/BitWise.php
index c861c21a..65192bf9 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/BitWise.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/BitWise.php
@@ -30,10 +30,10 @@ class BitWise
* Excel Function:
* BITAND(number1, number2)
*
- * @param null|array|bool|float|int|string $number1 Or can be an array of values
- * @param null|array|bool|float|int|string $number2 Or can be an array of values
+ * @param null|array|bool|float|int|string $number1 Or can be an array of values
+ * @param null|array|bool|float|int|string $number2 Or can be an array of values
*
- * @return array|int|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|int|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function BITAND(null|array|bool|float|int|string $number1, null|array|bool|float|int|string $number2): array|string|int|float
@@ -62,10 +62,10 @@ class BitWise
* Excel Function:
* BITOR(number1, number2)
*
- * @param null|array|bool|float|int|string $number1 Or can be an array of values
- * @param null|array|bool|float|int|string $number2 Or can be an array of values
+ * @param null|array|bool|float|int|string $number1 Or can be an array of values
+ * @param null|array|bool|float|int|string $number2 Or can be an array of values
*
- * @return array|int|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|int|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function BITOR(null|array|bool|float|int|string $number1, null|array|bool|float|int|string $number2): array|string|int|float
@@ -95,10 +95,10 @@ class BitWise
* Excel Function:
* BITXOR(number1, number2)
*
- * @param null|array|bool|float|int|string $number1 Or can be an array of values
- * @param null|array|bool|float|int|string $number2 Or can be an array of values
+ * @param null|array|bool|float|int|string $number1 Or can be an array of values
+ * @param null|array|bool|float|int|string $number2 Or can be an array of values
*
- * @return array|int|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|int|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function BITXOR(null|array|bool|float|int|string $number1, null|array|bool|float|int|string $number2): array|string|int|float
@@ -128,10 +128,10 @@ class BitWise
* Excel Function:
* BITLSHIFT(number, shift_amount)
*
- * @param null|array|bool|float|int|string $number Or can be an array of values
- * @param null|array|bool|float|int|string $shiftAmount Or can be an array of values
+ * @param null|array|bool|float|int|string $number Or can be an array of values
+ * @param null|array|bool|float|int|string $shiftAmount Or can be an array of values
*
- * @return array|float|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|float|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function BITLSHIFT(null|array|bool|float|int|string $number, null|array|bool|float|int|string $shiftAmount): array|string|float
@@ -163,10 +163,10 @@ class BitWise
* Excel Function:
* BITRSHIFT(number, shift_amount)
*
- * @param null|array|bool|float|int|string $number Or can be an array of values
- * @param null|array|bool|float|int|string $shiftAmount Or can be an array of values
+ * @param null|array|bool|float|int|string $number Or can be an array of values
+ * @param null|array|bool|float|int|string $shiftAmount Or can be an array of values
*
- * @return array|float|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|float|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function BITRSHIFT(null|array|bool|float|int|string $number, null|array|bool|float|int|string $shiftAmount): array|string|float
@@ -221,7 +221,7 @@ class BitWise
$value = self::nullFalseTrueToNumber($value);
if (is_numeric($value)) {
- if (abs($value) > 53) {
+ if (abs($value + 0) > 53) {
throw new Exception(ExcelError::NAN());
}
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/Compare.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/Compare.php
index 9e3275fc..ac7ef514 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/Compare.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/Compare.php
@@ -20,12 +20,12 @@ class Compare
* functions you calculate the count of equal pairs. This function is also known as the
* Kronecker Delta function.
*
- * @param array|bool|float|int|string $a the first number
+ * @param array|bool|float|int|string $a the first number
* Or can be an array of values
- * @param array|bool|float|int|string $b The second number. If omitted, b is assumed to be zero.
+ * @param array|bool|float|int|string $b The second number. If omitted, b is assumed to be zero.
* Or can be an array of values
*
- * @return array|int|string (string in the event of an error)
+ * @return array|int|string (string in the event of an error)
* If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
@@ -55,12 +55,12 @@ class Compare
* Use this function to filter a set of values. For example, by summing several GESTEP
* functions you calculate the count of values that exceed a threshold.
*
- * @param array|bool|float|int|string $number the value to test against step
+ * @param array|bool|float|int|string $number the value to test against step
* Or can be an array of values
- * @param null|array|bool|float|int|string $step The threshold value. If you omit a value for step, GESTEP uses zero.
+ * @param null|array|bool|float|int|string $step The threshold value. If you omit a value for step, GESTEP uses zero.
* Or can be an array of values
*
- * @return array|int|string (string in the event of an error)
+ * @return array|int|string (string in the event of an error)
* If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/Complex.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/Complex.php
index 3e41371b..83d110c2 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/Complex.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/Complex.php
@@ -28,7 +28,7 @@ class Complex
* If omitted, the suffix is assumed to be "i".
* Or can be an array of values
*
- * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function COMPLEX(mixed $realNumber = 0.0, mixed $imaginary = 0.0, mixed $suffix = 'i'): array|string
@@ -65,11 +65,11 @@ class Complex
* Excel Function:
* IMAGINARY(complexNumber)
*
- * @param array|string $complexNumber the complex number for which you want the imaginary
+ * @param array|string $complexNumber the complex number for which you want the imaginary
* coefficient
* Or can be an array of values
*
- * @return array|float|string (string if an error)
+ * @return array|float|string (string if an error)
* If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
@@ -96,10 +96,10 @@ class Complex
* Excel Function:
* IMREAL(complexNumber)
*
- * @param array|string $complexNumber the complex number for which you want the real coefficient
+ * @param array|string $complexNumber the complex number for which you want the real coefficient
* Or can be an array of values
*
- * @return array|float|string (string if an error)
+ * @return array|float|string (string if an error)
* If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ComplexFunctions.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ComplexFunctions.php
index d1b7764a..5dd51be9 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ComplexFunctions.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ComplexFunctions.php
@@ -19,10 +19,10 @@ class ComplexFunctions
* Excel Function:
* IMABS(complexNumber)
*
- * @param array|string $complexNumber the complex number for which you want the absolute value
+ * @param array|string $complexNumber the complex number for which you want the absolute value
* Or can be an array of values
*
- * @return array|float|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|float|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function IMABS(array|string $complexNumber): array|float|string
@@ -49,10 +49,10 @@ class ComplexFunctions
* Excel Function:
* IMARGUMENT(complexNumber)
*
- * @param array|string $complexNumber the complex number for which you want the argument theta
+ * @param array|string $complexNumber the complex number for which you want the argument theta
* Or can be an array of values
*
- * @return array|float|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|float|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function IMARGUMENT(array|string $complexNumber): array|float|string
@@ -82,10 +82,10 @@ class ComplexFunctions
* Excel Function:
* IMCONJUGATE(complexNumber)
*
- * @param array|string $complexNumber the complex number for which you want the conjugate
+ * @param array|string $complexNumber the complex number for which you want the conjugate
* Or can be an array of values
*
- * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function IMCONJUGATE(array|string $complexNumber): array|string
@@ -111,10 +111,10 @@ class ComplexFunctions
* Excel Function:
* IMCOS(complexNumber)
*
- * @param array|string $complexNumber the complex number for which you want the cosine
+ * @param array|string $complexNumber the complex number for which you want the cosine
* Or can be an array of values
*
- * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function IMCOS(array|string $complexNumber): array|string
@@ -140,10 +140,10 @@ class ComplexFunctions
* Excel Function:
* IMCOSH(complexNumber)
*
- * @param array|string $complexNumber the complex number for which you want the hyperbolic cosine
+ * @param array|string $complexNumber the complex number for which you want the hyperbolic cosine
* Or can be an array of values
*
- * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function IMCOSH(array|string $complexNumber): array|string
@@ -169,10 +169,10 @@ class ComplexFunctions
* Excel Function:
* IMCOT(complexNumber)
*
- * @param array|string $complexNumber the complex number for which you want the cotangent
+ * @param array|string $complexNumber the complex number for which you want the cotangent
* Or can be an array of values
*
- * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function IMCOT(array|string $complexNumber): array|string
@@ -198,10 +198,10 @@ class ComplexFunctions
* Excel Function:
* IMCSC(complexNumber)
*
- * @param array|string $complexNumber the complex number for which you want the cosecant
+ * @param array|string $complexNumber the complex number for which you want the cosecant
* Or can be an array of values
*
- * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function IMCSC(array|string $complexNumber): array|string
@@ -227,10 +227,10 @@ class ComplexFunctions
* Excel Function:
* IMCSCH(complexNumber)
*
- * @param array|string $complexNumber the complex number for which you want the hyperbolic cosecant
+ * @param array|string $complexNumber the complex number for which you want the hyperbolic cosecant
* Or can be an array of values
*
- * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function IMCSCH(array|string $complexNumber): array|string
@@ -256,10 +256,10 @@ class ComplexFunctions
* Excel Function:
* IMSIN(complexNumber)
*
- * @param array|string $complexNumber the complex number for which you want the sine
+ * @param array|string $complexNumber the complex number for which you want the sine
* Or can be an array of values
*
- * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function IMSIN(array|string $complexNumber): array|string
@@ -285,10 +285,10 @@ class ComplexFunctions
* Excel Function:
* IMSINH(complexNumber)
*
- * @param array|string $complexNumber the complex number for which you want the hyperbolic sine
+ * @param array|string $complexNumber the complex number for which you want the hyperbolic sine
* Or can be an array of values
*
- * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function IMSINH(array|string $complexNumber): array|string
@@ -314,10 +314,10 @@ class ComplexFunctions
* Excel Function:
* IMSEC(complexNumber)
*
- * @param array|string $complexNumber the complex number for which you want the secant
+ * @param array|string $complexNumber the complex number for which you want the secant
* Or can be an array of values
*
- * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function IMSEC(array|string $complexNumber): array|string
@@ -343,10 +343,10 @@ class ComplexFunctions
* Excel Function:
* IMSECH(complexNumber)
*
- * @param array|string $complexNumber the complex number for which you want the hyperbolic secant
+ * @param array|string $complexNumber the complex number for which you want the hyperbolic secant
* Or can be an array of values
*
- * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function IMSECH(array|string $complexNumber): array|string
@@ -372,10 +372,10 @@ class ComplexFunctions
* Excel Function:
* IMTAN(complexNumber)
*
- * @param array|string $complexNumber the complex number for which you want the tangent
+ * @param array|string $complexNumber the complex number for which you want the tangent
* Or can be an array of values
*
- * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function IMTAN(array|string $complexNumber): array|string
@@ -401,10 +401,10 @@ class ComplexFunctions
* Excel Function:
* IMSQRT(complexNumber)
*
- * @param array|string $complexNumber the complex number for which you want the square root
+ * @param array|string $complexNumber the complex number for which you want the square root
* Or can be an array of values
*
- * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function IMSQRT(array|string $complexNumber): array|string
@@ -435,10 +435,10 @@ class ComplexFunctions
* Excel Function:
* IMLN(complexNumber)
*
- * @param array|string $complexNumber the complex number for which you want the natural logarithm
+ * @param array|string $complexNumber the complex number for which you want the natural logarithm
* Or can be an array of values
*
- * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function IMLN(array|string $complexNumber): array|string
@@ -468,10 +468,10 @@ class ComplexFunctions
* Excel Function:
* IMLOG10(complexNumber)
*
- * @param array|string $complexNumber the complex number for which you want the common logarithm
+ * @param array|string $complexNumber the complex number for which you want the common logarithm
* Or can be an array of values
*
- * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function IMLOG10(array|string $complexNumber): array|string
@@ -501,10 +501,10 @@ class ComplexFunctions
* Excel Function:
* IMLOG2(complexNumber)
*
- * @param array|string $complexNumber the complex number for which you want the base-2 logarithm
+ * @param array|string $complexNumber the complex number for which you want the base-2 logarithm
* Or can be an array of values
*
- * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function IMLOG2(array|string $complexNumber): array|string
@@ -534,10 +534,10 @@ class ComplexFunctions
* Excel Function:
* IMEXP(complexNumber)
*
- * @param array|string $complexNumber the complex number for which you want the exponential
+ * @param array|string $complexNumber the complex number for which you want the exponential
* Or can be an array of values
*
- * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function IMEXP(array|string $complexNumber): array|string
@@ -563,12 +563,12 @@ class ComplexFunctions
* Excel Function:
* IMPOWER(complexNumber,realNumber)
*
- * @param array|string $complexNumber the complex number you want to raise to a power
+ * @param array|string $complexNumber the complex number you want to raise to a power
* Or can be an array of values
- * @param array|float|int|string $realNumber the power to which you want to raise the complex number
+ * @param array|float|int|string $realNumber the power to which you want to raise the complex number
* Or can be an array of values
*
- * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function IMPOWER(array|string $complexNumber, array|float|int|string $realNumber): array|string
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ComplexOperations.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ComplexOperations.php
index 61efa847..ed66bea1 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ComplexOperations.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ComplexOperations.php
@@ -20,12 +20,12 @@ class ComplexOperations
* Excel Function:
* IMDIV(complexDividend,complexDivisor)
*
- * @param array|string $complexDividend the complex numerator or dividend
+ * @param array|string $complexDividend the complex numerator or dividend
* Or can be an array of values
- * @param array|string $complexDivisor the complex denominator or divisor
+ * @param array|string $complexDivisor the complex denominator or divisor
* Or can be an array of values
*
- * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function IMDIV(array|string $complexDividend, array|string $complexDivisor): array|string
@@ -49,12 +49,12 @@ class ComplexOperations
* Excel Function:
* IMSUB(complexNumber1,complexNumber2)
*
- * @param array|string $complexNumber1 the complex number from which to subtract complexNumber2
+ * @param array|string $complexNumber1 the complex number from which to subtract complexNumber2
* Or can be an array of values
- * @param array|string $complexNumber2 the complex number to subtract from complexNumber1
+ * @param array|string $complexNumber2 the complex number to subtract from complexNumber1
* Or can be an array of values
*
- * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function IMSUB(array|string $complexNumber1, array|string $complexNumber2): array|string
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ConvertBase.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ConvertBase.php
index 1222831a..6aa631a1 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ConvertBase.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ConvertBase.php
@@ -6,6 +6,7 @@ use PhpOffice\PhpSpreadsheet\Calculation\ArrayEnabled;
use PhpOffice\PhpSpreadsheet\Calculation\Exception;
use PhpOffice\PhpSpreadsheet\Calculation\Functions;
use PhpOffice\PhpSpreadsheet\Calculation\Information\ExcelError;
+use PhpOffice\PhpSpreadsheet\Shared\StringHelper;
abstract class ConvertBase
{
@@ -26,7 +27,7 @@ abstract class ConvertBase
}
}
- return strtoupper((string) $value);
+ return strtoupper(StringHelper::convertToString($value));
}
protected static function validatePlaces(mixed $places = null): ?int
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ConvertBinary.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ConvertBinary.php
index 9c00dcb5..3b84ce3a 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ConvertBinary.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ConvertBinary.php
@@ -15,7 +15,7 @@ class ConvertBinary extends ConvertBase
* Excel Function:
* BIN2DEC(x)
*
- * @param array|bool|float|int|string $value The binary number (as a string) that you want to convert. The number
+ * @param array|bool|float|int|string $value The binary number (as a string) that you want to convert. The number
* cannot contain more than 10 characters (10 bits). The most significant
* bit of number is the sign bit. The remaining 9 bits are magnitude bits.
* Negative numbers are represented using two's-complement notation.
@@ -23,7 +23,7 @@ class ConvertBinary extends ConvertBase
* 10 characters (10 bits), BIN2DEC returns the #NUM! error value.
* Or can be an array of values
*
- * @return array|string Result, or an error
+ * @return array|float|int|string Result, or an error
* If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
@@ -44,10 +44,10 @@ class ConvertBinary extends ConvertBase
// Two's Complement
$value = substr($value, -9);
- return '-' . (512 - bindec($value));
+ return -(512 - bindec($value));
}
- return (string) bindec($value);
+ return bindec($value);
}
/**
@@ -58,14 +58,14 @@ class ConvertBinary extends ConvertBase
* Excel Function:
* BIN2HEX(x[,places])
*
- * @param array|bool|float|int|string $value The binary number (as a string) that you want to convert. The number
+ * @param array|bool|float|int|string $value The binary number (as a string) that you want to convert. The number
* cannot contain more than 10 characters (10 bits). The most significant
* bit of number is the sign bit. The remaining 9 bits are magnitude bits.
* Negative numbers are represented using two's-complement notation.
* If number is not a valid binary number, or if number contains more than
* 10 characters (10 bits), BIN2HEX returns the #NUM! error value.
* Or can be an array of values
- * @param null|array|float|int|string $places The number of characters to use. If places is omitted, BIN2HEX uses the
+ * @param null|array|float|int|string $places The number of characters to use. If places is omitted, BIN2HEX uses the
* minimum number of characters necessary. Places is useful for padding the
* return value with leading 0s (zeros).
* If places is not an integer, it is truncated.
@@ -73,7 +73,7 @@ class ConvertBinary extends ConvertBase
* If places is negative, BIN2HEX returns the #NUM! error value.
* Or can be an array of values
*
- * @return array|string Result, or an error
+ * @return array|string Result, or an error
* If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
@@ -111,14 +111,14 @@ class ConvertBinary extends ConvertBase
* Excel Function:
* BIN2OCT(x[,places])
*
- * @param array|bool|float|int|string $value The binary number (as a string) that you want to convert. The number
+ * @param array|bool|float|int|string $value The binary number (as a string) that you want to convert. The number
* cannot contain more than 10 characters (10 bits). The most significant
* bit of number is the sign bit. The remaining 9 bits are magnitude bits.
* Negative numbers are represented using two's-complement notation.
* If number is not a valid binary number, or if number contains more than
* 10 characters (10 bits), BIN2OCT returns the #NUM! error value.
* Or can be an array of values
- * @param null|array|float|int|string $places The number of characters to use. If places is omitted, BIN2OCT uses the
+ * @param null|array|float|int|string $places The number of characters to use. If places is omitted, BIN2OCT uses the
* minimum number of characters necessary. Places is useful for padding the
* return value with leading 0s (zeros).
* If places is not an integer, it is truncated.
@@ -126,7 +126,7 @@ class ConvertBinary extends ConvertBase
* If places is negative, BIN2OCT returns the #NUM! error value.
* Or can be an array of values
*
- * @return array|string Result, or an error
+ * @return array|string Result, or an error
* If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ConvertDecimal.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ConvertDecimal.php
index 923caa96..9834bccb 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ConvertDecimal.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ConvertDecimal.php
@@ -22,7 +22,7 @@ class ConvertDecimal extends ConvertBase
* Excel Function:
* DEC2BIN(x[,places])
*
- * @param array|bool|float|int|string $value The decimal integer you want to convert. If number is negative,
+ * @param array|bool|float|int|string $value The decimal integer you want to convert. If number is negative,
* valid place values are ignored and DEC2BIN returns a 10-character
* (10-bit) binary number in which the most significant bit is the sign
* bit. The remaining 9 bits are magnitude bits. Negative numbers are
@@ -33,7 +33,7 @@ class ConvertDecimal extends ConvertBase
* If DEC2BIN requires more than places characters, it returns the #NUM!
* error value.
* Or can be an array of values
- * @param null|array|float|int|string $places The number of characters to use. If places is omitted, DEC2BIN uses
+ * @param null|array|float|int|string $places The number of characters to use. If places is omitted, DEC2BIN uses
* the minimum number of characters necessary. Places is useful for
* padding the return value with leading 0s (zeros).
* If places is not an integer, it is truncated.
@@ -41,7 +41,7 @@ class ConvertDecimal extends ConvertBase
* If places is zero or negative, DEC2BIN returns the #NUM! error value.
* Or can be an array of values
*
- * @return array|string Result, or an error
+ * @return array|string Result, or an error
* If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
@@ -79,7 +79,7 @@ class ConvertDecimal extends ConvertBase
* Excel Function:
* DEC2HEX(x[,places])
*
- * @param array|bool|float|int|string $value The decimal integer you want to convert. If number is negative,
+ * @param array|bool|float|int|string $value The decimal integer you want to convert. If number is negative,
* places is ignored and DEC2HEX returns a 10-character (40-bit)
* hexadecimal number in which the most significant bit is the sign
* bit. The remaining 39 bits are magnitude bits. Negative numbers
@@ -90,7 +90,7 @@ class ConvertDecimal extends ConvertBase
* If DEC2HEX requires more than places characters, it returns the
* #NUM! error value.
* Or can be an array of values
- * @param null|array|float|int|string $places The number of characters to use. If places is omitted, DEC2HEX uses
+ * @param null|array|float|int|string $places The number of characters to use. If places is omitted, DEC2HEX uses
* the minimum number of characters necessary. Places is useful for
* padding the return value with leading 0s (zeros).
* If places is not an integer, it is truncated.
@@ -98,7 +98,7 @@ class ConvertDecimal extends ConvertBase
* If places is zero or negative, DEC2HEX returns the #NUM! error value.
* Or can be an array of values
*
- * @return array|string Result, or an error
+ * @return array|string Result, or an error
* If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
@@ -155,7 +155,7 @@ class ConvertDecimal extends ConvertBase
* Excel Function:
* DEC2OCT(x[,places])
*
- * @param array|bool|float|int|string $value The decimal integer you want to convert. If number is negative,
+ * @param array|bool|float|int|string $value The decimal integer you want to convert. If number is negative,
* places is ignored and DEC2OCT returns a 10-character (30-bit)
* octal number in which the most significant bit is the sign bit.
* The remaining 29 bits are magnitude bits. Negative numbers are
@@ -166,7 +166,7 @@ class ConvertDecimal extends ConvertBase
* If DEC2OCT requires more than places characters, it returns the
* #NUM! error value.
* Or can be an array of values
- * @param array|int $places The number of characters to use. If places is omitted, DEC2OCT uses
+ * @param array|int $places The number of characters to use. If places is omitted, DEC2OCT uses
* the minimum number of characters necessary. Places is useful for
* padding the return value with leading 0s (zeros).
* If places is not an integer, it is truncated.
@@ -174,7 +174,7 @@ class ConvertDecimal extends ConvertBase
* If places is zero or negative, DEC2OCT returns the #NUM! error value.
* Or can be an array of values
*
- * @return array|string Result, or an error
+ * @return array|string Result, or an error
* If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ConvertHex.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ConvertHex.php
index 0003a9fd..be8ed39f 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ConvertHex.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ConvertHex.php
@@ -15,7 +15,7 @@ class ConvertHex extends ConvertBase
* Excel Function:
* HEX2BIN(x[,places])
*
- * @param array|bool|float|string $value The hexadecimal number you want to convert.
+ * @param array|bool|float|string $value The hexadecimal number you want to convert.
* Number cannot contain more than 10 characters.
* The most significant bit of number is the sign bit (40th bit from the right).
* The remaining 9 bits are magnitude bits.
@@ -26,7 +26,7 @@ class ConvertHex extends ConvertBase
* If number is not a valid hexadecimal number, HEX2BIN returns the #NUM! error value.
* If HEX2BIN requires more than places characters, it returns the #NUM! error value.
* Or can be an array of values
- * @param array|int $places The number of characters to use. If places is omitted,
+ * @param array|int $places The number of characters to use. If places is omitted,
* HEX2BIN uses the minimum number of characters necessary. Places
* is useful for padding the return value with leading 0s (zeros).
* If places is not an integer, it is truncated.
@@ -34,7 +34,7 @@ class ConvertHex extends ConvertBase
* If places is negative, HEX2BIN returns the #NUM! error value.
* Or can be an array of values
*
- * @return array|string Result, or an error
+ * @return array|string Result, or an error
* If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
@@ -65,7 +65,7 @@ class ConvertHex extends ConvertBase
* Excel Function:
* HEX2DEC(x)
*
- * @param array|bool|float|int|string $value The hexadecimal number you want to convert. This number cannot
+ * @param array|bool|float|int|string $value The hexadecimal number you want to convert. This number cannot
* contain more than 10 characters (40 bits). The most significant
* bit of number is the sign bit. The remaining 39 bits are magnitude
* bits. Negative numbers are represented using two's-complement
@@ -74,7 +74,7 @@ class ConvertHex extends ConvertBase
* #NUM! error value.
* Or can be an array of values
*
- * @return array|string Result, or an error
+ * @return array|float|int|string Result, or an error
* If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
@@ -96,7 +96,7 @@ class ConvertHex extends ConvertBase
}
$binX = '';
- foreach (str_split($value) as $char) {
+ foreach (mb_str_split($value, 1, 'UTF-8') as $char) {
$binX .= str_pad(base_convert($char, 16, 2), 4, '0', STR_PAD_LEFT);
}
if (strlen($binX) == 40 && $binX[0] == '1') {
@@ -104,10 +104,10 @@ class ConvertHex extends ConvertBase
$binX[$i] = ($binX[$i] == '1' ? '0' : '1');
}
- return (string) ((bindec($binX) + 1) * -1);
+ return (bindec($binX) + 1) * -1;
}
- return (string) bindec($binX);
+ return bindec($binX);
}
/**
@@ -118,7 +118,7 @@ class ConvertHex extends ConvertBase
* Excel Function:
* HEX2OCT(x[,places])
*
- * @param array|bool|float|int|string $value The hexadecimal number you want to convert. Number cannot
+ * @param array|bool|float|int|string $value The hexadecimal number you want to convert. Number cannot
* contain more than 10 characters. The most significant bit of
* number is the sign bit. The remaining 39 bits are magnitude
* bits. Negative numbers are represented using two's-complement
@@ -132,7 +132,7 @@ class ConvertHex extends ConvertBase
* If HEX2OCT requires more than places characters, it returns
* the #NUM! error value.
* Or can be an array of values
- * @param array|int $places The number of characters to use. If places is omitted, HEX2OCT
+ * @param array|int $places The number of characters to use. If places is omitted, HEX2OCT
* uses the minimum number of characters necessary. Places is
* useful for padding the return value with leading 0s (zeros).
* If places is not an integer, it is truncated.
@@ -141,7 +141,7 @@ class ConvertHex extends ConvertBase
* If places is negative, HEX2OCT returns the #NUM! error value.
* Or can be an array of values
*
- * @return array|string Result, or an error
+ * @return array|string Result, or an error
* If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ConvertOctal.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ConvertOctal.php
index 5e3c1248..03a906d0 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ConvertOctal.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ConvertOctal.php
@@ -15,7 +15,7 @@ class ConvertOctal extends ConvertBase
* Excel Function:
* OCT2BIN(x[,places])
*
- * @param array|bool|float|int|string $value The octal number you want to convert. Number may not
+ * @param array|bool|float|int|string $value The octal number you want to convert. Number may not
* contain more than 10 characters. The most significant
* bit of number is the sign bit. The remaining 29 bits
* are magnitude bits. Negative numbers are represented
@@ -29,7 +29,7 @@ class ConvertOctal extends ConvertBase
* If OCT2BIN requires more than places characters, it
* returns the #NUM! error value.
* Or can be an array of values
- * @param array|int $places The number of characters to use. If places is omitted,
+ * @param array|int $places The number of characters to use. If places is omitted,
* OCT2BIN uses the minimum number of characters necessary.
* Places is useful for padding the return value with
* leading 0s (zeros).
@@ -40,7 +40,7 @@ class ConvertOctal extends ConvertBase
* value.
* Or can be an array of values
*
- * @return array|string Result, or an error
+ * @return array|string Result, or an error
* If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
@@ -69,7 +69,7 @@ class ConvertOctal extends ConvertBase
* Excel Function:
* OCT2DEC(x)
*
- * @param array|bool|float|int|string $value The octal number you want to convert. Number may not contain
+ * @param array|bool|float|int|string $value The octal number you want to convert. Number may not contain
* more than 10 octal characters (30 bits). The most significant
* bit of number is the sign bit. The remaining 29 bits are
* magnitude bits. Negative numbers are represented using
@@ -78,7 +78,7 @@ class ConvertOctal extends ConvertBase
* #NUM! error value.
* Or can be an array of values
*
- * @return array|string Result, or an error
+ * @return array|float|int|string Result, or an error
* If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
@@ -96,7 +96,7 @@ class ConvertOctal extends ConvertBase
}
$binX = '';
- foreach (str_split($value) as $char) {
+ foreach (mb_str_split($value, 1, 'UTF-8') as $char) {
$binX .= str_pad(decbin((int) $char), 3, '0', STR_PAD_LEFT);
}
if (strlen($binX) == 30 && $binX[0] == '1') {
@@ -104,10 +104,10 @@ class ConvertOctal extends ConvertBase
$binX[$i] = ($binX[$i] == '1' ? '0' : '1');
}
- return (string) ((bindec($binX) + 1) * -1);
+ return (bindec($binX) + 1) * -1;
}
- return (string) bindec($binX);
+ return bindec($binX);
}
/**
@@ -118,7 +118,7 @@ class ConvertOctal extends ConvertBase
* Excel Function:
* OCT2HEX(x[,places])
*
- * @param array|bool|float|int|string $value The octal number you want to convert. Number may not contain
+ * @param array|bool|float|int|string $value The octal number you want to convert. Number may not contain
* more than 10 octal characters (30 bits). The most significant
* bit of number is the sign bit. The remaining 29 bits are
* magnitude bits. Negative numbers are represented using
@@ -130,7 +130,7 @@ class ConvertOctal extends ConvertBase
* If OCT2HEX requires more than places characters, it returns
* the #NUM! error value.
* Or can be an array of values
- * @param array|int $places The number of characters to use. If places is omitted, OCT2HEX
+ * @param array|int $places The number of characters to use. If places is omitted, OCT2HEX
* uses the minimum number of characters necessary. Places is useful
* for padding the return value with leading 0s (zeros).
* If places is not an integer, it is truncated.
@@ -138,7 +138,7 @@ class ConvertOctal extends ConvertBase
* If places is negative, OCT2HEX returns the #NUM! error value.
* Or can be an array of values
*
- * @return array|string Result, or an error
+ * @return array|string Result, or an error
* If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ConvertUOM.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ConvertUOM.php
index 969c270a..db1161c0 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ConvertUOM.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ConvertUOM.php
@@ -230,7 +230,7 @@ class ConvertUOM
/**
* Details of the Multiplier prefixes that can be used with Units of Measure in CONVERTUOM().
*
- ** @var array
+ * @var array
*/
private static array $binaryConversionMultipliers = [
'Yi' => ['multiplier' => 2 ** 80, 'name' => 'yobi'],
@@ -435,6 +435,8 @@ class ConvertUOM
/**
* getConversionGroups
* Returns a list of the different conversion groups for UOM conversions.
+ *
+ * @return string[]
*/
public static function getConversionCategories(): array
{
@@ -451,6 +453,8 @@ class ConvertUOM
* Returns an array of units of measure, for a specified conversion group, or for all groups.
*
* @param ?string $category The group whose units of measure you want to retrieve
+ *
+ * @return string[][]
*/
public static function getConversionCategoryUnits(?string $category = null): array
{
@@ -468,6 +472,8 @@ class ConvertUOM
* getConversionGroupUnitDetails.
*
* @param ?string $category The group whose units of measure you want to retrieve
+ *
+ * @return array>>
*/
public static function getConversionCategoryUnitDetails(?string $category = null): array
{
@@ -488,7 +494,7 @@ class ConvertUOM
* getConversionMultipliers
* Returns an array of the Multiplier prefixes that can be used with Units of Measure in CONVERTUOM().
*
- * @return mixed[]
+ * @return array
*/
public static function getConversionMultipliers(): array
{
@@ -499,7 +505,7 @@ class ConvertUOM
* getBinaryConversionMultipliers
* Returns an array of the additional Multiplier prefixes that can be used with Information Units of Measure in CONVERTUOM().
*
- * @return mixed[]
+ * @return array
*/
public static function getBinaryConversionMultipliers(): array
{
@@ -516,14 +522,14 @@ class ConvertUOM
* Excel Function:
* CONVERT(value,fromUOM,toUOM)
*
- * @param array|float|int|string $value the value in fromUOM to convert
+ * @param array|float|int|string $value the value in fromUOM to convert
* Or can be an array of values
- * @param array|string $fromUOM the units for value
+ * @param string|string[] $fromUOM the units for value
* Or can be an array of values
- * @param array|string $toUOM the units for the result
+ * @param string|string[] $toUOM the units for the result
* Or can be an array of values
*
- * @return array|float|string Result, or a string containing an error
+ * @return float|mixed[]|string Result, or a string containing an error
* If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
@@ -566,6 +572,7 @@ class ConvertUOM
return ($baseValue * self::$unitConversions[$fromCategory][$toUOM]) / $toMultiplier;
}
+ /** @return array{0: string, 1: string, 2: float} */
private static function getUOMDetails(string $uom): array
{
if (isset(self::$conversionUnits[$uom])) {
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/Erf.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/Erf.php
index aee7e317..7f599dd4 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/Erf.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/Erf.php
@@ -31,7 +31,7 @@ class Erf
* If omitted, ERF integrates between zero and lower_limit
* Or can be an array of values
*
- * @return array|float|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|float|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function ERF(mixed $lower, mixed $upper = null): array|float|string
@@ -63,7 +63,7 @@ class Erf
* @param mixed $limit Float bound for integrating ERF, other bound is zero
* Or can be an array of values
*
- * @return array|float|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|float|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function ERFPRECISE(mixed $limit)
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ErfC.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ErfC.php
index 4365fecc..4cca2d6b 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ErfC.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Engineering/ErfC.php
@@ -26,7 +26,7 @@ class ErfC
* @param mixed $value The float lower bound for integrating ERFC
* Or can be an array of values
*
- * @return array|float|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|float|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function ERFC(mixed $value)
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Amortization.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Amortization.php
index b53829b7..4344244e 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Amortization.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Amortization.php
@@ -54,9 +54,7 @@ class Amortization
$salvage = Functions::flattenSingleValue($salvage);
$period = Functions::flattenSingleValue($period);
$rate = Functions::flattenSingleValue($rate);
- $basis = ($basis === null)
- ? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD
- : Functions::flattenSingleValue($basis);
+ $basis = Functions::flattenSingleValue($basis) ?? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD;
try {
$cost = FinancialValidations::validateFloat($cost);
@@ -141,9 +139,7 @@ class Amortization
$salvage = Functions::flattenSingleValue($salvage);
$period = Functions::flattenSingleValue($period);
$rate = Functions::flattenSingleValue($rate);
- $basis = ($basis === null)
- ? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD
- : Functions::flattenSingleValue($basis);
+ $basis = Functions::flattenSingleValue($basis) ?? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD;
try {
$cost = FinancialValidations::validateFloat($cost);
@@ -171,9 +167,13 @@ class Amortization
if (
$basis == FinancialConstants::BASIS_DAYS_PER_YEAR_ACTUAL
&& $yearFrac < 1
- && DateTimeExcel\Helpers::isLeapYear(Functions::scalar($purchasedYear))
) {
- $yearFrac *= 365 / 366;
+ $temp = Functions::scalar($purchasedYear);
+ if (is_int($temp) || is_string($temp)) {
+ if (DateTimeExcel\Helpers::isLeapYear($temp)) {
+ $yearFrac *= 365 / 366;
+ }
+ }
}
$f0Rate = $yearFrac * $rate * $cost;
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/CashFlow/Constant/Periodic.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/CashFlow/Constant/Periodic.php
index 08cef3ee..2744c55d 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/CashFlow/Constant/Periodic.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/CashFlow/Constant/Periodic.php
@@ -38,9 +38,9 @@ class Periodic
): string|float {
$rate = Functions::flattenSingleValue($rate);
$numberOfPeriods = Functions::flattenSingleValue($numberOfPeriods);
- $payment = ($payment === null) ? 0.0 : Functions::flattenSingleValue($payment);
- $presentValue = ($presentValue === null) ? 0.0 : Functions::flattenSingleValue($presentValue);
- $type = ($type === null) ? FinancialConstants::PAYMENT_END_OF_PERIOD : Functions::flattenSingleValue($type);
+ $payment = Functions::flattenSingleValue($payment) ?? 0.0;
+ $presentValue = Functions::flattenSingleValue($presentValue) ?? 0.0;
+ $type = Functions::flattenSingleValue($type) ?? FinancialConstants::PAYMENT_END_OF_PERIOD;
try {
$rate = CashFlowValidations::validateRate($rate);
@@ -77,9 +77,9 @@ class Periodic
): string|float {
$rate = Functions::flattenSingleValue($rate);
$numberOfPeriods = Functions::flattenSingleValue($numberOfPeriods);
- $payment = ($payment === null) ? 0.0 : Functions::flattenSingleValue($payment);
- $futureValue = ($futureValue === null) ? 0.0 : Functions::flattenSingleValue($futureValue);
- $type = ($type === null) ? FinancialConstants::PAYMENT_END_OF_PERIOD : Functions::flattenSingleValue($type);
+ $payment = Functions::flattenSingleValue($payment) ?? 0.0;
+ $futureValue = Functions::flattenSingleValue($futureValue) ?? 0.0;
+ $type = Functions::flattenSingleValue($type) ?? FinancialConstants::PAYMENT_END_OF_PERIOD;
try {
$rate = CashFlowValidations::validateRate($rate);
@@ -122,8 +122,8 @@ class Periodic
$rate = Functions::flattenSingleValue($rate);
$payment = Functions::flattenSingleValue($payment);
$presentValue = Functions::flattenSingleValue($presentValue);
- $futureValue = ($futureValue === null) ? 0.0 : Functions::flattenSingleValue($futureValue);
- $type = ($type === null) ? FinancialConstants::PAYMENT_END_OF_PERIOD : Functions::flattenSingleValue($type);
+ $futureValue = Functions::flattenSingleValue($futureValue) ?? 0.0;
+ $type = Functions::flattenSingleValue($type) ?? FinancialConstants::PAYMENT_END_OF_PERIOD;
try {
$rate = CashFlowValidations::validateRate($rate);
@@ -150,7 +150,7 @@ class Periodic
float $presentValue,
int $type
): float {
- if ($rate !== null && $rate != 0) {
+ if ($rate != 0) {
return -$presentValue
* (1 + $rate) ** $numberOfPeriods - $payment * (1 + $rate * $type) * ((1 + $rate) ** $numberOfPeriods - 1)
/ $rate;
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/CashFlow/Constant/Periodic/Cumulative.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/CashFlow/Constant/Periodic/Cumulative.php
index 94359090..a19a61b2 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/CashFlow/Constant/Periodic/Cumulative.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/CashFlow/Constant/Periodic/Cumulative.php
@@ -41,7 +41,7 @@ class Cumulative
$presentValue = Functions::flattenSingleValue($presentValue);
$start = Functions::flattenSingleValue($start);
$end = Functions::flattenSingleValue($end);
- $type = ($type === null) ? FinancialConstants::PAYMENT_END_OF_PERIOD : Functions::flattenSingleValue($type);
+ $type = Functions::flattenSingleValue($type) ?? FinancialConstants::PAYMENT_END_OF_PERIOD;
try {
$rate = CashFlowValidations::validateRate($rate);
@@ -104,7 +104,7 @@ class Cumulative
$presentValue = Functions::flattenSingleValue($presentValue);
$start = Functions::flattenSingleValue($start);
$end = Functions::flattenSingleValue($end);
- $type = ($type === null) ? FinancialConstants::PAYMENT_END_OF_PERIOD : Functions::flattenSingleValue($type);
+ $type = Functions::flattenSingleValue($type) ?? FinancialConstants::PAYMENT_END_OF_PERIOD;
try {
$rate = CashFlowValidations::validateRate($rate);
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/CashFlow/Constant/Periodic/Interest.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/CashFlow/Constant/Periodic/Interest.php
index ad68ec13..68ac57b8 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/CashFlow/Constant/Periodic/Interest.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/CashFlow/Constant/Periodic/Interest.php
@@ -43,7 +43,7 @@ class Interest
$numberOfPeriods = Functions::flattenSingleValue($numberOfPeriods);
$presentValue = Functions::flattenSingleValue($presentValue);
$futureValue = ($futureValue === null) ? 0.0 : Functions::flattenSingleValue($futureValue);
- $type = ($type === null) ? FinancialConstants::PAYMENT_END_OF_PERIOD : Functions::flattenSingleValue($type);
+ $type = Functions::flattenSingleValue($type) ?? FinancialConstants::PAYMENT_END_OF_PERIOD;
try {
$interestRate = CashFlowValidations::validateRate($interestRate);
@@ -160,9 +160,9 @@ class Interest
$numberOfPeriods = Functions::flattenSingleValue($numberOfPeriods);
$payment = Functions::flattenSingleValue($payment);
$presentValue = Functions::flattenSingleValue($presentValue);
- $futureValue = ($futureValue === null) ? 0.0 : Functions::flattenSingleValue($futureValue);
- $type = ($type === null) ? FinancialConstants::PAYMENT_END_OF_PERIOD : Functions::flattenSingleValue($type);
- $guess = ($guess === null) ? 0.1 : Functions::flattenSingleValue($guess);
+ $futureValue = Functions::flattenSingleValue($futureValue) ?? 0.0;
+ $type = Functions::flattenSingleValue($type) ?? FinancialConstants::PAYMENT_END_OF_PERIOD;
+ $guess = Functions::flattenSingleValue($guess) ?? 0.1;
try {
$numberOfPeriods = CashFlowValidations::validateFloat($numberOfPeriods);
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/CashFlow/Constant/Periodic/Payments.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/CashFlow/Constant/Periodic/Payments.php
index 41e88f9a..d16be446 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/CashFlow/Constant/Periodic/Payments.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/CashFlow/Constant/Periodic/Payments.php
@@ -27,14 +27,14 @@ class Payments
mixed $interestRate,
mixed $numberOfPeriods,
mixed $presentValue,
- mixed $futureValue = 0,
+ mixed $futureValue = 0.0,
mixed $type = FinancialConstants::PAYMENT_END_OF_PERIOD
): string|float {
$interestRate = Functions::flattenSingleValue($interestRate);
$numberOfPeriods = Functions::flattenSingleValue($numberOfPeriods);
$presentValue = Functions::flattenSingleValue($presentValue);
- $futureValue = ($futureValue === null) ? 0.0 : Functions::flattenSingleValue($futureValue);
- $type = ($type === null) ? FinancialConstants::PAYMENT_END_OF_PERIOD : Functions::flattenSingleValue($type);
+ $futureValue = Functions::flattenSingleValue($futureValue) ?? 0.0;
+ $type = Functions::flattenSingleValue($type) ?? FinancialConstants::PAYMENT_END_OF_PERIOD;
try {
$interestRate = CashFlowValidations::validateRate($interestRate);
@@ -83,7 +83,7 @@ class Payments
$numberOfPeriods = Functions::flattenSingleValue($numberOfPeriods);
$presentValue = Functions::flattenSingleValue($presentValue);
$futureValue = ($futureValue === null) ? 0.0 : Functions::flattenSingleValue($futureValue);
- $type = ($type === null) ? FinancialConstants::PAYMENT_END_OF_PERIOD : Functions::flattenSingleValue($type);
+ $type = Functions::flattenSingleValue($type) ?? FinancialConstants::PAYMENT_END_OF_PERIOD;
try {
$interestRate = CashFlowValidations::validateRate($interestRate);
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/CashFlow/Single.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/CashFlow/Single.php
index 6f60a2af..ba3da6bb 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/CashFlow/Single.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/CashFlow/Single.php
@@ -77,13 +77,13 @@ class Single
*
* Calculates the interest rate required for an investment to grow to a specified future value .
*
- * @param array|float $periods The number of periods over which the investment is made
- * @param array|float $presentValue Present Value
- * @param array|float $futureValue Future Value
+ * @param mixed $periods The number of periods over which the investment is made, expect array|float
+ * @param mixed $presentValue Present Value, expect array|float
+ * @param mixed $futureValue Future Value, expect array|float
*
* @return float|string Result, or a string containing an error
*/
- public static function interestRate(array|float $periods = 0.0, array|float $presentValue = 0.0, array|float $futureValue = 0.0): string|float
+ public static function interestRate(mixed $periods = 0.0, mixed $presentValue = 0.0, mixed $futureValue = 0.0): string|float
{
$periods = Functions::flattenSingleValue($periods);
$presentValue = Functions::flattenSingleValue($presentValue);
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/CashFlow/Variable/NonPeriodic.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/CashFlow/Variable/NonPeriodic.php
index 8c6f615b..53668d13 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/CashFlow/Variable/NonPeriodic.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/CashFlow/Variable/NonPeriodic.php
@@ -6,6 +6,7 @@ use PhpOffice\PhpSpreadsheet\Calculation\DateTimeExcel;
use PhpOffice\PhpSpreadsheet\Calculation\Exception;
use PhpOffice\PhpSpreadsheet\Calculation\Functions;
use PhpOffice\PhpSpreadsheet\Calculation\Information\ExcelError;
+use PhpOffice\PhpSpreadsheet\Shared\StringHelper;
class NonPeriodic
{
@@ -23,16 +24,17 @@ class NonPeriodic
* Excel Function:
* =XIRR(values,dates,guess)
*
- * @param float[] $values A series of cash flow payments
+ * @param array $values A series of cash flow payments, expecting float[]
* The series of values must contain at least one positive value & one negative value
- * @param mixed[] $dates A series of payment dates
+ * @param array $dates A series of payment dates
* The first payment date indicates the beginning of the schedule of payments
* All other dates must be later than this date, but they may occur in any order
* @param mixed $guess An optional guess at the expected answer
*/
- public static function rate(array $values, array $dates, mixed $guess = self::DEFAULT_GUESS): float|string
+ public static function rate(mixed $values, $dates, mixed $guess = self::DEFAULT_GUESS): float|string
{
$rslt = self::xirrPart1($values, $dates);
+ /** @var array $dates */
if ($rslt !== '') {
return $rslt;
}
@@ -91,6 +93,7 @@ class NonPeriodic
$x2 += 0.5;
}
if ($found) {
+ /** @var array $dates */
return self::xirrBisection($values, $dates, $x1, $x2);
}
@@ -106,18 +109,18 @@ class NonPeriodic
* Excel Function:
* =XNPV(rate,values,dates)
*
- * @param array|float $rate the discount rate to apply to the cash flows
- * @param float[] $values A series of cash flows that corresponds to a schedule of payments in dates.
+ * @param mixed $rate the discount rate to apply to the cash flows, expect array|float
+ * @param array $values A series of cash flows that corresponds to a schedule of payments in dates, expecting float[].
* The first payment is optional and corresponds to a cost or payment that occurs
* at the beginning of the investment.
* If the first value is a cost or payment, it must be a negative value.
* All succeeding payments are discounted based on a 365-day year.
* The series of values must contain at least one positive value and one negative value.
- * @param mixed[] $dates A schedule of payment dates that corresponds to the cash flow payments.
+ * @param mixed $dates A schedule of payment dates that corresponds to the cash flow payments, expecting mixed[].
* The first payment date indicates the beginning of the schedule of payments.
* All other dates must be later than this date, but they may occur in any order.
*/
- public static function presentValue(array|float $rate, array $values, array $dates): float|string
+ public static function presentValue(mixed $rate, mixed $values, mixed $dates): float|string
{
return self::xnpvOrdered($rate, $values, $dates, true);
}
@@ -127,9 +130,12 @@ class NonPeriodic
return $neg && $pos;
}
+ /** @param array $values */
private static function xirrPart1(mixed &$values, mixed &$dates): string
{
- $values = Functions::flattenArray($values);
+ /** @var array */
+ $temp = Functions::flattenArray($values);
+ $values = $temp;
$dates = Functions::flattenArray($dates);
$valuesIsArray = count($values) > 1;
$datesIsArray = count($dates) > 1;
@@ -152,6 +158,7 @@ class NonPeriodic
return self::xirrPart2($values);
}
+ /** @param array $values */
private static function xirrPart2(array &$values): string
{
$valCount = count($values);
@@ -159,7 +166,7 @@ class NonPeriodic
$foundneg = false;
for ($i = 0; $i < $valCount; ++$i) {
$fld = $values[$i];
- if (!is_numeric($fld)) {
+ if (!is_numeric($fld)) { //* @phpstan-ignore-line
return ExcelError::VALUE();
} elseif ($fld > 0) {
$foundpos = true;
@@ -174,6 +181,10 @@ class NonPeriodic
return '';
}
+ /**
+ * @param array $values
+ * @param array $dates
+ */
private static function xirrPart3(array $values, array $dates, float $x1, float $x2): float|string
{
$f = self::xnpvOrdered($x1, $values, $dates, false);
@@ -203,6 +214,10 @@ class NonPeriodic
return $rslt;
}
+ /**
+ * @param array $values
+ * @param array $dates
+ */
private static function xirrBisection(array $values, array $dates, float $x1, float $x2): string|float
{
$rslt = ExcelError::NAN();
@@ -239,9 +254,13 @@ class NonPeriodic
return $rslt;
}
+ /** @param array $values> */
private static function xnpvOrdered(mixed $rate, mixed $values, mixed $dates, bool $ordered = true, bool $capAtNegative1 = false): float|string
{
$rate = Functions::flattenSingleValue($rate);
+ if (!is_numeric($rate)) {
+ return ExcelError::VALUE();
+ }
$values = Functions::flattenArray($values);
$dates = Functions::flattenArray($dates);
$valCount = count($values);
@@ -273,10 +292,10 @@ class NonPeriodic
$dif = Functions::scalar(DateTimeExcel\Difference::interval($date0, $datei, 'd'));
}
if (!is_numeric($dif)) {
- return $dif;
+ return StringHelper::convertToString($dif);
}
if ($rate <= -1.0) {
- $xnpv += -abs($values[$i]) / (-1 - $rate) ** ($dif / 365);
+ $xnpv += -abs($values[$i] + 0) / (-1 - $rate) ** ($dif / 365);
} else {
$xnpv += $values[$i] / (1 + $rate) ** ($dif / 365);
}
@@ -285,6 +304,10 @@ class NonPeriodic
return is_finite($xnpv) ? $xnpv : ExcelError::VALUE();
}
+ /**
+ * @param mixed[] $values
+ * @param mixed[] $dates
+ */
private static function validateXnpv(mixed $rate, array $values, array $dates): void
{
if (!is_numeric($rate)) {
@@ -294,7 +317,7 @@ class NonPeriodic
if ($valCount != count($dates)) {
throw new Exception(ExcelError::NAN());
}
- if ($valCount > 1 && ((min($values) > 0) || (max($values) < 0))) {
+ if (count($values) > 1 && ((min($values) > 0) || (max($values) < 0))) {
throw new Exception(ExcelError::NAN());
}
}
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/CashFlow/Variable/Periodic.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/CashFlow/Variable/Periodic.php
index 21e537be..c5c3bcb9 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/CashFlow/Variable/Periodic.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/CashFlow/Variable/Periodic.php
@@ -36,6 +36,9 @@ class Periodic
}
$values = Functions::flattenArray($values);
$guess = Functions::flattenSingleValue($guess);
+ if (!is_numeric($guess)) {
+ return ExcelError::VALUE();
+ }
// create an initial range, with a root somewhere between 0 and guess
$x1 = 0.0;
@@ -103,7 +106,9 @@ class Periodic
return ExcelError::DIV0();
}
$values = Functions::flattenArray($values);
+ /** @var float */
$financeRate = Functions::flattenSingleValue($financeRate);
+ /** @var float */
$reinvestmentRate = Functions::flattenSingleValue($reinvestmentRate);
$n = count($values);
@@ -112,6 +117,7 @@ class Periodic
$npvPos = $npvNeg = 0.0;
foreach ($values as $i => $v) {
+ /** @var float $v */
if ($v >= 0) {
$npvPos += $v / $rr ** $i;
} else {
@@ -134,12 +140,13 @@ class Periodic
*
* Returns the Net Present Value of a cash flow series given a discount rate.
*
- * @param array $args
+ * @param array $args
*/
public static function presentValue(mixed $rate, ...$args): int|float
{
$returnValue = 0;
+ /** @var float */
$rate = Functions::flattenSingleValue($rate);
$aArgs = Functions::flattenArray($args);
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Coupons.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Coupons.php
index c2fcab39..e6c9c85e 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Coupons.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Coupons.php
@@ -49,9 +49,7 @@ class Coupons
$settlement = Functions::flattenSingleValue($settlement);
$maturity = Functions::flattenSingleValue($maturity);
$frequency = Functions::flattenSingleValue($frequency);
- $basis = ($basis === null)
- ? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD
- : Functions::flattenSingleValue($basis);
+ $basis = Functions::flattenSingleValue($basis) ?? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD;
try {
$settlement = FinancialValidations::validateSettlementDate($settlement);
@@ -110,9 +108,7 @@ class Coupons
$settlement = Functions::flattenSingleValue($settlement);
$maturity = Functions::flattenSingleValue($maturity);
$frequency = Functions::flattenSingleValue($frequency);
- $basis = ($basis === null)
- ? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD
- : Functions::flattenSingleValue($basis);
+ $basis = Functions::flattenSingleValue($basis) ?? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD;
try {
$settlement = FinancialValidations::validateSettlementDate($settlement);
@@ -179,9 +175,7 @@ class Coupons
$settlement = Functions::flattenSingleValue($settlement);
$maturity = Functions::flattenSingleValue($maturity);
$frequency = Functions::flattenSingleValue($frequency);
- $basis = ($basis === null)
- ? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD
- : Functions::flattenSingleValue($basis);
+ $basis = Functions::flattenSingleValue($basis) ?? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD;
try {
$settlement = FinancialValidations::validateSettlementDate($settlement);
@@ -244,9 +238,7 @@ class Coupons
$settlement = Functions::flattenSingleValue($settlement);
$maturity = Functions::flattenSingleValue($maturity);
$frequency = Functions::flattenSingleValue($frequency);
- $basis = ($basis === null)
- ? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD
- : Functions::flattenSingleValue($basis);
+ $basis = Functions::flattenSingleValue($basis) ?? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD;
try {
$settlement = FinancialValidations::validateSettlementDate($settlement);
@@ -296,9 +288,7 @@ class Coupons
$settlement = Functions::flattenSingleValue($settlement);
$maturity = Functions::flattenSingleValue($maturity);
$frequency = Functions::flattenSingleValue($frequency);
- $basis = ($basis === null)
- ? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD
- : Functions::flattenSingleValue($basis);
+ $basis = Functions::flattenSingleValue($basis) ?? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD;
try {
$settlement = FinancialValidations::validateSettlementDate($settlement);
@@ -355,9 +345,7 @@ class Coupons
$settlement = Functions::flattenSingleValue($settlement);
$maturity = Functions::flattenSingleValue($maturity);
$frequency = Functions::flattenSingleValue($frequency);
- $basis = ($basis === null)
- ? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD
- : Functions::flattenSingleValue($basis);
+ $basis = Functions::flattenSingleValue($basis) ?? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD;
try {
$settlement = FinancialValidations::validateSettlementDate($settlement);
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Dollar.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Dollar.php
index b0581f66..3fc64b6e 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Dollar.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Dollar.php
@@ -25,7 +25,7 @@ class Dollar
* If you omit precision, it is assumed to be 2
* Or can be an array of precision values
*
- * @return array|string If an array of values is passed for either of the arguments, then the returned result
+ * @return array|string If an array of values is passed for either of the arguments, then the returned result
* will also be an array with matching dimensions
*/
public static function format(mixed $number, mixed $precision = 2)
@@ -47,6 +47,8 @@ class Dollar
* Or can be an array of values
* @param mixed $fraction Fraction
* Or can be an array of values
+ *
+ * @return array|float|string
*/
public static function decimal(mixed $fractionalDollar = null, mixed $fraction = 0): array|string|float
{
@@ -93,6 +95,8 @@ class Dollar
* Or can be an array of values
* @param mixed $fraction Fraction
* Or can be an array of values
+ *
+ * @return array|float|string
*/
public static function fractional(mixed $decimalDollar = null, mixed $fraction = 0): array|string|float
{
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Helpers.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Helpers.php
index aa287129..c983ecf4 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Helpers.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Helpers.php
@@ -14,7 +14,7 @@ class Helpers
*
* Returns the number of days in a specified year, as defined by the "basis" value
*
- * @param int|string $year The year against which we're testing
+ * @param mixed $year The year against which we're testing, expect int|string
* @param int|string $basis The type of day count:
* 0 or omitted US (NASD) 360
* 1 Actual (365 or 366 in a leap year)
@@ -24,8 +24,11 @@ class Helpers
*
* @return int|string Result, or a string containing an error
*/
- public static function daysPerYear($year, $basis = 0): string|int
+ public static function daysPerYear(mixed $year, $basis = 0): string|int
{
+ if (!is_int($year) && !is_string($year)) {
+ return ExcelError::VALUE();
+ }
if (!is_numeric($basis)) {
return ExcelError::NAN();
}
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Securities/AccruedInterest.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Securities/AccruedInterest.php
index eb57abfc..5f4379d4 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Securities/AccruedInterest.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Securities/AccruedInterest.php
@@ -6,6 +6,7 @@ use PhpOffice\PhpSpreadsheet\Calculation\DateTimeExcel\YearFrac;
use PhpOffice\PhpSpreadsheet\Calculation\Exception;
use PhpOffice\PhpSpreadsheet\Calculation\Financial\Constants as FinancialConstants;
use PhpOffice\PhpSpreadsheet\Calculation\Functions;
+use PhpOffice\PhpSpreadsheet\Shared\StringHelper;
class AccruedInterest
{
@@ -59,12 +60,8 @@ class AccruedInterest
$settlement = Functions::flattenSingleValue($settlement);
$rate = Functions::flattenSingleValue($rate);
$parValue = ($parValue === null) ? 1000 : Functions::flattenSingleValue($parValue);
- $frequency = ($frequency === null)
- ? FinancialConstants::FREQUENCY_ANNUAL
- : Functions::flattenSingleValue($frequency);
- $basis = ($basis === null)
- ? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD
- : Functions::flattenSingleValue($basis);
+ $frequency = Functions::flattenSingleValue($frequency) ?? FinancialConstants::FREQUENCY_ANNUAL;
+ $basis = Functions::flattenSingleValue($basis) ?? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD;
try {
$issue = SecurityValidations::validateIssueDate($issue);
@@ -81,12 +78,12 @@ class AccruedInterest
$daysBetweenIssueAndSettlement = Functions::scalar(YearFrac::fraction($issue, $settlement, $basis));
if (!is_numeric($daysBetweenIssueAndSettlement)) {
// return date error
- return $daysBetweenIssueAndSettlement;
+ return StringHelper::convertToString($daysBetweenIssueAndSettlement);
}
$daysBetweenFirstInterestAndSettlement = Functions::scalar(YearFrac::fraction($firstInterest, $settlement, $basis));
if (!is_numeric($daysBetweenFirstInterestAndSettlement)) {
// return date error
- return $daysBetweenFirstInterestAndSettlement;
+ return StringHelper::convertToString($daysBetweenFirstInterestAndSettlement);
}
return $parValue * $rate * $daysBetweenIssueAndSettlement;
@@ -125,9 +122,7 @@ class AccruedInterest
$settlement = Functions::flattenSingleValue($settlement);
$rate = Functions::flattenSingleValue($rate);
$parValue = ($parValue === null) ? 1000 : Functions::flattenSingleValue($parValue);
- $basis = ($basis === null)
- ? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD
- : Functions::flattenSingleValue($basis);
+ $basis = Functions::flattenSingleValue($basis) ?? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD;
try {
$issue = SecurityValidations::validateIssueDate($issue);
@@ -143,7 +138,7 @@ class AccruedInterest
$daysBetweenIssueAndSettlement = Functions::scalar(YearFrac::fraction($issue, $settlement, $basis));
if (!is_numeric($daysBetweenIssueAndSettlement)) {
// return date error
- return $daysBetweenIssueAndSettlement;
+ return StringHelper::convertToString($daysBetweenIssueAndSettlement);
}
return $parValue * $rate * $daysBetweenIssueAndSettlement;
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Securities/Price.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Securities/Price.php
index b07b2c9f..12c26c2d 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Securities/Price.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Securities/Price.php
@@ -9,6 +9,7 @@ use PhpOffice\PhpSpreadsheet\Calculation\Financial\Coupons;
use PhpOffice\PhpSpreadsheet\Calculation\Financial\Helpers;
use PhpOffice\PhpSpreadsheet\Calculation\Functions;
use PhpOffice\PhpSpreadsheet\Calculation\Information\ExcelError;
+use PhpOffice\PhpSpreadsheet\Shared\StringHelper;
class Price
{
@@ -52,9 +53,7 @@ class Price
$yield = Functions::flattenSingleValue($yield);
$redemption = Functions::flattenSingleValue($redemption);
$frequency = Functions::flattenSingleValue($frequency);
- $basis = ($basis === null)
- ? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD
- : Functions::flattenSingleValue($basis);
+ $basis = Functions::flattenSingleValue($basis) ?? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD;
try {
$settlement = SecurityValidations::validateSettlementDate($settlement);
@@ -119,9 +118,7 @@ class Price
$maturity = Functions::flattenSingleValue($maturity);
$discount = Functions::flattenSingleValue($discount);
$redemption = Functions::flattenSingleValue($redemption);
- $basis = ($basis === null)
- ? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD
- : Functions::flattenSingleValue($basis);
+ $basis = Functions::flattenSingleValue($basis) ?? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD;
try {
$settlement = SecurityValidations::validateSettlementDate($settlement);
@@ -137,7 +134,7 @@ class Price
$daysBetweenSettlementAndMaturity = Functions::scalar(DateTimeExcel\YearFrac::fraction($settlement, $maturity, $basis));
if (!is_numeric($daysBetweenSettlementAndMaturity)) {
// return date error
- return $daysBetweenSettlementAndMaturity;
+ return StringHelper::convertToString($daysBetweenSettlementAndMaturity);
}
return $redemption * (1 - $discount * $daysBetweenSettlementAndMaturity);
@@ -178,9 +175,7 @@ class Price
$issue = Functions::flattenSingleValue($issue);
$rate = Functions::flattenSingleValue($rate);
$yield = Functions::flattenSingleValue($yield);
- $basis = ($basis === null)
- ? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD
- : Functions::flattenSingleValue($basis);
+ $basis = Functions::flattenSingleValue($basis) ?? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD;
try {
$settlement = SecurityValidations::validateSettlementDate($settlement);
@@ -201,19 +196,19 @@ class Price
$daysBetweenIssueAndSettlement = Functions::scalar(DateTimeExcel\YearFrac::fraction($issue, $settlement, $basis));
if (!is_numeric($daysBetweenIssueAndSettlement)) {
// return date error
- return $daysBetweenIssueAndSettlement;
+ return StringHelper::convertToString($daysBetweenIssueAndSettlement);
}
$daysBetweenIssueAndSettlement *= $daysPerYear;
$daysBetweenIssueAndMaturity = Functions::scalar(DateTimeExcel\YearFrac::fraction($issue, $maturity, $basis));
if (!is_numeric($daysBetweenIssueAndMaturity)) {
// return date error
- return $daysBetweenIssueAndMaturity;
+ return StringHelper::convertToString($daysBetweenIssueAndMaturity);
}
$daysBetweenIssueAndMaturity *= $daysPerYear;
$daysBetweenSettlementAndMaturity = Functions::scalar(DateTimeExcel\YearFrac::fraction($settlement, $maturity, $basis));
if (!is_numeric($daysBetweenSettlementAndMaturity)) {
// return date error
- return $daysBetweenSettlementAndMaturity;
+ return StringHelper::convertToString($daysBetweenSettlementAndMaturity);
}
$daysBetweenSettlementAndMaturity *= $daysPerYear;
@@ -254,9 +249,7 @@ class Price
$maturity = Functions::flattenSingleValue($maturity);
$investment = Functions::flattenSingleValue($investment);
$discount = Functions::flattenSingleValue($discount);
- $basis = ($basis === null)
- ? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD
- : Functions::flattenSingleValue($basis);
+ $basis = Functions::flattenSingleValue($basis) ?? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD;
try {
$settlement = SecurityValidations::validateSettlementDate($settlement);
@@ -275,7 +268,7 @@ class Price
$daysBetweenSettlementAndMaturity = DateTimeExcel\YearFrac::fraction($settlement, $maturity, $basis);
if (!is_numeric($daysBetweenSettlementAndMaturity)) {
// return date error
- return Functions::scalar($daysBetweenSettlementAndMaturity);
+ return StringHelper::convertToString(Functions::scalar($daysBetweenSettlementAndMaturity));
}
return $investment / (1 - ($discount * $daysBetweenSettlementAndMaturity));
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Securities/Rates.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Securities/Rates.php
index 2989a29b..4a120e2f 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Securities/Rates.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Securities/Rates.php
@@ -7,6 +7,7 @@ use PhpOffice\PhpSpreadsheet\Calculation\Exception;
use PhpOffice\PhpSpreadsheet\Calculation\Financial\Constants as FinancialConstants;
use PhpOffice\PhpSpreadsheet\Calculation\Functions;
use PhpOffice\PhpSpreadsheet\Calculation\Information\ExcelError;
+use PhpOffice\PhpSpreadsheet\Shared\StringHelper;
class Rates
{
@@ -43,9 +44,7 @@ class Rates
$maturity = Functions::flattenSingleValue($maturity);
$price = Functions::flattenSingleValue($price);
$redemption = Functions::flattenSingleValue($redemption);
- $basis = ($basis === null)
- ? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD
- : Functions::flattenSingleValue($basis);
+ $basis = Functions::flattenSingleValue($basis) ?? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD;
try {
$settlement = SecurityValidations::validateSettlementDate($settlement);
@@ -65,7 +64,7 @@ class Rates
$daysBetweenSettlementAndMaturity = Functions::scalar(DateTimeExcel\YearFrac::fraction($settlement, $maturity, $basis));
if (!is_numeric($daysBetweenSettlementAndMaturity)) {
// return date error
- return $daysBetweenSettlementAndMaturity;
+ return StringHelper::convertToString($daysBetweenSettlementAndMaturity);
}
return (1 - $price / $redemption) / $daysBetweenSettlementAndMaturity;
@@ -104,9 +103,7 @@ class Rates
$maturity = Functions::flattenSingleValue($maturity);
$investment = Functions::flattenSingleValue($investment);
$redemption = Functions::flattenSingleValue($redemption);
- $basis = ($basis === null)
- ? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD
- : Functions::flattenSingleValue($basis);
+ $basis = Functions::flattenSingleValue($basis) ?? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD;
try {
$settlement = SecurityValidations::validateSettlementDate($settlement);
@@ -126,7 +123,7 @@ class Rates
$daysBetweenSettlementAndMaturity = Functions::scalar(DateTimeExcel\YearFrac::fraction($settlement, $maturity, $basis));
if (!is_numeric($daysBetweenSettlementAndMaturity)) {
// return date error
- return $daysBetweenSettlementAndMaturity;
+ return StringHelper::convertToString($daysBetweenSettlementAndMaturity);
}
return (($redemption / $investment) - 1) / ($daysBetweenSettlementAndMaturity);
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Securities/Yields.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Securities/Yields.php
index a4c5a48f..7ccb1325 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Securities/Yields.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Financial/Securities/Yields.php
@@ -7,6 +7,7 @@ use PhpOffice\PhpSpreadsheet\Calculation\Exception;
use PhpOffice\PhpSpreadsheet\Calculation\Financial\Constants as FinancialConstants;
use PhpOffice\PhpSpreadsheet\Calculation\Financial\Helpers;
use PhpOffice\PhpSpreadsheet\Calculation\Functions;
+use PhpOffice\PhpSpreadsheet\Shared\StringHelper;
class Yields
{
@@ -42,9 +43,7 @@ class Yields
$maturity = Functions::flattenSingleValue($maturity);
$price = Functions::flattenSingleValue($price);
$redemption = Functions::flattenSingleValue($redemption);
- $basis = ($basis === null)
- ? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD
- : Functions::flattenSingleValue($basis);
+ $basis = Functions::flattenSingleValue($basis) ?? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD;
try {
$settlement = SecurityValidations::validateSettlementDate($settlement);
@@ -64,7 +63,7 @@ class Yields
$daysBetweenSettlementAndMaturity = Functions::scalar(DateTimeExcel\YearFrac::fraction($settlement, $maturity, $basis));
if (!is_numeric($daysBetweenSettlementAndMaturity)) {
// return date error
- return $daysBetweenSettlementAndMaturity;
+ return StringHelper::convertToString($daysBetweenSettlementAndMaturity);
}
$daysBetweenSettlementAndMaturity *= $daysPerYear;
@@ -106,9 +105,7 @@ class Yields
$issue = Functions::flattenSingleValue($issue);
$rate = Functions::flattenSingleValue($rate);
$price = Functions::flattenSingleValue($price);
- $basis = ($basis === null)
- ? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD
- : Functions::flattenSingleValue($basis);
+ $basis = Functions::flattenSingleValue($basis) ?? FinancialConstants::BASIS_DAYS_PER_YEAR_NASD;
try {
$settlement = SecurityValidations::validateSettlementDate($settlement);
@@ -129,19 +126,19 @@ class Yields
$daysBetweenIssueAndSettlement = Functions::scalar(DateTimeExcel\YearFrac::fraction($issue, $settlement, $basis));
if (!is_numeric($daysBetweenIssueAndSettlement)) {
// return date error
- return $daysBetweenIssueAndSettlement;
+ return StringHelper::convertToString($daysBetweenIssueAndSettlement);
}
$daysBetweenIssueAndSettlement *= $daysPerYear;
$daysBetweenIssueAndMaturity = Functions::scalar(DateTimeExcel\YearFrac::fraction($issue, $maturity, $basis));
if (!is_numeric($daysBetweenIssueAndMaturity)) {
// return date error
- return $daysBetweenIssueAndMaturity;
+ return StringHelper::convertToString($daysBetweenIssueAndMaturity);
}
$daysBetweenIssueAndMaturity *= $daysPerYear;
$daysBetweenSettlementAndMaturity = Functions::scalar(DateTimeExcel\YearFrac::fraction($settlement, $maturity, $basis));
if (!is_numeric($daysBetweenSettlementAndMaturity)) {
// return date error
- return $daysBetweenSettlementAndMaturity;
+ return StringHelper::convertToString($daysBetweenSettlementAndMaturity);
}
$daysBetweenSettlementAndMaturity *= $daysPerYear;
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/FormulaParser.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/FormulaParser.php
index 9868b828..f07b5b5d 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/FormulaParser.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/FormulaParser.php
@@ -213,7 +213,7 @@ class FormulaParser
// scientific notation check
if (str_contains(self::OPERATORS_SN, $this->formula[$index])) {
if (strlen($value) > 1) {
- if (preg_match('/^[1-9]{1}(\\.\\d+)?E{1}$/', $this->formula[$index]) != 0) {
+ if (preg_match('/^[1-9]{1}(\.\d+)?E{1}$/', $this->formula[$index]) != 0) {
$value .= $this->formula[$index];
++$index;
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Functions.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Functions.php
index 77f8317a..16f6a3d0 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Functions.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Functions.php
@@ -4,6 +4,7 @@ namespace PhpOffice\PhpSpreadsheet\Calculation;
use PhpOffice\PhpSpreadsheet\Cell\Cell;
use PhpOffice\PhpSpreadsheet\Shared\Date;
+use PhpOffice\PhpSpreadsheet\Shared\StringHelper;
class Functions
{
@@ -130,16 +131,22 @@ class Functions
public static function isMatrixValue(mixed $idx): bool
{
+ $idx = StringHelper::convertToString($idx);
+
return (substr_count($idx, '.') <= 1) || (preg_match('/\.[A-Z]/', $idx) > 0);
}
public static function isValue(mixed $idx): bool
{
+ $idx = StringHelper::convertToString($idx);
+
return substr_count($idx, '.') === 0;
}
public static function isCellValue(mixed $idx): bool
{
+ $idx = StringHelper::convertToString($idx);
+
return substr_count($idx, '.') > 1;
}
@@ -154,7 +161,8 @@ class Functions
$condition = self::operandSpecialHandling($condition);
if (is_bool($condition)) {
return '=' . ($condition ? 'TRUE' : 'FALSE');
- } elseif (!is_numeric($condition)) {
+ }
+ if (!is_numeric($condition)) {
if ($condition !== '""') { // Not an empty string
// Escape any quotes in the string value
$condition = (string) preg_replace('/"/ui', '""', $condition);
@@ -162,27 +170,32 @@ class Functions
$condition = Calculation::wrapResult(strtoupper($condition));
}
- return str_replace('""""', '""', '=' . $condition);
+ return str_replace('""""', '""', '=' . StringHelper::convertToString($condition));
+ }
+ $operator = $operand = '';
+ if (1 === preg_match('/(=|<[>=]?|>=?)(.*)/', $condition, $matches)) {
+ [, $operator, $operand] = $matches;
}
- preg_match('/(=|<[>=]?|>=?)(.*)/', $condition, $matches);
- [, $operator, $operand] = $matches;
- $operand = self::operandSpecialHandling($operand);
+ $operand = (string) self::operandSpecialHandling($operand);
if (is_numeric(trim($operand, '"'))) {
$operand = trim($operand, '"');
} elseif (!is_numeric($operand) && $operand !== 'FALSE' && $operand !== 'TRUE') {
$operand = str_replace('"', '""', $operand);
$operand = Calculation::wrapResult(strtoupper($operand));
+ $operand = StringHelper::convertToString($operand);
}
return str_replace('""""', '""', $operator . $operand);
}
- private static function operandSpecialHandling(mixed $operand): mixed
+ private static function operandSpecialHandling(mixed $operand): bool|float|int|string
{
if (is_numeric($operand) || is_bool($operand)) {
return $operand;
- } elseif (strtoupper($operand) === Calculation::getTRUE() || strtoupper($operand) === Calculation::getFALSE()) {
+ }
+ $operand = StringHelper::convertToString($operand);
+ if (strtoupper($operand) === Calculation::getTRUE() || strtoupper($operand) === Calculation::getFALSE()) {
return strtoupper($operand);
}
@@ -204,7 +217,7 @@ class Functions
*
* @param mixed $array Array to be flattened
*
- * @return array Flattened array
+ * @return array Flattened array
*/
public static function flattenArray(mixed $array): array
{
@@ -228,6 +241,32 @@ class Functions
return $flattened;
}
+ /**
+ * Convert a multi-dimensional array to a simple 1-dimensional array.
+ * Same as above but argument is specified in ... format.
+ *
+ * @param mixed $array Array to be flattened
+ *
+ * @return array Flattened array
+ */
+ public static function flattenArray2(mixed ...$array): array
+ {
+ $flattened = [];
+ $stack = array_values($array);
+
+ while (!empty($stack)) {
+ $value = array_shift($stack);
+
+ if (is_array($value)) {
+ array_unshift($stack, ...array_values($value));
+ } else {
+ $flattened[] = $value;
+ }
+ }
+
+ return $flattened;
+ }
+
public static function scalar(mixed $value): mixed
{
if (!is_array($value)) {
@@ -246,7 +285,7 @@ class Functions
*
* @param array|mixed $array Array to be flattened
*
- * @return array Flattened array
+ * @return array Flattened array
*/
public static function flattenArrayIndexed($array): array
{
@@ -310,7 +349,7 @@ class Functions
public static function trimTrailingRange(string $coordinate): string
{
- return (string) preg_replace('/:[\\w\$]+$/', '', $coordinate);
+ return (string) preg_replace('/:[\w\$]+$/', '', $coordinate);
}
public static function trimSheetFromCellReference(string $coordinate): string
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Information/ErrorValue.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Information/ErrorValue.php
index f3a74627..00a5d657 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Information/ErrorValue.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Information/ErrorValue.php
@@ -15,7 +15,7 @@ class ErrorValue
* @param mixed $value Value to check
* Or can be an array of values
*
- * @return array|bool If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|bool If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function isErr(mixed $value = ''): array|bool
@@ -33,7 +33,7 @@ class ErrorValue
* @param mixed $value Value to check
* Or can be an array of values
*
- * @return array|bool If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|bool If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function isError(mixed $value = '', bool $tryNotImplemented = false): array|bool
@@ -58,7 +58,7 @@ class ErrorValue
* @param mixed $value Value to check
* Or can be an array of values
*
- * @return array|bool If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|bool If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function isNa(mixed $value = ''): array|bool
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Information/ExcelError.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Information/ExcelError.php
index d9aabfd0..94a4a7ce 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Information/ExcelError.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Information/ExcelError.php
@@ -39,6 +39,8 @@ class ExcelError
* ERROR_TYPE.
*
* @param mixed $value Value to check
+ *
+ * @return array|int|string
*/
public static function type(mixed $value = ''): array|int|string
{
@@ -152,4 +154,14 @@ class ExcelError
{
return self::ERROR_CODES['calculation'];
}
+
+ /**
+ * SPILL.
+ *
+ * @return string #SPILL!
+ */
+ public static function SPILL(): string
+ {
+ return self::ERROR_CODES['spill'];
+ }
}
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Information/Value.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Information/Value.php
index c9a7a0af..57a6704d 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Information/Value.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Information/Value.php
@@ -7,7 +7,9 @@ use PhpOffice\PhpSpreadsheet\Calculation\Calculation;
use PhpOffice\PhpSpreadsheet\Calculation\Functions;
use PhpOffice\PhpSpreadsheet\Cell\Cell;
use PhpOffice\PhpSpreadsheet\Cell\Coordinate;
+use PhpOffice\PhpSpreadsheet\Exception as SpreadsheetException;
use PhpOffice\PhpSpreadsheet\NamedRange;
+use PhpOffice\PhpSpreadsheet\Shared\StringHelper;
use PhpOffice\PhpSpreadsheet\Worksheet\Worksheet;
class Value
@@ -20,7 +22,7 @@ class Value
* @param mixed $value Value to check
* Or can be an array of values
*
- * @return array|bool If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|bool If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function isBlank(mixed $value = null): array|bool
@@ -39,13 +41,14 @@ class Value
*/
public static function isRef(mixed $value, ?Cell $cell = null): bool
{
- if ($cell === null || $value === $cell->getCoordinate()) {
+ if ($cell === null) {
return false;
}
+ $value = StringHelper::convertToString($value);
$cellValue = Functions::trimTrailingRange($value);
if (preg_match('/^' . Calculation::CALCULATION_REGEXP_CELLREF . '$/ui', $cellValue) === 1) {
- [$worksheet, $cellValue] = Worksheet::extractSheetTitle($cellValue, true);
+ [$worksheet, $cellValue] = Worksheet::extractSheetTitle($cellValue, true, true);
if (!empty($worksheet) && $cell->getWorksheet()->getParentOrThrow()->getSheetByName($worksheet) === null) {
return false;
}
@@ -68,7 +71,7 @@ class Value
* @param mixed $value Value to check
* Or can be an array of values
*
- * @return array|bool|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|bool|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function isEven(mixed $value = null): array|string|bool
@@ -79,11 +82,12 @@ class Value
if ($value === null) {
return ExcelError::NAME();
- } elseif ((is_bool($value)) || ((is_string($value)) && (!is_numeric($value)))) {
+ }
+ if (!is_numeric($value)) {
return ExcelError::VALUE();
}
- return ((int) fmod($value, 2)) === 0;
+ return ((int) fmod($value + 0, 2)) === 0;
}
/**
@@ -92,7 +96,7 @@ class Value
* @param mixed $value Value to check
* Or can be an array of values
*
- * @return array|bool|string If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|bool|string If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function isOdd(mixed $value = null): array|string|bool
@@ -103,11 +107,12 @@ class Value
if ($value === null) {
return ExcelError::NAME();
- } elseif ((is_bool($value)) || ((is_string($value)) && (!is_numeric($value)))) {
+ }
+ if (!is_numeric($value)) {
return ExcelError::VALUE();
}
- return ((int) fmod($value, 2)) !== 0;
+ return ((int) fmod($value + 0, 2)) !== 0;
}
/**
@@ -116,7 +121,7 @@ class Value
* @param mixed $value Value to check
* Or can be an array of values
*
- * @return array|bool If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|bool If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function isNumber(mixed $value = null): array|bool
@@ -138,7 +143,7 @@ class Value
* @param mixed $value Value to check
* Or can be an array of values
*
- * @return array|bool If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|bool If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function isLogical(mixed $value = null): array|bool
@@ -156,7 +161,7 @@ class Value
* @param mixed $value Value to check
* Or can be an array of values
*
- * @return array|bool If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|bool If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function isText(mixed $value = null): array|bool
@@ -174,7 +179,7 @@ class Value
* @param mixed $value Value to check
* Or can be an array of values
*
- * @return array|bool If an array of numbers is passed as an argument, then the returned result will also be an array
+ * @return array|bool If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
public static function isNonText(mixed $value = null): array|bool
@@ -191,14 +196,17 @@ class Value
*
* @param mixed $cellReference The cell to check
* @param ?Cell $cell The current cell (containing this formula)
+ *
+ * @return array|bool|string
*/
public static function isFormula(mixed $cellReference = '', ?Cell $cell = null): array|bool|string
{
if ($cell === null) {
return ExcelError::REF();
}
+ $cellReference = StringHelper::convertToString($cellReference);
- $fullCellReference = Functions::expandDefinedName((string) $cellReference, $cell);
+ $fullCellReference = Functions::expandDefinedName($cellReference, $cell);
if (str_contains($cellReference, '!')) {
$cellReference = Functions::trimSheetFromCellReference($cellReference);
@@ -210,16 +218,24 @@ class Value
$fullCellReference = Functions::trimTrailingRange($fullCellReference);
- preg_match('/^' . Calculation::CALCULATION_REGEXP_CELLREF . '$/i', $fullCellReference, $matches);
-
- $fullCellReference = $matches[6] . $matches[7];
- $worksheetName = str_replace("''", "'", trim($matches[2], "'"));
+ $worksheetName = '';
+ if (1 == preg_match('/^' . Calculation::CALCULATION_REGEXP_CELLREF . '$/i', $fullCellReference, $matches)) {
+ $fullCellReference = $matches[6] . $matches[7];
+ $worksheetName = str_replace("''", "'", trim($matches[2], "'"));
+ }
$worksheet = (!empty($worksheetName))
? $cell->getWorksheet()->getParentOrThrow()->getSheetByName($worksheetName)
: $cell->getWorksheet();
+ if ($worksheet === null) {
+ return ExcelError::REF();
+ }
- return ($worksheet !== null) ? $worksheet->getCell($fullCellReference)->isFormula() : ExcelError::REF();
+ try {
+ return $worksheet->getCell($fullCellReference)->isFormula();
+ } catch (SpreadsheetException) {
+ return true;
+ }
}
/**
@@ -243,21 +259,14 @@ class Value
while (is_array($value)) {
$value = array_shift($value);
}
-
- switch (gettype($value)) {
- case 'double':
- case 'float':
- case 'integer':
- return $value;
- case 'boolean':
- return (int) $value;
- case 'string':
- // Errors
- if (($value !== '') && ($value[0] == '#')) {
- return $value;
- }
-
- break;
+ if (is_float($value) || is_int($value)) {
+ return $value;
+ }
+ if (is_bool($value)) {
+ return (int) $value;
+ }
+ if (is_string($value) && substr($value, 0, 1) === '#') {
+ return $value;
}
return 0;
@@ -281,7 +290,7 @@ class Value
public static function type($value = null): int
{
$value = Functions::flattenArrayIndexed($value);
- if (is_array($value) && (count($value) > 1)) {
+ if (count($value) > 1) {
end($value);
$a = key($value);
// Range of cells is an error
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Internal/MakeMatrix.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Internal/MakeMatrix.php
index 22c95e86..f289c3f0 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Internal/MakeMatrix.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Internal/MakeMatrix.php
@@ -4,7 +4,11 @@ namespace PhpOffice\PhpSpreadsheet\Calculation\Internal;
class MakeMatrix
{
- /** @param array $args */
+ /**
+ * @param mixed[] $args
+ *
+ * @return mixed[]
+ */
public static function make(...$args): array
{
return $args;
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Internal/WildcardMatch.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Internal/WildcardMatch.php
index 371ad8b3..8282ea28 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Internal/WildcardMatch.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Internal/WildcardMatch.php
@@ -6,10 +6,10 @@ class WildcardMatch
{
private const SEARCH_SET = [
'~~', // convert double tilde to unprintable value
- '~\\*', // convert tilde backslash asterisk to [*] (matches literal asterisk in regexp)
- '\\*', // convert backslash asterisk to .* (matches string of any length in regexp)
- '~\\?', // convert tilde backslash question to [?] (matches literal question mark in regexp)
- '\\?', // convert backslash question to . (matches one character in regexp)
+ '~\*', // convert tilde backslash asterisk to [*] (matches literal asterisk in regexp)
+ '\*', // convert backslash asterisk to .* (matches string of any length in regexp)
+ '~\?', // convert tilde backslash question to [?] (matches literal question mark in regexp)
+ '\?', // convert backslash question to . (matches one character in regexp)
"\x1c", // convert original double tilde to single tilde
];
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Logical/Operations.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Logical/Operations.php
index 16bb5dd4..f0a54765 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Logical/Operations.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/Logical/Operations.php
@@ -106,7 +106,7 @@ class Operations
* @param mixed $logical A value or expression that can be evaluated to TRUE or FALSE
* Or can be an array of values
*
- * @return array|bool|string the boolean inverse of the argument
+ * @return array|bool|string the boolean inverse of the argument
* If an array of values is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
@@ -130,6 +130,10 @@ class Operations
return !$logical;
}
+ /**
+ * @param mixed[] $args
+ * @param callable(int, int): bool $func
+ */
private static function countTrueValues(array $args, callable $func): bool|string
{
$trueValueCount = 0;
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Address.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Address.php
index 0a5347b8..a17e5549 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Address.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Address.php
@@ -6,6 +6,7 @@ use PhpOffice\PhpSpreadsheet\Calculation\ArrayEnabled;
use PhpOffice\PhpSpreadsheet\Calculation\Information\ExcelError;
use PhpOffice\PhpSpreadsheet\Cell\AddressHelper;
use PhpOffice\PhpSpreadsheet\Cell\Coordinate;
+use PhpOffice\PhpSpreadsheet\Shared\StringHelper;
class Address
{
@@ -44,7 +45,7 @@ class Address
* @param mixed $sheetName Optional Name of worksheet to use
* Or can be an array of values
*
- * @return array|string If an array of values is passed as the $testValue argument, then the returned result will also be
+ * @return mixed[]|string If an array of values is passed as the $testValue argument, then the returned result will also be
* an array with the same dimensions
*/
public static function cell(mixed $row, mixed $column, mixed $relativity = 1, mixed $referenceStyle = true, mixed $sheetName = ''): array|string
@@ -63,14 +64,16 @@ class Address
);
}
- $relativity = $relativity ?? 1;
+ $relativity = ($relativity === null) ? 1 : (int) StringHelper::convertToString($relativity);
$referenceStyle = $referenceStyle ?? true;
+ $row = (int) StringHelper::convertToString($row);
+ $column = (int) StringHelper::convertToString($column);
if (($row < 1) || ($column < 1)) {
return ExcelError::VALUE();
}
- $sheetName = self::sheetName($sheetName);
+ $sheetName = self::sheetName(StringHelper::convertToString($sheetName));
if (is_int($referenceStyle)) {
$referenceStyle = (bool) $referenceStyle;
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/ExcelMatch.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/ExcelMatch.php
index 43e89c9b..3a1a6c41 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/ExcelMatch.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/ExcelMatch.php
@@ -30,7 +30,7 @@ class ExcelMatch
* @param mixed $matchType The number -1, 0, or 1. -1 means above, 0 means exact match, 1 means below.
* If match_type is 1 or -1, the list has to be ordered.
*
- * @return array|float|int|string The relative position of the found item
+ * @return array|float|int|string The relative position of the found item
*/
public static function MATCH(mixed $lookupValue, mixed $lookupArray, mixed $matchType = self::MATCHTYPE_LARGEST_VALUE): array|string|int|float
{
@@ -70,13 +70,14 @@ class ExcelMatch
};
if ($valueKey !== null) {
- return ++$valueKey;
+ return ++$valueKey; //* @phpstan-ignore-line
}
// Unsuccessful in finding a match, return #N/A error value
return ExcelError::NA();
}
+ /** @param mixed[] $lookupArray */
private static function matchFirstValue(array $lookupArray, mixed $lookupValue): int|string|null
{
if (is_string($lookupValue)) {
@@ -113,6 +114,10 @@ class ExcelMatch
return null;
}
+ /**
+ * @param mixed[] $lookupArray
+ * @param mixed[] $keySet
+ */
private static function matchLargestValue(array $lookupArray, mixed $lookupValue, array $keySet): mixed
{
if (is_string($lookupValue)) {
@@ -147,6 +152,7 @@ class ExcelMatch
return null;
}
+ /** @param mixed[] $lookupArray */
private static function matchSmallestValue(array $lookupArray, mixed $lookupValue): int|string|null
{
$valueKey = null;
@@ -215,6 +221,7 @@ class ExcelMatch
return self::MATCHTYPE_FIRST_VALUE;
}
+ /** @param mixed[] $lookupArray */
private static function validateLookupArray(array $lookupArray): void
{
// Lookup_array should not be empty
@@ -224,6 +231,11 @@ class ExcelMatch
}
}
+ /**
+ * @param mixed[] $lookupArray
+ *
+ * @return mixed[]
+ */
private static function prepareLookupArray(array $lookupArray, mixed $matchType): array
{
// Lookup_array should contain only number, text, or logical values, or empty (null) cells
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Filter.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Filter.php
index e3b6cbe5..486194e4 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Filter.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Filter.php
@@ -6,8 +6,12 @@ use PhpOffice\PhpSpreadsheet\Calculation\Information\ExcelError;
class Filter
{
- public static function filter(array $lookupArray, mixed $matchArray, mixed $ifEmpty = null): mixed
+ public static function filter(mixed $lookupArray, mixed $matchArray, mixed $ifEmpty = null): mixed
{
+ if (!is_array($lookupArray)) {
+ return ExcelError::VALUE();
+ }
+ /** @var mixed[] $lookupArray */
if (!is_array($matchArray)) {
return ExcelError::VALUE();
}
@@ -21,10 +25,17 @@ class Filter
if (empty($result)) {
return $ifEmpty ?? ExcelError::CALC();
}
+ /** @var callable(mixed): mixed */
+ $func = 'array_values';
- return array_values(array_map('array_values', $result));
+ return array_values(array_map($func, $result));
}
+ /**
+ * @param mixed[] $sortArray
+ *
+ * @return mixed[]
+ */
private static function enumerateArrayKeys(array $sortArray): array
{
array_walk(
@@ -39,17 +50,29 @@ class Filter
return array_values($sortArray);
}
+ /**
+ * @param mixed[] $lookupArray
+ * @param mixed[] $matchArray
+ *
+ * @return mixed[]
+ */
private static function filterByRow(array $lookupArray, array $matchArray): array
{
- $matchArray = array_values(array_column($matchArray, 0));
+ $matchArray = array_values(array_column($matchArray, 0)); // @phpstan-ignore-line
return array_filter(
array_values($lookupArray),
- fn ($index): bool => (bool) $matchArray[$index],
+ fn ($index): bool => (bool) ($matchArray[$index] ?? null),
ARRAY_FILTER_USE_KEY
);
}
+ /**
+ * @param mixed[] $lookupArray
+ * @param mixed[] $matchArray
+ *
+ * @return mixed[]
+ */
private static function filterByColumn(array $lookupArray, array $matchArray): array
{
$lookupArray = Matrix::transpose($lookupArray);
@@ -57,7 +80,7 @@ class Filter
if (count($matchArray) === 1) {
$matchArray = array_pop($matchArray);
}
-
+ /** @var mixed[] $matchArray */
array_walk(
$matchArray,
function (&$value): void {
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Formula.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Formula.php
index 5c7f4051..f4982a0e 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Formula.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Formula.php
@@ -5,6 +5,7 @@ namespace PhpOffice\PhpSpreadsheet\Calculation\LookupRef;
use PhpOffice\PhpSpreadsheet\Calculation\Calculation;
use PhpOffice\PhpSpreadsheet\Calculation\Information\ExcelError;
use PhpOffice\PhpSpreadsheet\Cell\Cell;
+use PhpOffice\PhpSpreadsheet\Shared\StringHelper;
class Formula
{
@@ -20,13 +21,15 @@ class Formula
return ExcelError::REF();
}
- preg_match('/^' . Calculation::CALCULATION_REGEXP_CELLREF . '$/i', $cellReference, $matches);
-
- $cellReference = $matches[6] . $matches[7];
- $worksheetName = trim($matches[3], "'");
- $worksheet = (!empty($worksheetName))
- ? $cell->getWorksheet()->getParentOrThrow()->getSheetByName($worksheetName)
- : $cell->getWorksheet();
+ $worksheet = null;
+ $cellReference = StringHelper::convertToString($cellReference);
+ if (1 === preg_match('/^' . Calculation::CALCULATION_REGEXP_CELLREF . '$/i', $cellReference, $matches)) {
+ $cellReference = $matches[6] . $matches[7];
+ $worksheetName = trim($matches[3], "'");
+ $worksheet = (!empty($worksheetName))
+ ? $cell->getWorksheet()->getParentOrThrow()->getSheetByName($worksheetName)
+ : $cell->getWorksheet();
+ }
if (
$worksheet === null
@@ -36,6 +39,6 @@ class Formula
return ExcelError::NA();
}
- return $worksheet->getCell($cellReference)->getValue();
+ return $worksheet->getCell($cellReference)->getValueString();
}
}
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/HLookup.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/HLookup.php
index fd83700b..30c021b6 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/HLookup.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/HLookup.php
@@ -18,14 +18,14 @@ class HLookup extends LookupBase
* in the same column based on the index_number.
*
* @param mixed $lookupValue The value that you want to match in lookup_array
- * @param mixed $lookupArray The range of cells being searched
- * @param mixed $indexNumber The row number in table_array from which the matching value must be returned.
+ * @param mixed[][] $lookupArray The range of cells being searched
+ * @param array|float|int|string $indexNumber The row number in table_array from which the matching value must be returned.
* The first row is 1.
* @param mixed $notExactMatch determines if you are looking for an exact match based on lookup_value
*
* @return mixed The value of the found cell
*/
- public static function lookup(mixed $lookupValue, mixed $lookupArray, mixed $indexNumber, mixed $notExactMatch = true): mixed
+ public static function lookup(mixed $lookupValue, $lookupArray, $indexNumber, mixed $notExactMatch = true): mixed
{
if (is_array($lookupValue) || is_array($indexNumber)) {
return self::evaluateArrayArgumentsIgnore([self::class, __FUNCTION__], 1, $lookupValue, $lookupArray, $indexNumber, $notExactMatch);
@@ -49,6 +49,7 @@ class HLookup extends LookupBase
$firstkey = $f[0] - 1;
$returnColumn = $firstkey + $indexNumber;
+ /** @var mixed[][] $lookupArray */
$firstColumn = array_shift($f) ?? 1;
$rowNumber = self::hLookupSearch($lookupValue, $lookupArray, $firstColumn, $notExactMatch);
@@ -62,17 +63,20 @@ class HLookup extends LookupBase
/**
* @param mixed $lookupValue The value that you want to match in lookup_array
+ * @param mixed[][] $lookupArray
* @param int|string $column
*/
private static function hLookupSearch(mixed $lookupValue, array $lookupArray, $column, bool $notExactMatch): ?int
{
- $lookupLower = StringHelper::strToLower((string) $lookupValue);
+ $lookupLower = StringHelper::strToLower(StringHelper::convertToString($lookupValue));
$rowNumber = null;
foreach ($lookupArray[$column] as $rowKey => $rowData) {
// break if we have passed possible keys
+ /** @var string $rowKey */
$bothNumeric = is_numeric($lookupValue) && is_numeric($rowData);
$bothNotNumeric = !is_numeric($lookupValue) && !is_numeric($rowData);
+ /** @var scalar $rowData */
$cellDataLower = StringHelper::strToLower((string) $rowData);
if (
@@ -96,6 +100,11 @@ class HLookup extends LookupBase
return $rowNumber;
}
+ /**
+ * @param mixed[] $lookupArray
+ *
+ * @return mixed[]
+ */
private static function convertLiteralArray(array $lookupArray): array
{
if (array_key_exists(0, $lookupArray)) {
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Helpers.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Helpers.php
index 191144bf..5f94ffaa 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Helpers.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Helpers.php
@@ -35,6 +35,7 @@ class Helpers
}
}
+ /** @return array{string, ?string, string} */
public static function extractCellAddresses(string $cellAddress, bool $a1, Worksheet $sheet, string $sheetName = '', ?int $baseRow = null, ?int $baseCol = null): array
{
$cellAddress1 = $cellAddress;
@@ -57,12 +58,12 @@ class Helpers
return [$cellAddress1, $cellAddress2, $cellAddress];
}
+ /** @return array{string, ?Worksheet, string} */
public static function extractWorksheet(string $cellAddress, Cell $cell): array
{
$sheetName = '';
if (str_contains($cellAddress, '!')) {
- [$sheetName, $cellAddress] = Worksheet::extractSheetTitle($cellAddress, true);
- $sheetName = trim($sheetName, "'");
+ [$sheetName, $cellAddress] = Worksheet::extractSheetTitle($cellAddress, true, true);
}
$worksheet = ($sheetName !== '')
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Hyperlink.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Hyperlink.php
index 455442a8..e7752aa4 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Hyperlink.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Hyperlink.php
@@ -5,6 +5,7 @@ namespace PhpOffice\PhpSpreadsheet\Calculation\LookupRef;
use PhpOffice\PhpSpreadsheet\Calculation\Functions;
use PhpOffice\PhpSpreadsheet\Calculation\Information\ExcelError;
use PhpOffice\PhpSpreadsheet\Cell\Cell;
+use PhpOffice\PhpSpreadsheet\Shared\StringHelper;
class Hyperlink
{
@@ -22,18 +23,23 @@ class Hyperlink
*/
public static function set(mixed $linkURL = '', mixed $displayName = null, ?Cell $cell = null): string
{
- $linkURL = ($linkURL === null) ? '' : Functions::flattenSingleValue($linkURL);
+ $linkURL = ($linkURL === null) ? '' : StringHelper::convertToString(Functions::flattenSingleValue($linkURL));
$displayName = ($displayName === null) ? '' : Functions::flattenSingleValue($displayName);
if ((!is_object($cell)) || (trim($linkURL) == '')) {
return ExcelError::REF();
}
- if ((is_object($displayName)) || trim($displayName) == '') {
+ if (is_object($displayName)) {
+ $displayName = $linkURL;
+ }
+ $displayName = StringHelper::convertToString($displayName);
+ if (trim($displayName) === '') {
$displayName = $linkURL;
}
- $cell->getHyperlink()->setUrl($linkURL);
+ $cell->getHyperlink()
+ ->setUrl($linkURL);
$cell->getHyperlink()->setTooltip($displayName);
return $displayName;
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Indirect.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Indirect.php
index d53900d4..756478c4 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Indirect.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Indirect.php
@@ -34,6 +34,8 @@ class Indirect
/**
* Convert cellAddress to string, verify not null string.
+ *
+ * @param null|mixed[]|string $cellAddress
*/
private static function validateAddress(array|string|null $cellAddress): string
{
@@ -54,12 +56,12 @@ class Indirect
* Excel Function:
* =INDIRECT(cellAddress, bool) where the bool argument is optional
*
- * @param array|string $cellAddress $cellAddress The cell address of the current cell (containing this formula)
+ * @param mixed[]|string $cellAddress $cellAddress The cell address of the current cell (containing this formula)
* @param mixed $a1fmt Expect bool Helpers::CELLADDRESS_USE_A1 or CELLADDRESS_USE_R1C1,
* but can be provided as numeric which is cast to bool
* @param Cell $cell The current cell (containing this formula)
*
- * @return array|string An array containing a cell or range of cells, or a string on error
+ * @return mixed[]|string An array containing a cell or range of cells, or a string on error
*/
public static function INDIRECT($cellAddress, mixed $a1fmt, Cell $cell): string|array
{
@@ -99,13 +101,13 @@ class Indirect
/**
* Extract range values.
*
- * @return array Array of values in range if range contains more than one element.
+ * @return mixed[] Array of values in range if range contains more than one element.
* Otherwise, a single value is returned.
*/
private static function extractRequiredCells(?Worksheet $worksheet, string $cellAddress): array
{
return Calculation::getInstance($worksheet !== null ? $worksheet->getParent() : null)
- ->extractCellRange($cellAddress, $worksheet, false);
+ ->extractCellRange($cellAddress, $worksheet, false, createCell: true);
}
private static function handleRowColumnRanges(?Worksheet $worksheet, string $start, string $end): string
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Lookup.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Lookup.php
index b1876207..9d3eea4f 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Lookup.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Lookup.php
@@ -28,6 +28,7 @@ class Lookup
if (!is_array($lookupVector)) {
return ExcelError::NA();
}
+ /** @var mixed[][] $lookupVector */
$hasResultVector = isset($resultVector);
$lookupRows = self::rowCount($lookupVector);
$lookupColumns = self::columnCount($lookupVector);
@@ -35,16 +36,19 @@ class Lookup
if (($lookupRows === 1 && $lookupColumns > 1) || (!$hasResultVector && $lookupRows === 2 && $lookupColumns !== 2)) {
$lookupVector = Matrix::transpose($lookupVector);
$lookupRows = self::rowCount($lookupVector);
+ /** @var mixed[][] $lookupVector */
$lookupColumns = self::columnCount($lookupVector);
}
- $resultVector = self::verifyResultVector($resultVector ?? $lookupVector);
+ $resultVector = self::verifyResultVector($resultVector ?? $lookupVector); //* @phpstan-ignore-line
if ($lookupRows === 2 && !$hasResultVector) {
$resultVector = array_pop($lookupVector);
$lookupVector = array_shift($lookupVector);
}
+ /** @var mixed[] $lookupVector */
+ /** @var mixed[] $resultVector */
if ($lookupColumns !== 2) {
$lookupVector = self::verifyLookupValues($lookupVector, $resultVector);
}
@@ -52,6 +56,12 @@ class Lookup
return VLookup::lookup($lookupValue, $lookupVector, 2);
}
+ /**
+ * @param mixed[] $lookupVector
+ * @param mixed[] $resultVector
+ *
+ * @return mixed[]
+ */
private static function verifyLookupValues(array $lookupVector, array $resultVector): array
{
foreach ($lookupVector as &$value) {
@@ -77,6 +87,11 @@ class Lookup
return $lookupVector;
}
+ /**
+ * @param mixed[][] $resultVector
+ *
+ * @return mixed[]
+ */
private static function verifyResultVector(array $resultVector): array
{
$resultRows = self::rowCount($resultVector);
@@ -90,11 +105,13 @@ class Lookup
return $resultVector;
}
+ /** @param mixed[] $dataArray */
private static function rowCount(array $dataArray): int
{
return count($dataArray);
}
+ /** @param mixed[][] $dataArray */
private static function columnCount(array $dataArray): int
{
$rowKeys = array_keys($dataArray);
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/LookupBase.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/LookupBase.php
index 7d21cce0..39c498f1 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/LookupBase.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/LookupBase.php
@@ -7,15 +7,18 @@ use PhpOffice\PhpSpreadsheet\Calculation\Information\ExcelError;
abstract class LookupBase
{
- protected static function validateLookupArray(mixed $lookup_array): void
+ protected static function validateLookupArray(mixed $lookupArray): void
{
- if (!is_array($lookup_array)) {
+ if (!is_array($lookupArray)) {
throw new Exception(ExcelError::REF());
}
}
- /** @param float|int|string $index_number */
- protected static function validateIndexLookup(array $lookup_array, $index_number): int
+ /**
+ * @param mixed[] $lookupArray
+ * @param float|int|string $index_number number >= 1
+ */
+ protected static function validateIndexLookup(array $lookupArray, $index_number): int
{
// index_number must be a number greater than or equal to 1.
// Excel results are inconsistent when index is non-numeric.
@@ -30,8 +33,8 @@ abstract class LookupBase
throw new Exception(ExcelError::VALUE());
}
- // index_number must be less than or equal to the number of columns in lookup_array
- if (empty($lookup_array)) {
+ // index_number must be less than or equal to the number of columns in lookupArray
+ if (empty($lookupArray)) {
throw new Exception(ExcelError::REF());
}
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/LookupRefValidations.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/LookupRefValidations.php
index 74c313cc..620705e9 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/LookupRefValidations.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/LookupRefValidations.php
@@ -11,7 +11,7 @@ class LookupRefValidations
public static function validateInt(mixed $value): int
{
if (!is_numeric($value)) {
- if (ErrorValue::isError($value)) {
+ if (is_string($value) && ErrorValue::isError($value)) {
throw new Exception($value);
}
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Matrix.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Matrix.php
index 228b4644..20cdd069 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Matrix.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Matrix.php
@@ -12,6 +12,8 @@ class Matrix
/**
* Helper function; NOT an implementation of any Excel Function.
+ *
+ * @param mixed[] $values
*/
public static function isColumnVector(array $values): bool
{
@@ -20,6 +22,8 @@ class Matrix
/**
* Helper function; NOT an implementation of any Excel Function.
+ *
+ * @param mixed[] $values
*/
public static function isRowVector(array $values): bool
{
@@ -30,7 +34,9 @@ class Matrix
/**
* TRANSPOSE.
*
- * @param array|mixed $matrixData A matrix of values
+ * @param mixed $matrixData A matrix of values
+ *
+ * @return mixed[]
*/
public static function transpose($matrixData): array
{
@@ -38,8 +44,12 @@ class Matrix
if (!is_array($matrixData)) {
$matrixData = [[$matrixData]];
}
+ if (!is_array(end($matrixData))) {
+ $matrixData = [$matrixData];
+ }
$column = 0;
+ /** @var mixed[][] $matrixData */
foreach ($matrixData as $matrixRow) {
$row = 0;
foreach ($matrixRow as $matrixCell) {
@@ -82,6 +92,15 @@ class Matrix
$rowNum = $rowNum ?? 0;
$columnNum = $columnNum ?? 0;
+ if (is_scalar($matrix)) {
+ if ($rowNum === 0 || $rowNum === 1) {
+ if ($columnNum === 0 || $columnNum === 1) {
+ if ($columnNum === 1 || $rowNum === 1) {
+ return $matrix;
+ }
+ }
+ }
+ }
try {
$rowNum = LookupRefValidations::validatePositiveInt($rowNum);
@@ -106,7 +125,7 @@ class Matrix
}
$rowKeys = array_keys($matrix);
- $columnKeys = @array_keys($matrix[$rowKeys[0]]);
+ $columnKeys = @array_keys($matrix[$rowKeys[0]]); //* @phpstan-ignore-line
if ($columnNum > count($columnKeys)) {
return ExcelError::REF();
@@ -124,10 +143,15 @@ class Matrix
);
}
$rowNum = $rowKeys[--$rowNum];
+ /** @var mixed[][] $matrix */
return $matrix[$rowNum][$columnNum];
}
+ /**
+ * @param mixed[] $matrix
+ * @param mixed[] $rowKeys
+ */
private static function extractRowValue(array $matrix, array $rowKeys, int $rowNum): mixed
{
if ($rowNum === 0) {
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Offset.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Offset.php
index 260ccc3a..6b072623 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Offset.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Offset.php
@@ -7,6 +7,7 @@ use PhpOffice\PhpSpreadsheet\Calculation\Functions;
use PhpOffice\PhpSpreadsheet\Calculation\Information\ExcelError;
use PhpOffice\PhpSpreadsheet\Cell\Cell;
use PhpOffice\PhpSpreadsheet\Cell\Coordinate;
+use PhpOffice\PhpSpreadsheet\Worksheet\Validations;
use PhpOffice\PhpSpreadsheet\Worksheet\Worksheet;
class Offset
@@ -24,28 +25,32 @@ class Offset
* @param null|string $cellAddress The reference from which you want to base the offset.
* Reference must refer to a cell or range of adjacent cells;
* otherwise, OFFSET returns the #VALUE! error value.
- * @param mixed $rows The number of rows, up or down, that you want the upper-left cell to refer to.
+ * @param int $rows The number of rows, up or down, that you want the upper-left cell to refer to.
* Using 5 as the rows argument specifies that the upper-left cell in the
* reference is five rows below reference. Rows can be positive (which means
* below the starting reference) or negative (which means above the starting
* reference).
- * @param mixed $columns The number of columns, to the left or right, that you want the upper-left cell
+ * @param int $columns The number of columns, to the left or right, that you want the upper-left cell
* of the result to refer to. Using 5 as the cols argument specifies that the
* upper-left cell in the reference is five columns to the right of reference.
* Cols can be positive (which means to the right of the starting reference)
* or negative (which means to the left of the starting reference).
- * @param mixed $height The height, in number of rows, that you want the returned reference to be.
+ * @param ?int $height The height, in number of rows, that you want the returned reference to be.
* Height must be a positive number.
- * @param mixed $width The width, in number of columns, that you want the returned reference to be.
+ * @param ?int $width The width, in number of columns, that you want the returned reference to be.
* Width must be a positive number.
*
- * @return array|string An array containing a cell or range of cells, or a string on error
+ * @return array|string An array containing a cell or range of cells, or a string on error
*/
- public static function OFFSET(?string $cellAddress = null, mixed $rows = 0, mixed $columns = 0, mixed $height = null, mixed $width = null, ?Cell $cell = null): string|array
+ public static function OFFSET(?string $cellAddress = null, $rows = 0, $columns = 0, $height = null, $width = null, ?Cell $cell = null): string|array
{
+ /** @var int */
$rows = Functions::flattenSingleValue($rows);
+ /** @var int */
$columns = Functions::flattenSingleValue($columns);
+ /** @var int */
$height = Functions::flattenSingleValue($height);
+ /** @var int */
$width = Functions::flattenSingleValue($width);
if ($cellAddress === null || $cellAddress === '') {
@@ -55,6 +60,10 @@ class Offset
if (!is_object($cell)) {
return ExcelError::REF();
}
+ $sheet = $cell->getParent()?->getParent(); // worksheet
+ if ($sheet !== null) {
+ $cellAddress = Validations::definedNameToCoordinate($cellAddress, $sheet);
+ }
[$cellAddress, $worksheet] = self::extractWorksheet($cellAddress, $cell);
@@ -62,12 +71,11 @@ class Offset
if (strpos($cellAddress, ':')) {
[$startCell, $endCell] = explode(':', $cellAddress);
}
- [$startCellColumn, $startCellRow] = Coordinate::coordinateFromString($startCell);
- [$endCellColumn, $endCellRow] = Coordinate::coordinateFromString($endCell);
+ [$startCellColumn, $startCellRow] = Coordinate::indexesFromString($startCell);
+ [, $endCellRow, $endCellColumn] = Coordinate::indexesFromString($endCell);
$startCellRow += $rows;
- $startCellColumn = Coordinate::columnIndexFromString($startCellColumn) - 1;
- $startCellColumn += $columns;
+ $startCellColumn += $columns - 1;
if (($startCellRow <= 0) || ($startCellColumn < 0)) {
return ExcelError::REF();
@@ -91,20 +99,21 @@ class Offset
return self::extractRequiredCells($worksheet, $cellAddress);
}
+ /** @return mixed[] */
private static function extractRequiredCells(?Worksheet $worksheet, string $cellAddress): array
{
return Calculation::getInstance($worksheet !== null ? $worksheet->getParent() : null)
->extractCellRange($cellAddress, $worksheet, false);
}
+ /** @return array{string, ?Worksheet} */
private static function extractWorksheet(?string $cellAddress, Cell $cell): array
{
$cellAddress = self::assessCellAddress($cellAddress ?? '', $cell);
$sheetName = '';
if (str_contains($cellAddress, '!')) {
- [$sheetName, $cellAddress] = Worksheet::extractSheetTitle($cellAddress, true);
- $sheetName = trim($sheetName, "'");
+ [$sheetName, $cellAddress] = Worksheet::extractSheetTitle($cellAddress, true, true);
}
$worksheet = ($sheetName !== '')
@@ -123,7 +132,11 @@ class Offset
return $cellAddress;
}
- private static function adjustEndCellColumnForWidth(string $endCellColumn, mixed $width, int $startCellColumn, mixed $columns): int
+ /**
+ * @param null|object|scalar $width
+ * @param scalar $columns
+ */
+ private static function adjustEndCellColumnForWidth(string $endCellColumn, $width, int $startCellColumn, $columns): int
{
$endCellColumn = Coordinate::columnIndexFromString($endCellColumn) - 1;
if (($width !== null) && (!is_object($width))) {
@@ -135,7 +148,11 @@ class Offset
return $endCellColumn;
}
- private static function adustEndCellRowForHeight(mixed $height, int $startCellRow, mixed $rows, mixed $endCellRow): int
+ /**
+ * @param null|object|scalar $height
+ * @param scalar $rows
+ */
+ private static function adustEndCellRowForHeight($height, int $startCellRow, $rows, int $endCellRow): int
{
if (($height !== null) && (!is_object($height))) {
$endCellRow = $startCellRow + (int) $height - 1;
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/RowColumnInformation.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/RowColumnInformation.php
index ea3ce44c..caf09831 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/RowColumnInformation.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/RowColumnInformation.php
@@ -3,9 +3,11 @@
namespace PhpOffice\PhpSpreadsheet\Calculation\LookupRef;
use PhpOffice\PhpSpreadsheet\Calculation\Calculation;
+use PhpOffice\PhpSpreadsheet\Calculation\Information\ErrorValue;
use PhpOffice\PhpSpreadsheet\Calculation\Information\ExcelError;
use PhpOffice\PhpSpreadsheet\Cell\Cell;
use PhpOffice\PhpSpreadsheet\Cell\Coordinate;
+use PhpOffice\PhpSpreadsheet\Exception as SpreadsheetException;
use PhpOffice\PhpSpreadsheet\Worksheet\Worksheet;
class RowColumnInformation
@@ -13,7 +15,7 @@ class RowColumnInformation
/**
* Test if cellAddress is null or whitespace string.
*
- * @param null|array|string $cellAddress A reference to a range of cells
+ * @param null|mixed[]|string $cellAddress A reference to a range of cells
*/
private static function cellAddressNullOrWhitespace($cellAddress): bool
{
@@ -38,11 +40,11 @@ class RowColumnInformation
* Excel Function:
* =COLUMN([cellAddress])
*
- * @param null|array|string $cellAddress A reference to a range of cells for which you want the column numbers
+ * @param null|mixed[]|string $cellAddress A reference to a range of cells for which you want the column numbers
*
- * @return int|int[]
+ * @return int|int[]|string
*/
- public static function COLUMN($cellAddress = null, ?Cell $cell = null): int|array
+ public static function COLUMN($cellAddress = null, ?Cell $cell = null): int|string|array
{
if (self::cellAddressNullOrWhitespace($cellAddress)) {
return self::cellColumn($cell);
@@ -79,7 +81,11 @@ class RowColumnInformation
$cellAddress = (string) preg_replace('/[^a-z]/i', '', $cellAddress);
- return Coordinate::columnIndexFromString($cellAddress);
+ try {
+ return Coordinate::columnIndexFromString($cellAddress);
+ } catch (SpreadsheetException) {
+ return ExcelError::NAME();
+ }
}
/**
@@ -90,7 +96,7 @@ class RowColumnInformation
* Excel Function:
* =COLUMNS(cellAddress)
*
- * @param null|array|string $cellAddress An array or array formula, or a reference to a range of cells
+ * @param null|mixed[]|string $cellAddress An array or array formula, or a reference to a range of cells
* for which you want the number of columns
*
* @return int|string The number of columns in cellAddress, or a string if arguments are invalid
@@ -100,6 +106,9 @@ class RowColumnInformation
if (self::cellAddressNullOrWhitespace($cellAddress)) {
return 1;
}
+ if (is_string($cellAddress) && ErrorValue::isError($cellAddress)) {
+ return $cellAddress;
+ }
if (!is_array($cellAddress)) {
return ExcelError::VALUE();
}
@@ -115,9 +124,18 @@ class RowColumnInformation
return $columns;
}
- private static function cellRow(?Cell $cell): int
+ private static function cellRow(?Cell $cell): int|string
{
- return ($cell !== null) ? $cell->getRow() : 1;
+ return ($cell !== null) ? self::convert0ToName($cell->getRow()) : 1;
+ }
+
+ private static function convert0ToName(int|string $result): int|string
+ {
+ if (is_int($result) && ($result <= 0 || $result > 1048576)) {
+ return ExcelError::NAME();
+ }
+
+ return $result;
}
/**
@@ -133,11 +151,11 @@ class RowColumnInformation
* Excel Function:
* =ROW([cellAddress])
*
- * @param null|array|string $cellAddress A reference to a range of cells for which you want the row numbers
+ * @param null|mixed[][]|string $cellAddress A reference to a range of cells for which you want the row numbers
*
- * @return int|mixed[]
+ * @return int|mixed[]|string
*/
- public static function ROW($cellAddress = null, ?Cell $cell = null): int|array
+ public static function ROW($cellAddress = null, ?Cell $cell = null): int|string|array
{
if (self::cellAddressNullOrWhitespace($cellAddress)) {
return self::cellRow($cell);
@@ -172,7 +190,7 @@ class RowColumnInformation
}
[$cellAddress] = explode(':', $cellAddress);
- return (int) preg_replace('/\D/', '', $cellAddress);
+ return self::convert0ToName((int) preg_replace('/\D/', '', $cellAddress));
}
/**
@@ -183,7 +201,7 @@ class RowColumnInformation
* Excel Function:
* =ROWS(cellAddress)
*
- * @param null|array|string $cellAddress An array or array formula, or a reference to a range of cells
+ * @param null|mixed[]|string $cellAddress An array or array formula, or a reference to a range of cells
* for which you want the number of rows
*
* @return int|string The number of rows in cellAddress, or a string if arguments are invalid
@@ -193,6 +211,9 @@ class RowColumnInformation
if (self::cellAddressNullOrWhitespace($cellAddress)) {
return 1;
}
+ if (is_string($cellAddress) && ErrorValue::isError($cellAddress)) {
+ return $cellAddress;
+ }
if (!is_array($cellAddress)) {
return ExcelError::VALUE();
}
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Sort.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Sort.php
index 9ad47b4e..352f5c8c 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Sort.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Sort.php
@@ -36,6 +36,7 @@ class Sort extends LookupRefValidations
return $sortArray;
}
+ /** @var mixed[][] */
$sortArray = self::enumerateArrayKeys($sortArray);
$byColumn = (bool) $byColumn;
@@ -43,7 +44,7 @@ class Sort extends LookupRefValidations
try {
// If $sortIndex and $sortOrder are scalars, then convert them into arrays
- if (is_scalar($sortIndex)) {
+ if (!is_array($sortIndex)) {
$sortIndex = [$sortIndex];
$sortOrder = is_scalar($sortOrder) ? [$sortOrder] : $sortOrder;
}
@@ -55,7 +56,11 @@ class Sort extends LookupRefValidations
}
// We want a simple, enumrated array of arrays where we can reference column by its index number.
- $sortArray = array_values(array_map('array_values', $sortArray));
+ /** @var callable(mixed): mixed */
+ $temp = 'array_values';
+ /** @var array $sortOrder */
+ $sortArray = array_values(array_map($temp, $sortArray));
+ /** @var int[] $sortIndex */
return ($byColumn === true)
? self::sortByColumn($sortArray, $sortIndex, $sortOrder)
@@ -104,6 +109,11 @@ class Sort extends LookupRefValidations
return self::processSortBy($sortArray, $sortBy, $sortOrder);
}
+ /**
+ * @param mixed[] $sortArray
+ *
+ * @return mixed[]
+ */
private static function enumerateArrayKeys(array $sortArray): array
{
array_walk(
@@ -133,6 +143,7 @@ class Sort extends LookupRefValidations
$sortOrder = self::validateSortOrder($sortOrder);
}
+ /** @return mixed[] */
private static function validateSortVector(mixed $sortVector, int $sortArraySize): array
{
if (!is_array($sortVector)) {
@@ -158,6 +169,7 @@ class Sort extends LookupRefValidations
return $sortOrder;
}
+ /** @param mixed[] $sortIndex */
private static function validateArrayArgumentsForSort(array &$sortIndex, mixed &$sortOrder, int $sortArraySize): void
{
// It doesn't matter if they're row or column vectors, it works either way
@@ -184,6 +196,11 @@ class Sort extends LookupRefValidations
}
}
+ /**
+ * @param mixed[] $sortVector
+ *
+ * @return mixed[]
+ */
private static function prepareSortVectorValues(array $sortVector): array
{
// Strings should be sorted case-insensitive; with booleans converted to locale-strings
@@ -202,14 +219,19 @@ class Sort extends LookupRefValidations
}
/**
- * @param array[] $sortIndex
+ * @param mixed[] $sortArray
+ * @param mixed[] $sortIndex
* @param int[] $sortOrder
+ *
+ * @return mixed[]
*/
private static function processSortBy(array $sortArray, array $sortIndex, array $sortOrder): array
{
$sortArguments = [];
+ /** @var mixed[] */
$sortData = [];
foreach ($sortIndex as $index => $sortValues) {
+ /** @var mixed[] $sortValues */
$sortData[] = $sortValues;
$sortArguments[] = self::prepareSortVectorValues($sortValues);
$sortArguments[] = $sortOrder[$index] === self::ORDER_ASCENDING ? SORT_ASC : SORT_DESC;
@@ -221,8 +243,11 @@ class Sort extends LookupRefValidations
}
/**
+ * @param mixed[] $sortArray
* @param int[] $sortIndex
* @param int[] $sortOrder
+ *
+ * @return mixed[]
*/
private static function sortByRow(array $sortArray, array $sortIndex, array $sortOrder): array
{
@@ -232,8 +257,11 @@ class Sort extends LookupRefValidations
}
/**
+ * @param mixed[] $sortArray
* @param int[] $sortIndex
* @param int[] $sortOrder
+ *
+ * @return mixed[]
*/
private static function sortByColumn(array $sortArray, array $sortIndex, array $sortOrder): array
{
@@ -244,8 +272,11 @@ class Sort extends LookupRefValidations
}
/**
+ * @param mixed[] $sortArray
* @param int[] $sortIndex
* @param int[] $sortOrder
+ *
+ * @return mixed[]
*/
private static function buildVectorForSort(array $sortArray, array $sortIndex, array $sortOrder): array
{
@@ -263,6 +294,12 @@ class Sort extends LookupRefValidations
return $sortData;
}
+ /**
+ * @param mixed[] $sortData
+ * @param mixed[] $sortArguments
+ *
+ * @return mixed[]
+ */
private static function executeVectorSortQuery(array $sortData, array $sortArguments): array
{
$sortData = Matrix::transpose($sortData);
@@ -287,6 +324,12 @@ class Sort extends LookupRefValidations
return $sortedData;
}
+ /**
+ * @param mixed[] $sortArray
+ * @param mixed[] $sortVector
+ *
+ * @return mixed[]
+ */
private static function sortLookupArrayFromVector(array $sortArray, array $sortVector): array
{
// Building a new array in the correct (sorted) order works; but may be memory heavy for larger arrays
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Unique.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Unique.php
index 220be2d1..40a9df70 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Unique.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/Unique.php
@@ -33,17 +33,31 @@ class Unique
: self::uniqueByRow($lookupVector, $exactlyOnce);
}
+ /** @param mixed[] $lookupVector */
private static function uniqueByRow(array $lookupVector, bool $exactlyOnce): mixed
{
// When not $byColumn, we count whole rows or values, not individual values
// so implode each row into a single string value
array_walk(
$lookupVector,
+ //* @phpstan-ignore-next-line
function (array &$value): void {
- $value = implode(chr(0x00), $value);
+ $valuex = '';
+ $separator = '';
+ $numericIndicator = "\x01";
+ foreach ($value as $cellValue) {
+ /** @var scalar $cellValue */
+ $valuex .= $separator . $cellValue;
+ $separator = "\x00";
+ if (is_int($cellValue) || is_float($cellValue)) {
+ $valuex .= $numericIndicator;
+ }
+ }
+ $value = $valuex;
}
);
+ /** @var string[] $lookupVector */
$result = self::countValuesCaseInsensitive($lookupVector);
if ($exactlyOnce === true) {
@@ -60,15 +74,24 @@ class Unique
array_walk(
$result,
function (string &$value): void {
- $value = explode(chr(0x00), $value);
+ $value = explode("\x00", $value);
+ foreach ($value as &$stringValue) {
+ if (str_ends_with($stringValue, "\x01")) {
+ // x01 should only end a string which is otherwise a float or int,
+ // so phpstan is technically correct but what it fears should not happen.
+ $stringValue = 0 + substr($stringValue, 0, -1); //@phpstan-ignore-line
+ }
+ }
}
);
return (count($result) === 1) ? array_pop($result) : $result;
}
+ /** @param mixed[] $lookupVector */
private static function uniqueByColumn(array $lookupVector, bool $exactlyOnce): mixed
{
+ /** @var string[] */
$flattenedLookupVector = Functions::flattenArray($lookupVector);
if (count($lookupVector, COUNT_RECURSIVE) > count($flattenedLookupVector, COUNT_RECURSIVE) + 1) {
@@ -94,6 +117,11 @@ class Unique
return $result;
}
+ /**
+ * @param string[] $caseSensitiveLookupValues
+ *
+ * @return mixed[]
+ */
private static function countValuesCaseInsensitive(array $caseSensitiveLookupValues): array
{
$caseInsensitiveCounts = array_count_values(
@@ -121,6 +149,11 @@ class Unique
return $caseSensitiveCounts;
}
+ /**
+ * @param mixed[] $values
+ *
+ * @return mixed[]
+ */
private static function exactlyOnceFilter(array $values): array
{
return array_filter(
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/VLookup.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/VLookup.php
index 247074cf..76929c5f 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/VLookup.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/LookupRef/VLookup.php
@@ -17,14 +17,14 @@ class VLookup extends LookupBase
* in the same row based on the index_number.
*
* @param mixed $lookupValue The value that you want to match in lookup_array
- * @param mixed $lookupArray The range of cells being searched
- * @param mixed $indexNumber The column number in table_array from which the matching value must be returned.
+ * @param mixed[] $lookupArray The range of cells being searched
+ * @param array|float|int|string $indexNumber The column number in table_array from which the matching value must be returned.
* The first column is 1.
* @param mixed $notExactMatch determines if you are looking for an exact match based on lookup_value
*
* @return mixed The value of the found cell
*/
- public static function lookup(mixed $lookupValue, mixed $lookupArray, mixed $indexNumber, mixed $notExactMatch = true): mixed
+ public static function lookup(mixed $lookupValue, $lookupArray, mixed $indexNumber, mixed $notExactMatch = true): mixed
{
if (is_array($lookupValue) || is_array($indexNumber)) {
return self::evaluateArrayArgumentsIgnore([self::class, __FUNCTION__], 1, $lookupValue, $lookupArray, $indexNumber, $notExactMatch);
@@ -54,6 +54,7 @@ class VLookup extends LookupBase
uasort($lookupArray, $callable);
}
+ /** @var string[][] $lookupArray */
$rowNumber = self::vLookupSearch($lookupValue, $lookupArray, $firstColumn, $notExactMatch);
if ($rowNumber !== null) {
@@ -64,6 +65,10 @@ class VLookup extends LookupBase
return ExcelError::NA();
}
+ /**
+ * @param scalar[] $a
+ * @param scalar[] $b
+ */
private static function vlookupSort(array $a, array $b): int
{
reset($a);
@@ -80,16 +85,17 @@ class VLookup extends LookupBase
/**
* @param mixed $lookupValue The value that you want to match in lookup_array
+ * @param string[][] $lookupArray
* @param int|string $column
*/
private static function vLookupSearch(mixed $lookupValue, array $lookupArray, $column, bool $notExactMatch): ?int
{
- $lookupLower = StringHelper::strToLower((string) $lookupValue);
+ $lookupLower = StringHelper::strToLower(StringHelper::convertToString($lookupValue));
$rowNumber = null;
foreach ($lookupArray as $rowKey => $rowData) {
- $bothNumeric = is_numeric($lookupValue) && is_numeric($rowData[$column]);
- $bothNotNumeric = !is_numeric($lookupValue) && !is_numeric($rowData[$column]);
+ $bothNumeric = self::numeric($lookupValue) && self::numeric($rowData[$column]);
+ $bothNotNumeric = !self::numeric($lookupValue) && !self::numeric($rowData[$column]);
$cellDataLower = StringHelper::strToLower((string) $rowData[$column]);
// break if we have passed possible keys
@@ -114,4 +120,9 @@ class VLookup extends LookupBase
return $rowNumber;
}
+
+ private static function numeric(mixed $value): bool
+ {
+ return is_int($value) || is_float($value);
+ }
}
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/MathTrig/Absolute.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/MathTrig/Absolute.php
index 03e61399..3b7e5324 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/MathTrig/Absolute.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/MathTrig/Absolute.php
@@ -16,7 +16,7 @@ class Absolute
*
* @param mixed $number Should be numeric, or can be an array of numbers
*
- * @return array|float|int|string rounded number
+ * @return array|float|int|string rounded number
* If an array of numbers is passed as the argument, then the returned result will also be an array
* with the same dimensions
*/
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/MathTrig/Angle.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/MathTrig/Angle.php
index e7de7aac..7c3597d3 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/MathTrig/Angle.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/MathTrig/Angle.php
@@ -16,7 +16,7 @@ class Angle
*
* @param mixed $number Should be numeric, or can be an array of numbers
*
- * @return array|float|string Rounded number
+ * @return array|float|string Rounded number
* If an array of numbers is passed as the argument, then the returned result will also be an array
* with the same dimensions
*/
@@ -42,7 +42,7 @@ class Angle
*
* @param mixed $number Should be numeric, or can be an array of numbers
*
- * @return array|float|string Rounded number
+ * @return array|float|string Rounded number
* If an array of numbers is passed as the argument, then the returned result will also be an array
* with the same dimensions
*/
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/MathTrig/Arabic.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/MathTrig/Arabic.php
index 98c3e3dc..ab58add3 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/MathTrig/Arabic.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/MathTrig/Arabic.php
@@ -22,6 +22,8 @@ class Arabic
/**
* Recursively calculate the arabic value of a roman numeral.
+ *
+ * @param mixed[] $roman
*/
private static function calculateArabic(array $roman, int &$sum = 0, int $subtract = 0): int
{
@@ -55,7 +57,7 @@ class Arabic
*
* @param string|string[] $roman Should be a string, or can be an array of strings
*
- * @return array|int|string the arabic numberal contrived from the roman numeral
+ * @return array|int|string the arabic numberal contrived from the roman numeral
* If an array of numbers is passed as the argument, then the returned result will also be an array
* with the same dimensions
*/
@@ -74,11 +76,14 @@ class Arabic
// Convert the roman numeral to an arabic number
$negativeNumber = $roman[0] === '-';
if ($negativeNumber) {
- $roman = substr($roman, 1);
+ $roman = trim(substr($roman, 1));
+ if ($roman === '') {
+ return ExcelError::NAN();
+ }
}
try {
- $arabic = self::calculateArabic(str_split($roman));
+ $arabic = self::calculateArabic(mb_str_split($roman, 1, 'UTF-8'));
} catch (Exception) {
return ExcelError::VALUE(); // Invalid character detected
}
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/MathTrig/Base.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/MathTrig/Base.php
index 7eda72c3..e9ef1912 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/MathTrig/Base.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/MathTrig/Base.php
@@ -25,7 +25,7 @@ class Base
* @param mixed $minLength expect int or null
* Or can be an array of values
*
- * @return array|string the text representation with the given radix (base)
+ * @return array|string the text representation with the given radix (base)
* If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
diff --git a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/MathTrig/Ceiling.php b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/MathTrig/Ceiling.php
index 365ec2e9..3458ca86 100644
--- a/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/MathTrig/Ceiling.php
+++ b/api/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/Calculation/MathTrig/Ceiling.php
@@ -22,12 +22,12 @@ class Ceiling
* Excel Function:
* CEILING(number[,significance])
*
- * @param array|float $number the number you want the ceiling
+ * @param array|float $number the number you want the ceiling
* Or can be an array of values
- * @param array|float $significance the multiple to which you want to round
+ * @param array|float $significance the multiple to which you want to round
* Or can be an array of values
*
- * @return array|float|string Rounded Number, or a string containing an error
+ * @return array|float|string Rounded Number, or a string containing an error
* If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
@@ -63,14 +63,14 @@ class Ceiling
* Or can be an array of values
* @param mixed $significance Significance
* Or can be an array of values
- * @param array|int $mode direction to round negative numbers
+ * @param array|int $mode direction to round negative numbers
* Or can be an array of values
*
- * @return array|float|string Rounded Number, or a string containing an error
+ * @return array|float|string Rounded Number, or a string containing an error
* If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
- public static function math(mixed $number, mixed $significance = null, $mode = 0): array|string|float
+ public static function math(mixed $number, mixed $significance = null, $mode = 0, bool $checkSigns = false): array|string|float
{
if (is_array($number) || is_array($significance) || is_array($mode)) {
return self::evaluateArrayArguments([self::class, __FUNCTION__], $number, $significance, $mode);
@@ -87,6 +87,11 @@ class Ceiling
if (empty($significance * $number)) {
return 0.0;
}
+ if ($checkSigns) {
+ if (($number > 0 && $significance < 0) || ($number < 0 && $significance > 0)) {
+ return ExcelError::NAN();
+ }
+ }
if (self::ceilingMathTest((float) $significance, (float) $number, (int) $mode)) {
return floor($number / $significance) * $significance;
}
@@ -104,10 +109,10 @@ class Ceiling
*
* @param mixed $number the number you want to round
* Or can be an array of values
- * @param array|float $significance the multiple to which you want to round
+ * @param array|float $significance the multiple to which you want to round
* Or can be an array of values
*
- * @return array|float|string Rounded Number, or a string containing an error
+ * @return array|float|string Rounded Number, or a string containing an error
* If an array of numbers is passed as an argument, then the returned result will also be an array
* with the same dimensions
*/
@@ -132,6 +137,23 @@ class Ceiling
return ceil($result) * $significance * (($significance < 0) ? -1 : 1);
}
+ /**
+ * CEILING.ODS, pseudo-function - CEILING as implemented in ODS.
+ *
+ * ODS Function (theoretical):
+ * CEILING.ODS(number[,significance[,mode]])
+ *
+ * @param mixed $number Number to round
+ * @param mixed $significance Significance
+ * @param array