Subversion Repositories Applications.framework

Rev

Rev 442 | Go to most recent revision | Only display areas with differences | Ignore whitespace | Details | Blame | Last modification | View Log | RSS feed

Rev 442 Rev 447
1
<?php
1
<?php
2
// declare(encoding='UTF-8');
2
// declare(encoding='UTF-8');
3
/**
3
/**
4
 * Config permet de charger automatiquement les fichiers ini du Framework et de l'application.
4
 * Config permet de charger automatiquement les fichiers ini du Framework et de l'application.
5
 * Elle offre l'accès en lecture seule aux paramètres de config.
5
 * Elle offre l'accès en lecture seule aux paramètres de config.
6
 * C'est une Singleton.
6
 * C'est une Singleton.
7
 * Si vous avez besoin de modifier dynamiquement des paramètres de configuration, utiliser le @see Registe, il est fait pour ça.
7
 * Si vous avez besoin de modifier dynamiquement des paramètres de configuration, utiliser le @see Registe, il est fait pour ça.
8
 *
8
 *
9
 * @category	PHP 5.2
9
 * @category	PHP 5.2
10
 * @package	Framework
10
 * @package	Framework
11
 * @author		Jean-Pascal MILCENT <jpm@tela-botanica.org>
11
 * @author		Jean-Pascal MILCENT <jpm@tela-botanica.org>
12
 * @copyright	Copyright (c) 2009, Tela Botanica (accueil@tela-botanica.org)
12
 * @copyright	Copyright (c) 2009, Tela Botanica (accueil@tela-botanica.org)
13
 * @license	http://www.gnu.org/licenses/gpl.html Licence GNU-GPL-v3
13
 * @license	http://www.gnu.org/licenses/gpl.html Licence GNU-GPL-v3
14
 * @license	http://www.cecill.info/licences/Licence_CeCILL_V2-fr.txt Licence CECILL-v2
14
 * @license	http://www.cecill.info/licences/Licence_CeCILL_V2-fr.txt Licence CECILL-v2
15
 * @version	$Id: Config.php 442 2013-10-22 10:00:19Z raphael $
15
 * @version	$Id: Config.php 447 2013-12-19 10:52:37Z jpm $
16
 * @link		/doc/framework/
16
 * @link		/doc/framework/
17
 */
17
 */
18
 
18
 
