{
    "id": "event_06b0fefe666cc6c8",
    "timestamp": 1777292681,
    "branch_id": "main",
    "parent_event_id": "event_3c01b1f1ba5c5d08",
    "type": "patch_apply",
    "label": "Fiabilise detection paiement livreur",
    "source": "patch",
    "author": "CNOC",
    "session_id": "43305eb2706f6eee2531a5a173355a18",
    "payload": [
        {
            "path": "caisse-aqp/public/api/_lib/onlinePayments.php",
            "kind": "file",
            "before": {
                "exists": true,
                "kind": "file",
                "size": 8570,
                "sha1": "ac233f109551d1f896d21e4bac85e6146ec3e196",
                "content_b64": "<?php
declare(strict_types=1);

/* doc-project | caisse-aqp/public/api/_lib/onlinePayments.php | Fournit l’accès centralisé aux paiements en ligne validés, leurs libellés et leurs montants agrégés par commande/point de vente afin d’alimenter la validation de règlement et le rattrapage d’impression automatique des tickets clients. | Expose: onlinePaymentsOrderInfoForStore, format_vads_trans_date, get_successful_online_payments_by_order_id, get_successful_online_payment_totals_by_order_id, has_successful_online_payment_for_order | Dépend de: PDO, config.php, pos_pay | Impacte: lecture des paiements, affichage des encaissements, logique de validation de règlement, détection des commandes intégralement payées | Tables: pos_pay(vads_order_id, vads_amount, vads_trans_date, vads_trans_id, vads_result, vads_order_info) */
/**
 * public/api/_lib/onlinePayments.php
 *
 * Récupération des paiements en ligne (pos_pay) groupée.
 * - Filtre "succès" : vads_result = '00'
 * - Filtre projet/PDV : vads_order_info = $orderInfo (dépend du store lan|pel : LANCON / PELISSANNE en bdd)
 */

/**
 * Mapping store -> vads_order_info.
 * Recommandé: définir ONLINE_PAYMENTS_ORDER_INFO_MAP dans config.php, ex:
 *   define('ONLINE_PAYMENTS_ORDER_INFO_MAP', ['lan' => 'LANCON', 'pel' => 'PELISSANNE']);
 *
 * Fallbacks supportés (si besoin) :
 * - define('LAN_ORDER_INFO', '...'), define('PEL_ORDER_INFO', '...')
 * - define('ORDER_INFO_LAN', '...'), define('ORDER_INFO_PEL', '...')
 */
function onlinePaymentsOrderInfoForStore(string $store): ?string {
  $s = strtolower(trim($store));
  if ($s !== 'lan' && $s !== 'pel') return null;

  if (defined('ONLINE_PAYMENTS_ORDER_INFO_MAP')) {
    $m = constant('ONLINE_PAYMENTS_ORDER_INFO_MAP');
    if (is_array($m) && isset($m[$s]) && is_string($m[$s]) && $m[$s] !== '') {
      return (string)$m[$s];
    }
  }

  if ($s === 'lan') {
    if (defined('LAN_ORDER_INFO') && is_string(constant('LAN_ORDER_INFO')) && constant('LAN_ORDER_INFO') !== '') return (string)constant('LAN_ORDER_INFO');
    if (defined('ORDER_INFO_LAN') && is_string(constant('ORDER_INFO_LAN')) && constant('ORDER_INFO_LAN') !== '') return (string)constant('ORDER_INFO_LAN');
  }
  if ($s === 'pel') {
    if (defined('PEL_ORDER_INFO') && is_string(constant('PEL_ORDER_INFO')) && constant('PEL_ORDER_INFO') !== '') return (string)constant('PEL_ORDER_INFO');
    if (defined('ORDER_INFO_PEL') && is_string(constant('ORDER_INFO_PEL')) && constant('ORDER_INFO_PEL') !== '') return (string)constant('ORDER_INFO_PEL');
  }

  // Defaults DB (strict business rule): LANCON / PELISSANNE
  if ($s === 'lan') return 'LANCON';
  if ($s === 'pel') return 'PELISSANNE';
  return null;
}

/**
 * Tente de formater vads_trans_date (souvent au format YYYYMMDDHHMMSS).
 * Retourne une chaîne lisible, sinon la valeur brute.
 */
function format_vads_trans_date($raw): ?string {
  if ($raw === null) return null;
  $s = (string)$raw;

  if (preg_match('/^\d{14}$/', $s)) {
    $dt = DateTime::createFromFormat('YmdHis', $s, new DateTimeZone('Europe/Paris'));
    if ($dt instanceof DateTime) return $dt->format('d/m/Y H:i:s');
  }
  if (preg_match('/^\d{8}$/', $s)) {
    $dt = DateTime::createFromFormat('Ymd', $s, new DateTimeZone('Europe/Paris'));
    if ($dt instanceof DateTime) return $dt->format('d/m/Y');
  }
  return $s;
}

/**
 * Retourne un tableau indexé par orderId :
 * [
 *   123 => [
 *     ['amount_eur' => 12.50, 'label' => '06/02/2026 12:34:56 • TX 000123'],
 *     ...
 *   ],
 * ]
 *
 * Filtre :
 * - Toujours: vads_result = '00'
 * - Toujours: vads_order_info = $orderInfo (anti-collision entre points de vente)
 *
 * $orderInfo peut être :
 * - soit directement la valeur attendue en bdd (LANCON / PELISSANNE),
 * - soit le code store ('lan' / 'pel') auquel cas on mappe automatiquement.
 * Si on ne parvient pas à déterminer vads_order_info, on retourne [] (strict).
 */
function get_successful_online_payments_by_order_id(PDO $pdo, array $orderIds, $orderInfo): array {
  $orderIds = array_values(array_unique(array_map('intval', $orderIds)));
  if (empty($orderIds)) return [];

  $oi = is_string($orderInfo) ? trim($orderInfo) : '';
  // Accept store code ('lan'|'pel') as input and map it to vads_order_info.
  $storeCandidate = strtolower($oi);
  if ($storeCandidate === 'lan' || $storeCandidate === 'pel') {
    $mapped = onlinePaymentsOrderInfoForStore($storeCandidate);
    $oi = is_string($mapped) ? trim($mapped) : '';
  }
  if ($oi === '') return []; // strict anti-collision: do not query without vads_order_info.

  $placeholders = implode(',', array_fill(0, count($orderIds), '?'));
  $sql = "
    SELECT
      vads_order_id,
      vads_amount,
      vads_trans_date,
      vads_trans_id
    FROM pos_pay
    WHERE vads_order_id IN ($placeholders)
      AND vads_result = '00'
      AND vads_order_info = ?
  ";
  $sql .= " ORDER BY vads_order_id ASC, vads_trans_date ASC, vads_trans_id ASC ";

  $stmt = $pdo->prepare($sql);
  $params = $orderIds;
  $params[] = $oi;
  $stmt->execute($params);

  $byOrder = [];
  while ($row = $stmt->fetch(PDO::FETCH_ASSOC)) {
    $oid = isset($row['vads_order_id']) ? (int)$row['vads_order_id'] : 0;
    if ($oid <= 0) continue;

    $amountCent = isset($row['vads_amount']) ? (int)$row['vads_amount'] : 0;
    $amountEur = $amountCent / 100;

    $dateLabel = format_vads_trans_date($row['vads_trans_date'] ?? null);
    $tx = isset($row['vads_trans_id']) && $row['vads_trans_id'] !== '' ? (string)$row['vads_trans_id'] : null;

    $labelParts = [];
    if (!empty($dateLabel)) $labelParts[] = $dateLabel;
    if (!empty($tx)) $labelParts[] = 'TX ' . $tx;
    $label = !empty($labelParts) ? implode(' • ', $labelParts) : 'Paiement';

    if (!isset($byOrder[$oid])) $byOrder[$oid] = [];
    $byOrder[$oid][] = [
      'amount_eur' => $amountEur,
      'label' => $label,
    ];
  }

  return $byOrder;
}

/**
 * Retourne un tableau indexé par orderId avec le total des paiements réussis en centimes :
 * [
 *   123 => 2500,
 *   124 => 1200,
 * ]
 *
 * Même règle stricte que get_successful_online_payments_by_order_id():
 * - vads_result = '00'
 * - vads_order_info = mapping strict du store / orderInfo
 */
function get_successful_online_payment_totals_by_order_id(PDO $pdo, array $orderIds, $orderInfo): array {
  $orderIds = array_values(array_unique(array_map('intval', $orderIds)));
  if (empty($orderIds)) return [];

  $oi = is_string($orderInfo) ? trim($orderInfo) : '';
  $storeCandidate = strtolower($oi);
  if ($storeCandidate === 'lan' || $storeCandidate === 'pel') {
    $mapped = onlinePaymentsOrderInfoForStore($storeCandidate);
    $oi = is_string($mapped) ? trim($mapped) : '';
  }
  if ($oi === '') return [];

  $placeholders = implode(',', array_fill(0, count($orderIds), '?'));
  $sql = "
    SELECT
      vads_order_id,
      COALESCE(SUM(vads_amount), 0) AS total_amount_cent
    FROM pos_pay
    WHERE vads_order_id IN ($placeholders)
      AND vads_result = '00'
      AND vads_order_info = ?
    GROUP BY vads_order_id
  ";

  $stmt = $pdo->prepare($sql);
  $params = $orderIds;
  $params[] = $oi;
  $stmt->execute($params);

  $totals = [];
  while ($row = $stmt->fetch(PDO::FETCH_ASSOC)) {
    $oid = isset($row['vads_order_id']) ? (int)$row['vads_order_id'] : 0;
    if ($oid <= 0) continue;
    $totals[$oid] = isset($row['total_amount_cent']) ? (int)$row['total_amount_cent'] : 0;
  }

  return $totals;
}

/**
 * has_successful_online_payment_for_order(PDO $pdo, string $store, int $orderId) -> bool
 * - Strict business rule for "deposit paid?":
 *   - pos_pay.vads_result = '00'
 *   - pos_pay.vads_order_id = orderId
 *   - pos_pay.vads_order_info = store-mapped value (LANCON / PELISSANNE)
 *
 * Note:
 * - If mapping is missing, we RETURN false (treat as unpaid) and caller should log.
 */
function has_successful_online_payment_for_order(PDO $pdo, string $store, int $orderId): bool {
  $oid = (int)$orderId;
  if ($oid <= 0) return false;

  $orderInfo = onlinePaymentsOrderInfoForStore($store);
  $oi = is_string($orderInfo) ? trim($orderInfo) : '';
  if ($oi === '') {
    // Strict: mapping missing => cannot assert payment belongs to this store.
    return false;
  }

  $stmt = $pdo->prepare("
    SELECT 1
    FROM pos_pay
    WHERE vads_order_id = :oid
      AND vads_result = '00'
      AND vads_order_info = :oi
    LIMIT 1
  ");
  $stmt->bindValue(':oid', $oid, PDO::PARAM_INT);
  $stmt->bindValue(':oi', $oi, PDO::PARAM_STR);
  $stmt->execute();
  $row = $stmt->fetch(PDO::FETCH_ASSOC);
  return is_array($row);
}
"
            },
            "after": {
                "exists": true,
                "kind": "file",
                "size": 11995,
                "sha1": "c884680d2f495d1259c53805135eeeaf1563cd9c",
                "content_b64": "<?php
declare(strict_types=1);

/* doc-project | caisse-aqp/public/api/_lib/onlinePayments.php | Fournit l’accès centralisé aux paiements en ligne validés, leurs libellés, leurs montants agrégés par commande/point de vente et une correspondance robuste multi-identifiants vads_order_id/vads_order_info/vads_order_info2 afin d’alimenter la validation de règlement, le livreur et le rattrapage d’impression automatique des tickets clients. | Expose: onlinePaymentsOrderInfoForStore, format_vads_trans_date, get_successful_online_payments_by_order_id, get_successful_online_payment_totals_by_order_id, get_successful_online_payment_total_centimes_for_order, has_successful_online_payment_for_order | Dépend de: PDO, config.php, pos_pay | Impacte: lecture des paiements, affichage des encaissements, logique de validation de règlement, détection des commandes intégralement payées | Tables: pos_pay(vads_order_id, vads_amount, vads_trans_date, vads_trans_id, vads_result, vads_order_info, vads_order_info2) */
/**
 * public/api/_lib/onlinePayments.php
 *
 * Récupération des paiements en ligne (pos_pay) groupée.
 * - Filtre "succès" : vads_result = '00'
 * - Filtre projet/PDV : vads_order_info = $orderInfo (dépend du store lan|pel : LANCON / PELISSANNE en bdd)
 */

/**
 * Mapping store -> vads_order_info.
 * Recommandé: définir ONLINE_PAYMENTS_ORDER_INFO_MAP dans config.php, ex:
 *   define('ONLINE_PAYMENTS_ORDER_INFO_MAP', ['lan' => 'LANCON', 'pel' => 'PELISSANNE']);
 *
 * Fallbacks supportés (si besoin) :
 * - define('LAN_ORDER_INFO', '...'), define('PEL_ORDER_INFO', '...')
 * - define('ORDER_INFO_LAN', '...'), define('ORDER_INFO_PEL', '...')
 */
function onlinePaymentsOrderInfoForStore(string $store): ?string {
  $s = strtolower(trim($store));
  if ($s !== 'lan' && $s !== 'pel') return null;

  if (defined('ONLINE_PAYMENTS_ORDER_INFO_MAP')) {
    $m = constant('ONLINE_PAYMENTS_ORDER_INFO_MAP');
    if (is_array($m) && isset($m[$s]) && is_string($m[$s]) && $m[$s] !== '') {
      return (string)$m[$s];
    }
  }

  if ($s === 'lan') {
    if (defined('LAN_ORDER_INFO') && is_string(constant('LAN_ORDER_INFO')) && constant('LAN_ORDER_INFO') !== '') return (string)constant('LAN_ORDER_INFO');
    if (defined('ORDER_INFO_LAN') && is_string(constant('ORDER_INFO_LAN')) && constant('ORDER_INFO_LAN') !== '') return (string)constant('ORDER_INFO_LAN');
  }
  if ($s === 'pel') {
    if (defined('PEL_ORDER_INFO') && is_string(constant('PEL_ORDER_INFO')) && constant('PEL_ORDER_INFO') !== '') return (string)constant('PEL_ORDER_INFO');
    if (defined('ORDER_INFO_PEL') && is_string(constant('ORDER_INFO_PEL')) && constant('ORDER_INFO_PEL') !== '') return (string)constant('ORDER_INFO_PEL');
  }

  // Defaults DB (strict business rule): LANCON / PELISSANNE
  if ($s === 'lan') return 'LANCON';
  if ($s === 'pel') return 'PELISSANNE';
  return null;
}

/**
 * Tente de formater vads_trans_date (souvent au format YYYYMMDDHHMMSS).
 * Retourne une chaîne lisible, sinon la valeur brute.
 */
function format_vads_trans_date($raw): ?string {
  if ($raw === null) return null;
  $s = (string)$raw;

  if (preg_match('/^\d{14}$/', $s)) {
    $dt = DateTime::createFromFormat('YmdHis', $s, new DateTimeZone('Europe/Paris'));
    if ($dt instanceof DateTime) return $dt->format('d/m/Y H:i:s');
  }
  if (preg_match('/^\d{8}$/', $s)) {
    $dt = DateTime::createFromFormat('Ymd', $s, new DateTimeZone('Europe/Paris'));
    if ($dt instanceof DateTime) return $dt->format('d/m/Y');
  }
  return $s;
}

/**
 * Normalise les valeurs pouvant identifier une commande dans pos_pay.
 *
 * Exemples de valeurs utiles :
 * - id interne commande
 * - id_date si certains retours historiques l’utilisent
 * - payLink stocké dans vads_order_info2 par le flux Systempay
 */
function onlinePaymentsNormalizeMatchValues(array $values): array {
  $out = [];
  foreach ($values as $value) {
    if ($value === null) continue;
    $s = trim((string)$value);
    if ($s === '') continue;
    $out[$s] = true;
  }
  return array_keys($out);
}

/**
 * Retourne le total payé réussi pour une seule commande, en centimes.
 *
 * La correspondance est volontairement plus robuste que le cas historique
 * vads_order_id + vads_order_info :
 * - cas strict courant : vads_order_id = orderId ET vads_order_info = store
 * - cas retour/IPN où le store est porté dans vads_order_info2
 * - cas où un identifiant commande/payLink est stocké dans vads_order_info ou vads_order_info2
 *
 * Le payLink, quand fourni dans $matchValues, reste le meilleur garde-fou
 * contre les collisions d’id entre pos_commandes et pos_commandes_pel.
 *
 * @return array{total_paid_centimes:int,matched_rows:int}
 */
function get_successful_online_payment_total_centimes_for_order(PDO $pdo, int $orderId, $orderInfo, array $matchValues = []): array {
  if ($orderId <= 0) {
    return [
      'total_paid_centimes' => 0,
      'matched_rows' => 0,
    ];
  }

  $oi = is_string($orderInfo) ? trim($orderInfo) : '';
  $storeCandidate = strtolower($oi);
  if ($storeCandidate === 'lan' || $storeCandidate === 'pel') {
    $mapped = onlinePaymentsOrderInfoForStore($storeCandidate);
    $oi = is_string($mapped) ? trim($mapped) : '';
  }

  $identifiers = onlinePaymentsNormalizeMatchValues(array_merge([(string)$orderId], $matchValues));
  $conditions = [];
  $params = [
    ':order_id' => $orderId,
  ];

  if ($oi !== '') {
    $params[':oi'] = $oi;
    $conditions[] = '(vads_order_id = :order_id AND (vads_order_info = :oi OR vads_order_info2 = :oi))';
  }

  if (!empty($identifiers)) {
    $placeholders = [];
    foreach ($identifiers as $idx => $identifier) {
      $placeholder = ':match_' . $idx;
      $placeholders[] = $placeholder;
      $params[$placeholder] = $identifier;
    }
    $inClause = implode(',', $placeholders);
    $conditions[] = "(vads_order_info IN ({$inClause}) OR vads_order_info2 IN ({$inClause}))";
  }

  if (empty($conditions)) {
    return [
      'total_paid_centimes' => 0,
      'matched_rows' => 0,
    ];
  }

  $sql = "
    SELECT
      COALESCE(SUM(vads_amount), 0) AS total_paid_centimes,
      COUNT(*) AS matched_rows
    FROM pos_pay
    WHERE vads_result = '00'
      AND (" . implode(' OR ', $conditions) . ")
  ";

  $stmt = $pdo->prepare($sql);
  foreach ($params as $name => $value) {
    if ($name === ':order_id') {
      $stmt->bindValue($name, (int)$value, PDO::PARAM_INT);
    } else {
      $stmt->bindValue($name, (string)$value, PDO::PARAM_STR);
    }
  }
  $stmt->execute();
  $row = $stmt->fetch(PDO::FETCH_ASSOC);

  return [
    'total_paid_centimes' => isset($row['total_paid_centimes']) ? (int)$row['total_paid_centimes'] : 0,
    'matched_rows' => isset($row['matched_rows']) ? (int)$row['matched_rows'] : 0,
  ];
}

/**
 * Retourne un tableau indexé par orderId :
 * [
 *   123 => [
 *     ['amount_eur' => 12.50, 'label' => '06/02/2026 12:34:56 • TX 000123'],
 *     ...
 *   ],
 * ]
 *
 * Filtre :
 * - Toujours: vads_result = '00'
 * - Toujours: vads_order_info = $orderInfo (anti-collision entre points de vente)
 *
 * $orderInfo peut être :
 * - soit directement la valeur attendue en bdd (LANCON / PELISSANNE),
 * - soit le code store ('lan' / 'pel') auquel cas on mappe automatiquement.
 * Si on ne parvient pas à déterminer vads_order_info, on retourne [] (strict).
 */
function get_successful_online_payments_by_order_id(PDO $pdo, array $orderIds, $orderInfo): array {
  $orderIds = array_values(array_unique(array_map('intval', $orderIds)));
  if (empty($orderIds)) return [];

  $oi = is_string($orderInfo) ? trim($orderInfo) : '';
  // Accept store code ('lan'|'pel') as input and map it to vads_order_info.
  $storeCandidate = strtolower($oi);
  if ($storeCandidate === 'lan' || $storeCandidate === 'pel') {
    $mapped = onlinePaymentsOrderInfoForStore($storeCandidate);
    $oi = is_string($mapped) ? trim($mapped) : '';
  }
  if ($oi === '') return []; // strict anti-collision: do not query without vads_order_info.

  $placeholders = implode(',', array_fill(0, count($orderIds), '?'));
  $sql = "
    SELECT
      vads_order_id,
      vads_amount,
      vads_trans_date,
      vads_trans_id
    FROM pos_pay
    WHERE vads_order_id IN ($placeholders)
      AND vads_result = '00'
      AND vads_order_info = ?
  ";
  $sql .= " ORDER BY vads_order_id ASC, vads_trans_date ASC, vads_trans_id ASC ";

  $stmt = $pdo->prepare($sql);
  $params = $orderIds;
  $params[] = $oi;
  $stmt->execute($params);

  $byOrder = [];
  while ($row = $stmt->fetch(PDO::FETCH_ASSOC)) {
    $oid = isset($row['vads_order_id']) ? (int)$row['vads_order_id'] : 0;
    if ($oid <= 0) continue;

    $amountCent = isset($row['vads_amount']) ? (int)$row['vads_amount'] : 0;
    $amountEur = $amountCent / 100;

    $dateLabel = format_vads_trans_date($row['vads_trans_date'] ?? null);
    $tx = isset($row['vads_trans_id']) && $row['vads_trans_id'] !== '' ? (string)$row['vads_trans_id'] : null;

    $labelParts = [];
    if (!empty($dateLabel)) $labelParts[] = $dateLabel;
    if (!empty($tx)) $labelParts[] = 'TX ' . $tx;
    $label = !empty($labelParts) ? implode(' • ', $labelParts) : 'Paiement';

    if (!isset($byOrder[$oid])) $byOrder[$oid] = [];
    $byOrder[$oid][] = [
      'amount_eur' => $amountEur,
      'label' => $label,
    ];
  }

  return $byOrder;
}

/**
 * Retourne un tableau indexé par orderId avec le total des paiements réussis en centimes :
 * [
 *   123 => 2500,
 *   124 => 1200,
 * ]
 *
 * Même règle stricte que get_successful_online_payments_by_order_id():
 * - vads_result = '00'
 * - vads_order_info = mapping strict du store / orderInfo
 */
function get_successful_online_payment_totals_by_order_id(PDO $pdo, array $orderIds, $orderInfo): array {
  $orderIds = array_values(array_unique(array_map('intval', $orderIds)));
  if (empty($orderIds)) return [];

  $oi = is_string($orderInfo) ? trim($orderInfo) : '';
  $storeCandidate = strtolower($oi);
  if ($storeCandidate === 'lan' || $storeCandidate === 'pel') {
    $mapped = onlinePaymentsOrderInfoForStore($storeCandidate);
    $oi = is_string($mapped) ? trim($mapped) : '';
  }
  if ($oi === '') return [];

  $placeholders = implode(',', array_fill(0, count($orderIds), '?'));
  $sql = "
    SELECT
      vads_order_id,
      COALESCE(SUM(vads_amount), 0) AS total_amount_cent
    FROM pos_pay
    WHERE vads_order_id IN ($placeholders)
      AND vads_result = '00'
      AND vads_order_info = ?
    GROUP BY vads_order_id
  ";

  $stmt = $pdo->prepare($sql);
  $params = $orderIds;
  $params[] = $oi;
  $stmt->execute($params);

  $totals = [];
  while ($row = $stmt->fetch(PDO::FETCH_ASSOC)) {
    $oid = isset($row['vads_order_id']) ? (int)$row['vads_order_id'] : 0;
    if ($oid <= 0) continue;
    $totals[$oid] = isset($row['total_amount_cent']) ? (int)$row['total_amount_cent'] : 0;
  }

  return $totals;
}

/**
 * has_successful_online_payment_for_order(PDO $pdo, string $store, int $orderId) -> bool
 * - Strict business rule for "deposit paid?":
 *   - pos_pay.vads_result = '00'
 *   - pos_pay.vads_order_id = orderId
 *   - pos_pay.vads_order_info = store-mapped value (LANCON / PELISSANNE)
 *
 * Note:
 * - If mapping is missing, we RETURN false (treat as unpaid) and caller should log.
 */
function has_successful_online_payment_for_order(PDO $pdo, string $store, int $orderId): bool {
  $oid = (int)$orderId;
  if ($oid <= 0) return false;

  $orderInfo = onlinePaymentsOrderInfoForStore($store);
  $oi = is_string($orderInfo) ? trim($orderInfo) : '';
  if ($oi === '') {
    // Strict: mapping missing => cannot assert payment belongs to this store.
    return false;
  }

  $stmt = $pdo->prepare("
    SELECT 1
    FROM pos_pay
    WHERE vads_order_id = :oid
      AND vads_result = '00'
      AND vads_order_info = :oi
    LIMIT 1
  ");
  $stmt->bindValue(':oid', $oid, PDO::PARAM_INT);
  $stmt->bindValue(':oi', $oi, PDO::PARAM_STR);
  $stmt->execute();
  $row = $stmt->fetch(PDO::FETCH_ASSOC);
  return is_array($row);
}
"
            }
        },
        {
            "path": "livreur/get_orders.php",
            "kind": "file",
            "before": {
                "exists": true,
                "kind": "file",
                "size": 8117,
                "sha1": "b07f6f9a01977b5a0dbbbc6118240a2e2fd60a13",
                "content_b64": "<?php
/* doc-project | livreur/get_orders.php | Fournit en JSON les commandes de livraison du jour pour le store courant via un mapping de tables mutualisé, avec résolution d’adresse effective (temporaire prioritaire) pour affichage et navigation. | Expose: calculateOrderPayment, buildEffectiveAddressData | Dépend de: livreur/auth_magic.php, livreur/config.php, livreur/lib/store.php, tables POS selon store, pos_pay, pos_temp_adresses | Impacte: réponse API JSON, lecture BDD multi-store, navigation livreur | Tables: pos_commandes*, std_clients, pos_pizzas_commandees*, pos_pizzas, pos_modifs_pizzas*, pos_pay, pos_temp_adresses */
require_once __DIR__ . '/auth_magic.php';
require_driver_auth('json');
header('Content-Type: application/json; charset=utf-8');

date_default_timezone_set('Europe/Paris');
$currentDate = new DateTime();
$currentDate->setTime(0, 0); // Début de la journée

$timestampStartOfDay = $currentDate->getTimestamp() * 1000;
$storeConfig = livreur_store_current_config();
$tables = $storeConfig['tables'];

// Fonction pour calculer les détails de paiement
function calculateOrderPayment(PDO $pdo, array $storeConfig, int $orderId): array {
    $tables = $storeConfig['tables'];
    // Récupérer les pizzas commandées et calculer le total de la commande
    $stmt = $pdo->prepare("
        SELECT ppc.id, ppc.id_pos_pizzas, ppc.is_web_addition, pp.price_large, pp.price_medium
        FROM {$tables['pos_pizzas_commandees']} ppc
        JOIN pos_pizzas pp ON ppc.id_pos_pizzas = pp.id
        WHERE ppc.{$tables['order_fk']} = :orderId
    ");
    $stmt->execute([':orderId' => $orderId]);
    $pizzasCommande = $stmt->fetchAll();

    $totalOrderAmount = 0;

    foreach ($pizzasCommande as $pizza) {
        // Déterminer le prix de base selon la taille
        $basePrice = $pizza['is_web_addition'] ? $pizza['price_medium'] : $pizza['price_large'];
        $totalOrderAmount += $basePrice;

        // Ajouter le prix des options de chaque pizza
        $stmt = $pdo->prepare("
            SELECT data_price
            FROM {$tables['pos_modifs_pizzas']}
            WHERE {$tables['pizza_option_fk']} = :pizzaId
        ");
        $stmt->execute([':pizzaId' => $pizza['id']]);
        $options = $stmt->fetchAll();

        foreach ($options as $option) {
            $totalOrderAmount += $option['data_price'];
        }
    }

    // Récupérer le montant total payé
    $stmt = $pdo->prepare("
        SELECT SUM(vads_amount) AS total_paid
        FROM pos_pay
        WHERE vads_order_id = :orderId
          AND vads_result = '00'
          AND vads_order_info = :orderInfo
    ");
    $stmt->execute([
        ':orderId' => $orderId,
        ':orderInfo' => $storeConfig['payment_order_info'],
    ]);
    $paymentData = $stmt->fetch();
    // Calculer le montant total payé en euros
    $totalPaidAmount = ($paymentData['total_paid'] ?? 0) / 100;

    // Calculer le montant restant dû en euros
    $remainingAmount = $totalOrderAmount - $totalPaidAmount;

    // Déterminer si la commande est payée
    $isPaid = $totalPaidAmount >= $totalOrderAmount;

    return [
        'totalAmount' => $totalOrderAmount,
        'totalPaid' => $totalPaidAmount,
        'remainingAmount' => $remainingAmount,
        'isPaid' => $isPaid
    ];
}

function buildEffectiveAddressData(array $order, ?array $tempAddress, array $storeConfig): array {
    if (is_array($tempAddress) && trim((string)($tempAddress['adresse'] ?? '')) !== '') {
        return [
            'isTemporary' => true,
            'adresse' => (string)($tempAddress['adresse'] ?? ''),
            'complement_adresse' => (string)($tempAddress['complement_adresse'] ?? ''),
            'code_postal' => (string)($tempAddress['code_postal'] ?? $storeConfig['default_postcode']),
            'ville' => (string)($tempAddress['ville'] ?? $storeConfig['default_city']),
            'numero_urgence' => (string)($tempAddress['numero_urgence'] ?? ''),
            'latitude' => (string)($tempAddress['latitude'] ?? ''),
            'longitude' => (string)($tempAddress['longitude'] ?? ''),
            'explications' => (string)($tempAddress['explications'] ?? ''),
        ];
    }

    return [
        'isTemporary' => false,
        'adresse' => (string)($order['adresse'] ?? ''),
        'complement_adresse' => (string)($order['complement_adresse'] ?? ''),
        'code_postal' => (string)($order['code_postal'] ?? $storeConfig['default_postcode']),
        'ville' => (string)($order['ville'] ?? $storeConfig['default_city']),
        'numero_urgence' => '',
        'latitude' => (string)($order['latitude'] ?? ''),
        'longitude' => (string)($order['longitude'] ?? ''),
        'explications' => '',
    ];
}

// Récupérer les commandes du jour
$stmt = $pdo->prepare("SELECT pc.*, sc.nom_prenom, sc.adresse, sc.complement_adresse, sc.phoneNumber, sc.code_postal, sc.ville, sc.num_supp1, sc.num_supp2, sc.latitude, sc.longitude
                       FROM {$tables['pos_commandes']} pc
                       JOIN std_clients sc ON pc.id_client = sc.id
                       WHERE pc.livraison = 1 
                       AND pc.statut != 'deleted'
                       AND pc.heure_prepa >= :timestampStartOfDay
                       ORDER BY pc.heure_prepa ASC");
$stmt->execute([':timestampStartOfDay' => $timestampStartOfDay]);
$orders = $stmt->fetchAll();

$response = [];

foreach ($orders as $order) {
    $orderId = (int)$order['id'];
    // Calculer les détails de paiement de la commande
    $paymentDetails = calculateOrderPayment($pdo, $storeConfig, $orderId);

    // Récupérer les pizzas et leurs modifications/options
    $stmt = $pdo->prepare("SELECT ppc.*, pp.name AS pizza_name, pp.price_large, pp.price_medium
                           FROM {$tables['pos_pizzas_commandees']} ppc
                           JOIN pos_pizzas pp ON ppc.id_pos_pizzas = pp.id
                           WHERE ppc.{$tables['order_fk']} = :orderId");
    $stmt->execute([':orderId' => $orderId]);
    $pizzas = $stmt->fetchAll();

    foreach ($pizzas as $key => $pizza) {
        $stmt = $pdo->prepare("
            SELECT nom_option, class_option, data_price
            FROM {$tables['pos_modifs_pizzas']}
            WHERE {$tables['pizza_option_fk']} = :pizzaId
        ");
        $stmt->execute([':pizzaId' => $pizza['id']]);
        $options = $stmt->fetchAll();
        $pizzas[$key]['options'] = $options;
    }

    // Récupérer les détails de l'adresse temporaire si disponible
    $tempAddress = null;
    $tempAddressId = null;
    if (isset($order['temp_address_id']) && $order['temp_address_id'] !== null) {
        $tempAddressId = $order['temp_address_id'];
    } elseif (isset($order['adresse_temporaire']) && $order['adresse_temporaire'] !== null) {
        $tempAddressId = $order['adresse_temporaire'];
    }

    if ($tempAddressId !== null) {
        $stmt = $pdo->prepare("SELECT * FROM pos_temp_adresses WHERE id = :tempAddressId");
        $stmt->execute([':tempAddressId' => $tempAddressId]);
        $tempAddress = $stmt->fetch();
    }

    $order['tempAddress'] = $tempAddress;
    $effectiveAddress = buildEffectiveAddressData($order, is_array($tempAddress) ? $tempAddress : null, $storeConfig);
    $hasCoordinates = trim((string)$effectiveAddress['latitude']) !== '' && trim((string)$effectiveAddress['longitude']) !== '';
    $order['effectiveAddress'] = $effectiveAddress;

    // Ajouter les détails de la commande, les pizzas, les détails de paiement, et l'adresse temporaire au tableau de réponse
    $response[] = [
        'store' => [
            'code' => $storeConfig['code'],
            'label' => $storeConfig['label'],
            'defaultCity' => $storeConfig['default_city'],
            'defaultPostcode' => $storeConfig['default_postcode'],
        ],
        'order' => $order,
        'pizzas' => $pizzas,
        'paymentDetails' => $paymentDetails,
        'hasCoordinates' => $hasCoordinates,
        'tempAddress' => $tempAddress,
        'effectiveAddress' => $effectiveAddress
    ];
}

echo json_encode($response);
?>
"
            },
            "after": {
                "exists": true,
                "kind": "file",
                "size": 9849,
                "sha1": "c01653cdc4fcb593b4eb6299587a708e65d6b1bc",
                "content_b64": "<?php
/* doc-project | livreur/get_orders.php | Fournit en JSON les commandes de livraison du jour pour le store courant via un mapping de tables mutualisé, avec résolution d’adresse effective (temporaire prioritaire) pour affichage et navigation, et détection robuste du paiement en ligne centralisée via onlinePayments.php en centimes. | Expose: livreurNormalizeAmountToCentimes, calculateOrderItemsTotalCentimes, calculateOrderPayment, buildEffectiveAddressData | Dépend de: livreur/auth_magic.php, livreur/config.php, livreur/lib/store.php, caisse-aqp/public/api/_lib/onlinePayments.php, tables POS selon store, pos_pay, pos_temp_adresses | Impacte: réponse API JSON, lecture BDD multi-store, badge paiement livreur, navigation livreur | Tables: pos_commandes*, std_clients, pos_pizzas_commandees*, pos_pizzas, pos_modifs_pizzas*, pos_pay, pos_temp_adresses */
require_once __DIR__ . '/auth_magic.php';
require_once dirname(__DIR__) . '/caisse-aqp/public/api/_lib/onlinePayments.php';
require_driver_auth('json');
header('Content-Type: application/json; charset=utf-8');

date_default_timezone_set('Europe/Paris');
$currentDate = new DateTime();
$currentDate->setTime(0, 0); // Début de la journée

$timestampStartOfDay = $currentDate->getTimestamp() * 1000;
$storeConfig = livreur_store_current_config();
$tables = $storeConfig['tables'];

function livreurNormalizeAmountToCentimes($amount): int {
    if ($amount === null || $amount === '') {
        return 0;
    }
    if (is_string($amount)) {
        $amount = str_replace([' ', ','], ['', '.'], $amount);
    }
    $value = (float)$amount;
    if (!is_finite($value)) {
        return 0;
    }
    return (int)round($value * 100);
}

function calculateOrderItemsTotalCentimes(PDO $pdo, array $storeConfig, int $orderId): int {
    $tables = $storeConfig['tables'];
    // Récupérer les pizzas commandées et calculer le total de la commande
    $stmt = $pdo->prepare("
        SELECT ppc.id, ppc.id_pos_pizzas, ppc.is_web_addition, pp.price_large, pp.price_medium
        FROM {$tables['pos_pizzas_commandees']} ppc
        JOIN pos_pizzas pp ON ppc.id_pos_pizzas = pp.id
        WHERE ppc.{$tables['order_fk']} = :orderId
    ");
    $stmt->execute([':orderId' => $orderId]);
    $pizzasCommande = $stmt->fetchAll();

    $totalOrderCentimes = 0;

    foreach ($pizzasCommande as $pizza) {
        // Déterminer le prix de base selon la taille
        $basePrice = $pizza['is_web_addition'] ? $pizza['price_medium'] : $pizza['price_large'];
        $totalOrderCentimes += livreurNormalizeAmountToCentimes($basePrice);

        // Ajouter le prix des options de chaque pizza
        $stmt = $pdo->prepare("
            SELECT data_price
            FROM {$tables['pos_modifs_pizzas']}
            WHERE {$tables['pizza_option_fk']} = :pizzaId
        ");
        $stmt->execute([':pizzaId' => $pizza['id']]);
        $options = $stmt->fetchAll();

        foreach ($options as $option) {
            $totalOrderCentimes += livreurNormalizeAmountToCentimes($option['data_price']);
        }
    }

    return $totalOrderCentimes;
}

// Fonction pour calculer les détails de paiement
function calculateOrderPayment(PDO $pdo, array $storeConfig, array $order): array {
    $orderId = isset($order['id']) ? (int)$order['id'] : 0;
    if ($orderId <= 0) {
        return [
            'totalAmount' => 0.0,
            'totalPaid' => 0.0,
            'remainingAmount' => 0.0,
            'totalAmountCentimes' => 0,
            'totalPaidCentimes' => 0,
            'remainingAmountCentimes' => 0,
            'isPaid' => false,
        ];
    }

    $totalOrderCentimes = livreurNormalizeAmountToCentimes($order['prix_com'] ?? null);
    if ($totalOrderCentimes <= 0) {
        $totalOrderCentimes = calculateOrderItemsTotalCentimes($pdo, $storeConfig, $orderId);
    }

    $paymentMatchValues = [
        $orderId,
        $order['id_date'] ?? null,
        $order['payLink'] ?? null,
        $order['paylink'] ?? null,
    ];

    $paymentData = get_successful_online_payment_total_centimes_for_order(
        $pdo,
        $orderId,
        $storeConfig['payment_order_info'] ?? $storeConfig['code'] ?? '',
        $paymentMatchValues
    );

    $totalPaidCentimes = (int)($paymentData['total_paid_centimes'] ?? 0);
    $remainingCentimes = max(0, $totalOrderCentimes - $totalPaidCentimes);

    // Tolérance d’un centime pour neutraliser les écarts de normalisation euros -> centimes.
    $isPaid = $totalOrderCentimes > 0 && ($totalPaidCentimes + 1) >= $totalOrderCentimes;

    return [
        'totalAmount' => round($totalOrderCentimes / 100, 2),
        'totalPaid' => round($totalPaidCentimes / 100, 2),
        'remainingAmount' => round($remainingCentimes / 100, 2),
        'totalAmountCentimes' => $totalOrderCentimes,
        'totalPaidCentimes' => $totalPaidCentimes,
        'remainingAmountCentimes' => $remainingCentimes,
        'matchedPaymentRows' => (int)($paymentData['matched_rows'] ?? 0),
        'isPaid' => $isPaid
    ];
}

function buildEffectiveAddressData(array $order, ?array $tempAddress, array $storeConfig): array {
    if (is_array($tempAddress) && trim((string)($tempAddress['adresse'] ?? '')) !== '') {
        return [
            'isTemporary' => true,
            'adresse' => (string)($tempAddress['adresse'] ?? ''),
            'complement_adresse' => (string)($tempAddress['complement_adresse'] ?? ''),
            'code_postal' => (string)($tempAddress['code_postal'] ?? $storeConfig['default_postcode']),
            'ville' => (string)($tempAddress['ville'] ?? $storeConfig['default_city']),
            'numero_urgence' => (string)($tempAddress['numero_urgence'] ?? ''),
            'latitude' => (string)($tempAddress['latitude'] ?? ''),
            'longitude' => (string)($tempAddress['longitude'] ?? ''),
            'explications' => (string)($tempAddress['explications'] ?? ''),
        ];
    }

    return [
        'isTemporary' => false,
        'adresse' => (string)($order['adresse'] ?? ''),
        'complement_adresse' => (string)($order['complement_adresse'] ?? ''),
        'code_postal' => (string)($order['code_postal'] ?? $storeConfig['default_postcode']),
        'ville' => (string)($order['ville'] ?? $storeConfig['default_city']),
        'numero_urgence' => '',
        'latitude' => (string)($order['latitude'] ?? ''),
        'longitude' => (string)($order['longitude'] ?? ''),
        'explications' => '',
    ];
}

// Récupérer les commandes du jour
$stmt = $pdo->prepare("SELECT pc.*, sc.nom_prenom, sc.adresse, sc.complement_adresse, sc.phoneNumber, sc.code_postal, sc.ville, sc.num_supp1, sc.num_supp2, sc.latitude, sc.longitude
                       FROM {$tables['pos_commandes']} pc
                       JOIN std_clients sc ON pc.id_client = sc.id
                       WHERE pc.livraison = 1 
                       AND pc.statut != 'deleted'
                       AND pc.heure_prepa >= :timestampStartOfDay
                       ORDER BY pc.heure_prepa ASC");
$stmt->execute([':timestampStartOfDay' => $timestampStartOfDay]);
$orders = $stmt->fetchAll();

$response = [];

foreach ($orders as $order) {
    $orderId = (int)$order['id'];
    // Calculer les détails de paiement de la commande
    $paymentDetails = calculateOrderPayment($pdo, $storeConfig, $order);

    // Récupérer les pizzas et leurs modifications/options
    $stmt = $pdo->prepare("SELECT ppc.*, pp.name AS pizza_name, pp.price_large, pp.price_medium
                           FROM {$tables['pos_pizzas_commandees']} ppc
                           JOIN pos_pizzas pp ON ppc.id_pos_pizzas = pp.id
                           WHERE ppc.{$tables['order_fk']} = :orderId");
    $stmt->execute([':orderId' => $orderId]);
    $pizzas = $stmt->fetchAll();

    foreach ($pizzas as $key => $pizza) {
        $stmt = $pdo->prepare("
            SELECT nom_option, class_option, data_price
            FROM {$tables['pos_modifs_pizzas']}
            WHERE {$tables['pizza_option_fk']} = :pizzaId
        ");
        $stmt->execute([':pizzaId' => $pizza['id']]);
        $options = $stmt->fetchAll();
        $pizzas[$key]['options'] = $options;
    }

    // Récupérer les détails de l'adresse temporaire si disponible
    $tempAddress = null;
    $tempAddressId = null;
    if (isset($order['temp_address_id']) && $order['temp_address_id'] !== null) {
        $tempAddressId = $order['temp_address_id'];
    } elseif (isset($order['adresse_temporaire']) && $order['adresse_temporaire'] !== null) {
        $tempAddressId = $order['adresse_temporaire'];
    }

    if ($tempAddressId !== null) {
        $stmt = $pdo->prepare("SELECT * FROM pos_temp_adresses WHERE id = :tempAddressId");
        $stmt->execute([':tempAddressId' => $tempAddressId]);
        $tempAddress = $stmt->fetch();
    }

    $order['tempAddress'] = $tempAddress;
    $effectiveAddress = buildEffectiveAddressData($order, is_array($tempAddress) ? $tempAddress : null, $storeConfig);
    $hasCoordinates = trim((string)$effectiveAddress['latitude']) !== '' && trim((string)$effectiveAddress['longitude']) !== '';
    $order['effectiveAddress'] = $effectiveAddress;

    // Ajouter les détails de la commande, les pizzas, les détails de paiement, et l'adresse temporaire au tableau de réponse
    $response[] = [
        'store' => [
            'code' => $storeConfig['code'],
            'label' => $storeConfig['label'],
            'defaultCity' => $storeConfig['default_city'],
            'defaultPostcode' => $storeConfig['default_postcode'],
        ],
        'order' => $order,
        'pizzas' => $pizzas,
        'paymentDetails' => $paymentDetails,
        'hasCoordinates' => $hasCoordinates,
        'tempAddress' => $tempAddress,
        'effectiveAddress' => $effectiveAddress
    ];
}

echo json_encode($response);
?>
"
            }
        }
    ],
    "meta": {
        "summary": {
            "changed": 2,
            "created": 0,
            "deleted": 0,
            "errors": 0
        },
        "patch_sha1": "188296c945228ee1268be7bb0e4d9517c5e0ffd9",
        "created_files": [],
        "branching": {
            "auto_branch_created": false,
            "auto_branch_id": null,
            "source_branch_id": "main",
            "base_event_id": "event_3c01b1f1ba5c5d08"
        },
        "transition_store": {
            "dir": "event-assets/event_06b0fefe666cc6c8",
            "manifest": "event-assets/event_06b0fefe666cc6c8/manifest.json",
            "files_count": 2
        }
    },
    "impacted_paths": [
        "caisse-aqp/public/api/_lib/onlinePayments.php",
        "livreur/get_orders.php"
    ]
}