Initial commit: CLEO ERP avec améliorations debug
- Configuration du debug conditionnel pour dev/recette - Fonction debug() globale avec niveaux - Logging des requêtes SQL - Handlers d'exceptions et d'erreurs globaux 🤖 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
148
config/conf.php
Normal file
148
config/conf.php
Normal file
@@ -0,0 +1,148 @@
|
||||
<?php
|
||||
|
||||
class Conf
|
||||
{
|
||||
const admin = 1; // TRUE ou FALSE pour indiquer si l'application est admin ou non
|
||||
const intra = 1; // Est-ce un intranet privé TRUE 1, ou un site public FALSE 0
|
||||
const erp = 1; //! Est-ce un ERP ? Utile pour la gestion documentaire avec les paths spéciaux pour l'ERP
|
||||
const magazine = 0; //! Est-ce qu'on veut transformer les PDF en JPG pour la lecture Magazine dans le d6tools.upload ?
|
||||
|
||||
public $_appname;
|
||||
public $_appscript;
|
||||
public $_appversion;
|
||||
public $_appenv;
|
||||
public $_apptitle;
|
||||
|
||||
public $_brandname;
|
||||
public $_brandadresse1;
|
||||
public $_brandadresse2;
|
||||
public $_brandcp;
|
||||
public $_brandville;
|
||||
public $_brandtel;
|
||||
public $_brandemail;
|
||||
public $_brandlogo;
|
||||
public $_brandgroupe;
|
||||
public $_brandmulti;
|
||||
|
||||
public $_piwikid;
|
||||
public $_googlid;
|
||||
|
||||
public $_excludeIp = "82.67.142.214"; //! IP à exclure pour le comptage des visites et pour le debug
|
||||
public $_clientIp;
|
||||
public $_devIp = false;
|
||||
|
||||
//! Configuration du debug
|
||||
public $_debug_level = 0; //! 0=off, 1=errors, 2=warnings, 3=info, 4=debug
|
||||
public $_log_sql = false; //! Logger les requêtes SQL
|
||||
public $_log_performance = false; //! Logger les temps d'exécution
|
||||
public $_log_file_path = ''; //! Chemin du fichier de log
|
||||
|
||||
public $_pathupload = "/pub/files/upload/"; //! le path de base pour les uploads
|
||||
|
||||
//! les infos de connexion de la base de données
|
||||
public $_dbhost = 'localhost';
|
||||
public $_dbname = 'uof_frontal';
|
||||
public $_dbuser = 'uof_front_user';
|
||||
public $_dbpass = 'd66,UnikOffice.User';
|
||||
|
||||
public $_dbghost = 'localhost';
|
||||
public $_dbgname = '';
|
||||
public $_dbguser = 'uof_linet_user';
|
||||
public $_dbgpass = 'd66,UOF-LinetRH.User';
|
||||
|
||||
public $_dbuhost = 'localhost';
|
||||
public $_dbuname = '';
|
||||
public $_dbuuser = 'uof_linet_user';
|
||||
public $_dbupass = 'd66,UOF-LinetRH.User';
|
||||
|
||||
public $_tbusers = ""; // Spécifie la table des users de cette application, par défaut uof_frontal.users, mais sur Linet c'est dans uof_linet.commerciaux
|
||||
|
||||
//! les infos de l'entité de l'utilisateur
|
||||
public $_entite = '';
|
||||
|
||||
//! indique si c'est une nouvelle version pour les tests de nouveaux modules et librairies
|
||||
public $_new_version = false;
|
||||
|
||||
public function __construct()
|
||||
{
|
||||
//! on va chercher la configuration de l'application dans la table ce_frontal.y_conf
|
||||
$mysqli = new mysqli($this->_dbhost, $this->_dbuser, $this->_dbpass, $this->_dbname);
|
||||
$sql = 'SELECT * FROM y_conf WHERE admin=' . self::admin . ' AND active=1 LIMIT 1;';
|
||||
$mysqli->set_charset("utf8");
|
||||
$res = $mysqli->query($sql);
|
||||
$resconf = $res->fetch_assoc();
|
||||
$this->_appenv = $resconf["appenv"];
|
||||
$this->_appversion = $resconf["appversion"];
|
||||
$this->_appscript = $resconf["appscript"]; //! le script à appeler par défaut si l'utilisateur n'est pas reconnu
|
||||
|
||||
$this->_brandgroupe = $resconf["brandgroupe"];
|
||||
$this->_brandmulti = $resconf["brandmulti"];
|
||||
|
||||
//! On va chercher les infos de base de cette appname dans ce_frontal.users_entites en fonction du http_host
|
||||
$http_host = $_SERVER['HTTP_HOST'];
|
||||
error_log("http_host : ".$http_host);
|
||||
$sql = 'SELECT * FROM users_entites WHERE http_host LIKE "%' . $http_host . '%" AND active=1 LIMIT 1;';
|
||||
$res = $mysqli->query($sql);
|
||||
$mysqli->close();
|
||||
$resentite = $res->fetch_assoc();
|
||||
if (empty($resentite)) {
|
||||
//! on ne trouve pas ce http_host, on part sur la demo
|
||||
$this->_appname = "udo_demo";
|
||||
$mysqli = new mysqli($this->_dbhost, $this->_dbuser, $this->_dbpass, $this->_dbname);
|
||||
$sql = 'SELECT * FROM users_entites WHERE rowid=1;'; // appname="' . $this->_appname . '" AND active=1 LIMIT 1;';
|
||||
$res = $mysqli->query($sql);
|
||||
$mysqli->close();
|
||||
$resentite = $res->fetch_assoc();
|
||||
}
|
||||
$this->_entite = $resentite;
|
||||
$this->_appname = $resentite["appname"];
|
||||
$this->_apptitle = $resentite["libelle"];
|
||||
$this->_brandname = $resentite["libelle"];
|
||||
$this->_brandadresse1 = $resentite["adresse1"];
|
||||
$this->_brandadresse2 = $resentite["adresse2"];
|
||||
$this->_brandcp = $resentite["cp"];
|
||||
$this->_brandville = $resentite["ville"];
|
||||
$this->_brandtel = $resentite["tel1"];
|
||||
$this->_brandemail = $resentite["email"];
|
||||
$this->_brandlogo = $resentite["appname"];
|
||||
|
||||
$this->_dbgname = $resentite["groupebase"];
|
||||
$this->_dbuname = $resentite["genbase"];
|
||||
$this->_tbusers = $resentite["table_users_gen"]; //! Spécifie la table des users de cette application, par défaut dans uof_frontal.users
|
||||
|
||||
if (!empty($_SERVER["HTTP_CLIENT_IP"])) {
|
||||
$this->_clientIp = $_SERVER["HTTP_CLIENT_IP"];
|
||||
} elseif (!empty($_SERVER["HTTP_X_FORWARDED_FOR"])) {
|
||||
$this->_clientIp = $_SERVER["HTTP_X_FORWARDED_FOR"];
|
||||
} else {
|
||||
$this->_clientIp = $_SERVER["REMOTE_ADDR"];
|
||||
}
|
||||
// Active le debug uniquement pour dev et recette
|
||||
if (strpos($http_host, 'dcleo.unikoffice.com') !== false || strpos($http_host, 'rcleo.unikoffice.com') !== false) {
|
||||
ini_set('error_reporting', -1);
|
||||
ini_set('display_errors', '1');
|
||||
$this->_devIp = true;
|
||||
|
||||
// Configuration avancée du debug pour dev/recette
|
||||
$this->_debug_level = 4; // Niveau debug complet
|
||||
$this->_log_sql = true; // Logger les requêtes SQL
|
||||
$this->_log_performance = true; // Mesurer les performances
|
||||
$this->_log_file_path = dirname(__DIR__) . '/' . $this->_appname . '_debug.log';
|
||||
|
||||
// Options PHP supplémentaires pour le debug
|
||||
ini_set('log_errors', '1');
|
||||
ini_set('error_log', $this->_log_file_path);
|
||||
ini_set('display_startup_errors', '1');
|
||||
ini_set('track_errors', '1');
|
||||
ini_set('html_errors', '1');
|
||||
ini_set('xmlrpc_errors', '0');
|
||||
} else {
|
||||
ini_set('error_reporting', 0);
|
||||
ini_set('display_errors', '0');
|
||||
ini_set('log_errors', '0');
|
||||
$this->_debug_level = 0;
|
||||
$this->_log_sql = false;
|
||||
$this->_log_performance = false;
|
||||
}
|
||||
}
|
||||
}
|
||||
131
config/init.php
Normal file
131
config/init.php
Normal file
@@ -0,0 +1,131 @@
|
||||
<?php
|
||||
$tpsdebut = microtime(true);
|
||||
setlocale(LC_ALL, 'fr_FR');
|
||||
|
||||
define('DS', DIRECTORY_SEPARATOR);
|
||||
define('LIBROOT', dirname(__FILE__));
|
||||
define('ROOT', dirname(LIBROOT));
|
||||
define('CONFROOT', ROOT . DS . 'config');
|
||||
define('CONTROOT', ROOT . DS . 'controllers');
|
||||
define('MODROOT', ROOT . DS . 'models');
|
||||
define('VIEWROOT', ROOT . DS . 'views');
|
||||
define('LAYROOT', VIEWROOT . DS . 'layouts');
|
||||
|
||||
require_once CONFROOT . DS . 'conf.php';
|
||||
$Conf = new Conf();
|
||||
|
||||
define('RESROOT', ROOT . DS . 'pub' . DS . 'res');
|
||||
define('FMKROOT', RESROOT . DS . 'd6');
|
||||
define('BLOCKROOT', FMKROOT . DS . 'blocks');
|
||||
|
||||
//! Chargement de la nouvelle version du d6tools allégée
|
||||
require_once FMKROOT . DS . 'd6_tools.php';
|
||||
|
||||
//! Chargement des fichiers spécifiques au projet
|
||||
require_once FMKROOT . DS . 'lib_cleo.php';
|
||||
|
||||
//! Handler d'exceptions global
|
||||
function exception_handler($exception) {
|
||||
global $Conf;
|
||||
|
||||
$error_data = array(
|
||||
'type' => 'EXCEPTION',
|
||||
'message' => $exception->getMessage(),
|
||||
'file' => $exception->getFile(),
|
||||
'line' => $exception->getLine(),
|
||||
'code' => $exception->getCode(),
|
||||
'trace' => $exception->getTraceAsString()
|
||||
);
|
||||
|
||||
// Logger l'exception
|
||||
if (isset($Conf->_debug_level) && $Conf->_debug_level > 0) {
|
||||
debug($error_data, "UNCAUGHT_EXCEPTION", 1);
|
||||
}
|
||||
|
||||
// Logger dans la table z_logs
|
||||
eLog(0, "Exception non gérée: " . $exception->getMessage() . " dans " . $exception->getFile() . ":" . $exception->getLine());
|
||||
|
||||
// Afficher une erreur propre à l'utilisateur
|
||||
if (isset($Conf->_devIp) && $Conf->_devIp && ini_get('display_errors')) {
|
||||
// En mode dev, afficher les détails
|
||||
echo "<div style='background:#fee; border:2px solid #c00; padding:20px; margin:20px; font-family:monospace;'>";
|
||||
echo "<h2 style='color:#c00;'>Exception non gérée</h2>";
|
||||
echo "<p><strong>Message:</strong> " . htmlspecialchars($exception->getMessage()) . "</p>";
|
||||
echo "<p><strong>Fichier:</strong> " . htmlspecialchars($exception->getFile()) . " ligne " . $exception->getLine() . "</p>";
|
||||
echo "<p><strong>Code:</strong> " . $exception->getCode() . "</p>";
|
||||
echo "<pre style='background:#fff; padding:10px; overflow:auto;'>" . htmlspecialchars($exception->getTraceAsString()) . "</pre>";
|
||||
echo "</div>";
|
||||
} else {
|
||||
// En production, afficher un message générique
|
||||
echo "<div style='text-align:center; padding:50px;'>";
|
||||
echo "<h2>Une erreur est survenue</h2>";
|
||||
echo "<p>Nous nous excusons pour la gêne occasionnée. L'erreur a été enregistrée.</p>";
|
||||
echo "<p><a href='/'>Retour à l'accueil</a></p>";
|
||||
echo "</div>";
|
||||
}
|
||||
}
|
||||
|
||||
//! Handler d'erreurs global
|
||||
function error_handler($errno, $errstr, $errfile, $errline) {
|
||||
global $Conf;
|
||||
|
||||
// Vérifier si l'erreur doit être rapportée selon error_reporting
|
||||
if (!(error_reporting() & $errno)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
$error_types = array(
|
||||
E_ERROR => 'ERROR',
|
||||
E_WARNING => 'WARNING',
|
||||
E_PARSE => 'PARSE',
|
||||
E_NOTICE => 'NOTICE',
|
||||
E_CORE_ERROR => 'CORE_ERROR',
|
||||
E_CORE_WARNING => 'CORE_WARNING',
|
||||
E_COMPILE_ERROR => 'COMPILE_ERROR',
|
||||
E_COMPILE_WARNING => 'COMPILE_WARNING',
|
||||
E_USER_ERROR => 'USER_ERROR',
|
||||
E_USER_WARNING => 'USER_WARNING',
|
||||
E_USER_NOTICE => 'USER_NOTICE',
|
||||
E_STRICT => 'STRICT',
|
||||
E_RECOVERABLE_ERROR => 'RECOVERABLE_ERROR',
|
||||
E_DEPRECATED => 'DEPRECATED',
|
||||
E_USER_DEPRECATED => 'USER_DEPRECATED'
|
||||
);
|
||||
|
||||
$error_type = isset($error_types[$errno]) ? $error_types[$errno] : 'UNKNOWN';
|
||||
|
||||
$error_data = array(
|
||||
'type' => $error_type,
|
||||
'errno' => $errno,
|
||||
'message' => $errstr,
|
||||
'file' => $errfile,
|
||||
'line' => $errline
|
||||
);
|
||||
|
||||
// Déterminer le niveau de debug pour ce type d'erreur
|
||||
$debug_level = 4; // Par défaut niveau le plus bas
|
||||
if (in_array($errno, array(E_ERROR, E_PARSE, E_CORE_ERROR, E_COMPILE_ERROR, E_USER_ERROR))) {
|
||||
$debug_level = 1; // Erreurs critiques
|
||||
} elseif (in_array($errno, array(E_WARNING, E_CORE_WARNING, E_COMPILE_WARNING, E_USER_WARNING))) {
|
||||
$debug_level = 2; // Warnings
|
||||
} elseif (in_array($errno, array(E_NOTICE, E_USER_NOTICE))) {
|
||||
$debug_level = 3; // Notices
|
||||
}
|
||||
|
||||
// Logger l'erreur
|
||||
if (isset($Conf->_debug_level) && $Conf->_debug_level >= $debug_level) {
|
||||
debug($error_data, "PHP_ERROR", $debug_level);
|
||||
}
|
||||
|
||||
// Pour les erreurs critiques, logger aussi dans z_logs
|
||||
if ($debug_level == 1) {
|
||||
eLog(0, "Erreur PHP $error_type: $errstr dans $errfile:$errline");
|
||||
}
|
||||
|
||||
// Ne pas exécuter le handler d'erreur PHP interne
|
||||
return true;
|
||||
}
|
||||
|
||||
//! Enregistrer les handlers
|
||||
set_exception_handler('exception_handler');
|
||||
set_error_handler('error_handler');
|
||||
Reference in New Issue
Block a user