19
class Config {
19
class Config {
20
 
20
 
21
	/** Instance de la classe pointant sur elle même (pour le pattern singleton). */
21
	/** Instance de la classe pointant sur elle même (pour le pattern singleton). */
22
	private static $instance = null;
22
	private static $instance = null;
23
 
23
 
24
	/** Paramètres de configuration. */
24
	/** Paramètres de configuration. */
25
	private static $parametres = array();
25
	private static $parametres = array();
26
 
26
 
27
	private function __construct() {
27
	private function __construct() {
28
		// Définition de paramètres avant chargement du config.ini
28
		// Définition de paramètres avant chargement du config.ini
29
		self::$parametres = array(
29
		self::$parametres = array(
30
			'fichier_config' => 'config%s.ini',
30
			'fichier_config' => 'config%s.ini',
31
			'chemin_framework' => dirname(__FILE__).DS
31
			'chemin_framework' => dirname(__FILE__).DS
32
			);
32
			);
33
 
33
 
34
		// Chargement du fichier config.ini du Framework
34
		// Chargement du fichier config.ini du Framework
35
		$existe = self::parserFichierIni(self::$parametres['chemin_framework'].sprintf(self::$parametres['fichier_config'], ''));
35
		$existe = self::parserFichierIni(self::$parametres['chemin_framework'].sprintf(self::$parametres['fichier_config'], ''));
36
		if ($existe === false) {
36
		if ($existe === false) {
37
			trigger_error("Veuillez configurer le Framework en renommant le fichier config.defaut.ini en config.ini.", E_USER_ERROR);
37
			trigger_error("Veuillez configurer le Framework en renommant le fichier config.defaut.ini en config.ini.", E_USER_ERROR);
38
		}
38
		}
39
 
39
 
40
		// Chargement du fichier config.ini par défaut de l'application
40
		// Chargement du fichier config.ini par défaut de l'application
41
		$chemin_config_defaut_appli = self::$parametres['chemin_configurations'].sprintf(self::$parametres['fichier_config'], '');
41
		$chemin_config_defaut_appli = self::$parametres['chemin_configurations'].sprintf(self::$parametres['fichier_config'], '');
42
		self::parserFichierIni($chemin_config_defaut_appli);
42
		self::parserFichierIni($chemin_config_defaut_appli);
43
 
43
 
44
		// Chargement des fichiers config.ini contextuels
44
		// Chargement des fichiers config.ini contextuels
45
		if (PHP_SAPI == 'cli') {// mode console
45
		if (PHP_SAPI == 'cli') {// mode console
46
			foreach ($_SERVER['argv'] as $cle => $valeur) {
46
			foreach ($_SERVER['argv'] as $cle => $valeur) {
47
				if ($valeur == '-contexte') {
47
				if ($valeur == '-contexte') {
48
					self::chargerFichierContexte($_SERVER['argv'][($cle+1)]);
48
					self::chargerFichierContexte($_SERVER['argv'][($cle+1)]);
49
					break;
49
					break;
50
				}
50
				}
51
			}
51
			}
52
		} else {// mode web
52
		} else {// mode web
53
			// Pour Papyrus
53
			// Pour Papyrus
54
			if (defined('PAP_VERSION')) {
54
			if (defined('PAP_VERSION')) {
55
				self::chargerFichierContexte('papyrus');
55
				self::chargerFichierContexte('papyrus');
56
			}
56
			}
57
			// Via le fichie .ini par défaut de l'appli
57
			// Via le fichie .ini par défaut de l'appli
58
			if (Config::existeValeur('info.contexte', self::$parametres)) {
58
			if (Config::existeValeur('info.contexte', self::$parametres)) {
59
				self::chargerFichierContexte(Config::get('info.contexte'));
59
				self::chargerFichierContexte(Config::get('info.contexte'));
60
			}
60
			}
61
 
61
 
62
			// Chargement du contexte présent dans le GET
62
			// Chargement du contexte présent dans le GET
63
			if (isset($_GET['contexte'])) {
63
			if (isset($_GET['contexte'])) {
64
				$_GET['contexte'] = strip_tags($_GET['contexte']);
64
				$_GET['contexte'] = strip_tags($_GET['contexte']);
65
				self::chargerFichierContexte($_GET['contexte']);
65
				self::chargerFichierContexte($_GET['contexte']);
66
			}
66
			}
67
 
67
 
68
			// Chargement du contexte présent dans le POST
68
			// Chargement du contexte présent dans le POST
69
			if (isset($_POST['contexte'])) {
69
			if (isset($_POST['contexte'])) {
70
				$_POST['contexte'] = strip_tags($_POST['contexte']);
70
				$_POST['contexte'] = strip_tags($_POST['contexte']);
71
				self::chargerFichierContexte($_POST['contexte']);
71
				self::chargerFichierContexte($_POST['contexte']);
72
			}
72
			}
73
		}
73
		}
74
	}
74
	}
75
 
75
 
76
	/**
76
	/**
77
	 * Charge le fichier de config correspondant au contexte
77
	 * Charge le fichier de config correspondant au contexte
78
	 * @param string $contexte le contexte
78
	 * @param string $contexte le contexte
79
	 */
79
	 */
80
	private static function chargerFichierContexte($contexte) {
80
	private static function chargerFichierContexte($contexte) {
81
		$chemin_config_appli_contextuel = self::$parametres['chemin_configurations'];
81
		$chemin_config_appli_contextuel = self::$parametres['chemin_configurations'];
82
		$chemin_config_appli_contextuel .= sprintf(self::$parametres['fichier_config'], '_'.$contexte);
82
		$chemin_config_appli_contextuel .= sprintf(self::$parametres['fichier_config'], '_'.$contexte);
83
		self::parserFichierIni($chemin_config_appli_contextuel);
83
		self::parserFichierIni($chemin_config_appli_contextuel);
84
	}
84
	}
85
 
85
 
86
	/**
86
	/**
87
	 * Parse le fichier ini donné en paramètre
87
	 * Parse le fichier ini donné en paramètre
88
	 * @param string $fichier_ini nom du fichier ini à parser
88
	 * @param string $fichier_ini nom du fichier ini à parser
89
	 * @return array tableau contenant les paramètres du fichier ini
89
	 * @return array tableau contenant les paramètres du fichier ini
90
	 */
90
	 */
91
	private static function parserFichierIni($fichier_ini) {
91
	private static function parserFichierIni($fichier_ini) {
-
 
92
		$retour = false;
-
 
93
		if (file_exists($fichier_ini)) {
92
        return self::analyserTableauIni(parse_ini_file($fichier_ini, true));
94
			$ini = parse_ini_file($fichier_ini, true);
-
 
95
			$ini = self::analyserTableauIni($ini);
-
 
96
			$retour = true;
-
 
97
		} else {
-
 
98
			error_log("Fichier de configuration introuvable : $fichier_ini");
-
 
99
		}
-
 
100
		return $retour;
93
	}
101
	}
94
 
102
 
95
	/**
103
	/**
96
	 * Fusionne un tableau de paramètres avec le tableau de paramètres global
104
	 * Fusionne un tableau de paramètres avec le tableau de paramètres global
97
	 * @param array $ini le tableau à fusionner
105
	 * @param array $ini le tableau à fusionner
98
	 */
106
	 */
99
	private static function fusionner(array $ini) {
107
	private static function fusionner(array $ini) {
100
		self::$parametres = array_merge(self::$parametres, $ini);
108
		self::$parametres = array_merge(self::$parametres, $ini);
101
	}
109
	}
102
 
110
 
103
	/**
111
	/**
104
	 * Renvoie la valeur demandée grâce une chaîne de paramètres
112
	 * Renvoie la valeur demandée grâce une chaîne de paramètres
105
	 * @param string $param la chaine de paramètres
113
	 * @param string $param la chaine de paramètres
106
	 * @param array $config le tableau de paramètre
114
	 * @param array $config le tableau de paramètre
107
	 * @return string la valeur de la chaine demandée
115
	 * @return string la valeur de la chaine demandée
108
	 */
116
	 */
109
	private static function getValeur($param, $config) {
117
	private static function getValeur($param, $config) {
110
		if ($param === null) {
118
		if ($param === null) {
111
			return null;
119
			return null;
112
		} else {
120
		} else {
113
			if (isset($config[$param])) {
121
			if (isset($config[$param])) {
114
				return $config[$param];
122
				return $config[$param];
115
			} else if (strpos($param, '.') !== false) {
123
			} else if (strpos($param, '.') !== false) {
116
				$pieces = explode('.', $param, 2);
124
				$pieces = explode('.', $param, 2);
117
				if (strlen($pieces[0]) && strlen($pieces[1])) {
125
				if (strlen($pieces[0]) && strlen($pieces[1])) {
118
					if (isset($config[$pieces[0]])) {
126
					if (isset($config[$pieces[0]])) {
119
					   if (is_array($config[$pieces[0]])) {
127
					   if (is_array($config[$pieces[0]])) {
120
					   		return self::getValeur($pieces[1], $config[$pieces[0]]);
128
					   		return self::getValeur($pieces[1], $config[$pieces[0]]);
121
					   }
129
					   }
122
					}
130
					}
123
				}
131
				}
124
			} else {
132
			} else {
125
				return null;
133
				return null;
126
			}
134
			}
127
		}
135
		}
128
	}
136
	}
129
 
137
 
130
	/**
138
	/**
131
	 * Teste si param existe dans le tableau config
139
	 * Teste si param existe dans le tableau config
132
	 * @param string $param nom du paramètre
140
	 * @param string $param nom du paramètre
133
	 * @param array tableau de configuration
141
	 * @param array tableau de configuration
134
	 */
142
	 */
135
	private static function existeValeur($param, $config) {
143
	private static function existeValeur($param, $config) {
136
		$retour = false;
144
		$retour = false;
137
		if (self::getValeur($param, $config) !== null) {
145
		if (self::getValeur($param, $config) !== null) {
138
			$retour = true;
146
			$retour = true;
139
		}
147
		}
140
		return $retour;
148
		return $retour;
141
	}
149
	}
142
 
150
 
143
	/**
151
	/**
144
	 * Vérifie si l'instance de classe à été crée, si non la crée
152
	 * Vérifie si l'instance de classe à été crée, si non la crée
145
	 */
153
	 */
146
	private static function verifierCreationInstance() {
154
	private static function verifierCreationInstance() {
147
		if (empty(self::$instance)) {
155
		if (empty(self::$instance)) {
148
			self::$instance = new Config();
156
			self::$instance = new Config();
149
		}
157
		}
150
	}
158
	}
151
 
159
 
152
	/**
160
	/**
153
	 * Analyse un tableau de paramètres.
161
	 * Analyse un tableau de paramètres.
154
	 * @param array $config le tableau de paramètres
162
	 * @param array $config le tableau de paramètres
155
	 * @return array le tableau analysé
163
	 * @return array le tableau analysé
156
	 */
164
	 */
157
	private static function analyserTableauIni($config = array()) {
165
	private static function analyserTableauIni($config = array()) {
158
		foreach ($config as $cle => &$valeur) {
166
		foreach ($config as $cle => &$valeur) {
159
			if (is_array($valeur)) {
167
			if (is_array($valeur)) {
160
				$config[$cle] = self::analyserTableauIni($valeur);
168
				$config[$cle] = self::analyserTableauIni($valeur);
161
			} else {
169
			} else {
162
				self::evaluerReferences($config, $cle);
170
				self::evaluerReferences($config, $cle);
163
				self::evaluerPhp($config, $cle);
171
				self::evaluerPhp($config, $cle);
164
				self::evaluerCle($config, $cle, $config[$cle]);
172
				self::evaluerCle($config, $cle, $config[$cle]);
165
			}
173
			}
166
			self::fusionner($config);
174
			self::fusionner($config);
167
		}
175
		}
168
		return $config;
176
		return $config;
169
	}
177
	}
170
 
178
 
171
	/**
179
	/**
172
	 * Dans le cas des chaine de configuration à sous clé (ex.: cle.souscle)
180
	 * Dans le cas des chaine de configuration à sous clé (ex.: cle.souscle)
173
	 * évalue les valeurs correspondantes et crée les sous tableaux associés.
181
	 * évalue les valeurs correspondantes et crée les sous tableaux associés.
174
	 * @param array $config tableau de configuration (par référence)
182
	 * @param array $config tableau de configuration (par référence)
175
	 * @param string $cle la cle dans le tableau
183
	 * @param string $cle la cle dans le tableau
176
	 * @param string $valeur la valeur à affecter
184
	 * @param string $valeur la valeur à affecter
177
	 */
185
	 */
178
	private static function evaluerCle(&$config, $cle, $valeur) {
186
	private static function evaluerCle(&$config, $cle, $valeur) {
179
		if (strpos($cle, '.') !== false) {
187
		if (strpos($cle, '.') !== false) {
180
			unset($config[$cle]);
188
			unset($config[$cle]);
181
			$pieces = explode('.', $cle, 2);
189
			$pieces = explode('.', $cle, 2);
182
			if (strlen($pieces[0]) && strlen($pieces[1])) {
190
			if (strlen($pieces[0]) && strlen($pieces[1])) {
183
				if (isset($config[$pieces[0]]) && !is_array($config[$pieces[0]])) {
191
				if (isset($config[$pieces[0]]) && !is_array($config[$pieces[0]])) {
184
					$m = "Ne peut pas créer de sous-clé pour '{$pieces[0]}' car la clé existe déjà";
192
					$m = "Ne peut pas créer de sous-clé pour '{$pieces[0]}' car la clé existe déjà";
185
					trigger_error($m, E_USER_WARNING);
193
					trigger_error($m, E_USER_WARNING);
186
				} else {
194
				} else {
187
					$config[$pieces[0]][$pieces[1]] = $valeur;
195
					$config[$pieces[0]][$pieces[1]] = $valeur;
188
					$config[$pieces[0]] = self::evaluerCle($config[$pieces[0]], $pieces[1], $valeur);
196
					$config[$pieces[0]] = self::evaluerCle($config[$pieces[0]], $pieces[1], $valeur);
189
				}
197
				}
190
			} else {
198
			} else {
191
				$m = "Clé invalide '$cle'";
199
				$m = "Clé invalide '$cle'";
192
				trigger_error($m, E_USER_WARNING);
200
				trigger_error($m, E_USER_WARNING);
193
			}
201
			}
194
		} else {
202
		} else {
195
			$config[$cle] = $valeur;
203
			$config[$cle] = $valeur;
196
		}
204
		}
197
		return $config;
205
		return $config;
198
	}
206
	}
199
 
207
 
200
	/**
208
	/**
201
	 * Évalue les valeurs de références à une clé dans le tableau config (cas du ref:cle).
209
	 * Évalue les valeurs de références à une clé dans le tableau config (cas du ref:cle).
202
	 * @param array $config tableau de configuration
210
	 * @param array $config tableau de configuration
203
	 * @param string $cle la clé dont il faut évaluer les références
211
	 * @param string $cle la clé dont il faut évaluer les références
204
	 */
212
	 */
205
	private static function evaluerReferences(&$config, $cle) {
213
	private static function evaluerReferences(&$config, $cle) {
206
		if (preg_match_all('/{ref:([A-Za-z0-9_.-]+)}/', $config[$cle], $correspondances,  PREG_SET_ORDER)) {
214
		if (preg_match_all('/{ref:([A-Za-z0-9_.-]+)}/', $config[$cle], $correspondances,  PREG_SET_ORDER)) {
207
			foreach ($correspondances as $ref) {
215
			foreach ($correspondances as $ref) {
208
				$config[$cle] = str_replace($ref[0], self::getValeur($ref[1], self::$parametres), $config[$cle]);
216
				$config[$cle] = str_replace($ref[0], self::getValeur($ref[1], self::$parametres), $config[$cle]);
209
			}
217
			}
210
		}
218
		}
211
	}
219
	}
212
 
220
 
213
	/**
221
	/**
214
	 * Évalue le code php contenu dans un clé tu tableau config.
222
	 * Évalue le code php contenu dans un clé tu tableau config.
215
	 * @param array $config tableau de configuration (par référence)
223
	 * @param array $config tableau de configuration (par référence)
216
	 * @param string $cle le clé du tableau dont il faut évaluer la valeur
224
	 * @param string $cle le clé du tableau dont il faut évaluer la valeur
217
	 */
225
	 */
218
	private static function evaluerPhp(&$config, $cle) {
226
	private static function evaluerPhp(&$config, $cle) {
219
		if (preg_match('/^php:(.+)$/', $config[$cle], $correspondances)) {
227
		if (preg_match('/^php:(.+)$/', $config[$cle], $correspondances)) {
220
			eval('$config["'.$cle.'"] = '.$correspondances[1].';');
228
			eval('$config["'.$cle.'"] = '.$correspondances[1].';');
221
		}
229
		}
222
	}
230
	}
223
 
231
 
224
	/**
232
	/**
225
	 * Charge un fichier ini dans le tableau des paramètres de l'appli.
233
	 * Charge un fichier ini dans le tableau des paramètres de l'appli.
226
	 * @param string $fichier_ini le nom du fichier à charger
234
	 * @param string $fichier_ini le nom du fichier à charger
227
	 * @return array le fichier ini parsé
235
	 * @return array le fichier ini parsé
228
	 */
236
	 */
229
	public static function charger($fichier_ini) {
237
	public static function charger($fichier_ini) {
230
		self::verifierCreationInstance();
238
		self::verifierCreationInstance();
231
		return self::parserFichierIni($fichier_ini);
239
		return self::parserFichierIni($fichier_ini);
232
	}
240
	}
233
 
241
 
234
	/**
242
	/**
235
	 * Accesseur pour la valeur d'un paramètre.
243
	 * Accesseur pour la valeur d'un paramètre.
236
	 * @param string $param le nom du paramètre
244
	 * @param string $param le nom du paramètre
237
	 * @return string la valeur du paramètre
245
	 * @return string la valeur du paramètre
238
	 */
246
	 */
239
	public static function get($param = null) {
247
	public static function get($param = null) {
240
		self::verifierCreationInstance();
248
		self::verifierCreationInstance();
241
		return self::getValeur($param, self::$parametres);
249
		return self::getValeur($param, self::$parametres);
242
	}
250
	}
243
 
251
 
244
	/**
252
	/**
245
	 * Vérifie si la valeur d'un paramètre existe.
253
	 * Vérifie si la valeur d'un paramètre existe.
246
	 * @param string $param le nom du paramètre
254
	 * @param string $param le nom du paramètre
247
	 * @return boolean vrai si le paramètre existe, false sinon
255
	 * @return boolean vrai si le paramètre existe, false sinon
248
	 */
256
	 */
249
	public static function existe($param) {
257
	public static function existe($param) {
250
		self::verifierCreationInstance();
258
		self::verifierCreationInstance();
251
		return self::existeValeur($param, self::$parametres);
259
		return self::existeValeur($param, self::$parametres);
252
	}
260
	}
253
 
261
 
254
	/**
262
	/**
255
	 * Vérifie que tous les paramêtres de config nécessaires au fonctionnement d'une classe existe dans les fichiers
263
	 * Vérifie que tous les paramêtres de config nécessaires au fonctionnement d'une classe existe dans les fichiers
256
	 * de configurations.
264
	 * de configurations.
257
	 * L'utilisation de cette méthode depuis la classe Config évite de faire appel à une classe supplémentaire.
265
	 * L'utilisation de cette méthode depuis la classe Config évite de faire appel à une classe supplémentaire.
258
	 *
266
	 *
259
	 * @param array $parametres tableau des noms des paramètres de la config à verifier.
267
	 * @param array $parametres tableau des noms des paramètres de la config à verifier.
260
	 * @return boolean true si tous les paramétres sont présents sinon false.
268
	 * @return boolean true si tous les paramétres sont présents sinon false.
261
	 */
269
	 */
262
	public static function verifierPresenceParametres(Array $parametres) {
270
	public static function verifierPresenceParametres(Array $parametres) {
263
		$ok = true;
271
		$ok = true;
264
		foreach ($parametres as $param) {
272
		foreach ($parametres as $param) {
265
			if (is_null(self::get($param))) {
273
			if (is_null(self::get($param))) {
266
				$classe = get_class();
274
				$classe = get_class();
267
				$m = "L'utilisation de la classe $classe nécessite de définir '$param' dans un fichier de configuration.";
275
				$m = "L'utilisation de la classe $classe nécessite de définir '$param' dans un fichier de configuration.";
268
				trigger_error($m, E_USER_ERROR);
276
				trigger_error($m, E_USER_ERROR);
269
				$ok = false;
277
				$ok = false;
270
			}
278
			}
271
		}
279
		}
272
		return $ok;
280
		return $ok;
273
	}
281
	}
274
}
282
}
275
?>
283
?>