Subversion Repositories eFlore/Applications.del

Rev

Rev 1345 | Go to most recent revision | Details | Compare with Previous | Last modification | View Log | RSS feed

Rev Author Line No. Line
709 gduche 1
<?php
2
// declare(encoding='UTF-8');
3
/**
4
	* Le web service image récupère toutes les données de la table del_obs_images
5
 * pour retourner une liste d'images associée à une observation
6
 *
7
 * @category	php 5.2
8
 * @package	del
9
 * @subpackage images
10
 * @author		Jean-Pascal MILCENT <jpm@tela-botanica.org>
11
 * @copyright	Copyright (c) 2012, Tela Botanica (accueil@tela-botanica.org)
12
 * @license	http://www.cecill.info/licences/Licence_CeCILL_V2-fr.txt Licence CECILL
13
 * @license	http://www.gnu.org/licenses/gpl.html Licence GNU-GPL
14
 * @version	$Id: Bdd.php 403 2012-02-22 14:35:20Z gduche $
15
 * @see http://www.tela-botanica.org/wikini/eflore/wakka.php?wiki=ApiIdentiplante01Images
16
 */
17
 
1326 gduche 18
 
19
/**
20
 * FONCTION TEMPORAIRE de debug pour afficher le contenu d'une variable en format lisible
21
 * @param $r la variable à afficher
22
 * */
23
function debug($r) {
24
	echo '<pre>'.print_r($r, true).'</pre>';
25
}
26
 
27
 
28
/**
29
 * Le service ListeImages récupère les données des tables observation et images
30
 * et les mets au format JSON pour identiplante / pictoflora
31
 * */
751 delphine 32
class ListeImages {
709 gduche 33
 
1326 gduche 34
 
35
	// Variables :
36
	// Configuration générale du service
719 gduche 37
	private $conteneur;
725 gduche 38
	private $navigation;
1326 gduche 39
	private $bdd;
725 gduche 40
	private $gestionBdd;
1326 gduche 41
 
42
	// Parametres
43
	private $ressources = array();
759 delphine 44
	private $parametres = array();
1326 gduche 45
	private $masque;
46
 
800 aurelien 47
	private $tri = 'date_transmission';
48
	private $directionTri = 'desc';
1345 aurelien 49
	private $formatRetour = 'XL';
709 gduche 50
 
1326 gduche 51
	private $imageIds = array();
52
 
53
 
54
	/**
55
	 * Constructeur de l'application
56
	 * Initialisation des variables générale de l'application
57
	 * @param Conteneu $conteneur le conteneur de classes de l'application
58
	 * */
719 gduche 59
	public function __construct(Conteneur $conteneur = null) {
60
		$this->conteneur = $conteneur == null ? new Conteneur() : $conteneur;
791 aurelien 61
		$this->conteneur->chargerConfiguration('config_departements_bruts.ini');
798 aurelien 62
		$this->conteneur->chargerConfiguration('config_mapping_votes.ini');
775 aurelien 63
		$this->conteneur->chargerConfiguration('config_images.ini');
725 gduche 64
		$this->navigation = $conteneur->getNavigation();
65
		$this->masque = $conteneur->getMasque();
66
		$this->gestionBdd = $conteneur->getGestionBdd();
775 aurelien 67
		$this->bdd = $this->gestionBdd->getBdd();
719 gduche 68
	}
69
 
1326 gduche 70
 
709 gduche 71
	/**
72
	 * Méthode principale de la classe.
1326 gduche 73
	 * Lance la récupération des images dans la base et les place dans un objet ResultatService
709 gduche 74
	 * pour l'afficher.
75
	 * @param array $ressources les ressources situées après l'url de base (ex : http://url/ressource1/ressource2)
76
	 * @param array $parametres les paramètres situés après le ? dans l'url
77
	 * */
78
	public function consulter($ressources, $parametres) {
1326 gduche 79
 
759 delphine 80
		$this->initialiserRessourcesEtParametres($ressources, $parametres);
709 gduche 81
 
82
		$this->configurer();
711 gduche 83
		$this->verifierConfiguration();
1326 gduche 84
 
800 aurelien 85
		$this->verifierParametresTri();
86
		$this->initialiserTri();
1345 aurelien 87
 
88
		$this->verifierParametreFormatRetour();
89
		$this->initialiserFormatRetour();
1326 gduche 90
 
91
		$parametres = $this->nettoyerParametres($parametres);
92
 
93
		// En fonction des paramètres de recherche, on n'effectue
94
		// pas la même requête, pour optimiser les jointures et les
95
		// rapidités d'éxécution.
96
		$type = $this->getTypeRequete($parametres);
97
 
98
		switch ($type) {
99
			case 'obs' :
100
				$liaisons = $this->chargerLiaisonsObs();
101
			break;
102
			case 'images' :
103
				$liaisons = $this->chargerLiaisonsImages();
104
			break;
105
			case 'obs-images' :
106
				$liaisons = $this->chargerLiaisons();
107
			break;
108
			default : //case simple !
109
			$liaisons = $this->chargerLiaisonsSimple();
110
		}
111
 
112
 
113
		// Partie commune à tous les cas : on complète les liaisons avec les informations des votes
114
		// et des images, puis on affiche sous forme de JSON
1321 gduche 115
		$images = $this->chargerImage($liaisons);
116
		$images = $this->chargerVotes($images);
117
 
118
		$resultat = new ResultatService();
119
		$resultat->corps = array('entete' => $this->conteneur->getEntete(), 'resultats' => $images);
1326 gduche 120
 
1321 gduche 121
		return $resultat;
709 gduche 122
	}
123
 
800 aurelien 124
 
125
 
1326 gduche 126
	/**************************************************************************************
127
	 *								FONCTION LIEES AUX REQUETES 						  *
128
	 **************************************************************************************/
709 gduche 129
 
130
	/**
1326 gduche 131
	 * Charger la clause WHERE en fonction des paramètres de masque
710 gduche 132
	 * */
1326 gduche 133
	private function chargerClauseWhere() {
728 gduche 134
 
709 gduche 135
		$where = array();
725 gduche 136
		$tableauMasque = $this->masque->getMasque();
137
		if (!empty($tableauMasque)) {
138
			foreach($tableauMasque as $idMasque => $valeurMasque) {
1326 gduche 139
				//TODO: scinder ceci en fonctions réutilisables ?
140
				// voir si c'est interessant par rapport à la recherche générale
725 gduche 141
				$idMasque = str_replace('masque.', '', $idMasque);
142
				switch ($idMasque) {
1326 gduche 143
				// nom du masque => nom BDD
725 gduche 144
					case 'auteur' :
1326 gduche 145
							$whereAuteur = ' '.$this->creerFiltreAuteur($this->masque->getMasque('auteur'));
146
									if($whereAuteur != '') {
147
										$where[] = $whereAuteur;
148
									}
149
									break;
791 aurelien 150
					case 'date' :
1326 gduche 151
					$whereDate = ' '.$this->creerFiltreDate($valeurMasque);
152
					if($whereDate != '') {
153
						$where[] = $whereDate;
154
					}
155
					break;
791 aurelien 156
					case 'departement' :
157
						$where[] = ' '.$this->creerFiltreIdZoneGeo($valeurMasque);
158
						break;
725 gduche 159
					case 'genre' :
719 gduche 160
						$where[] = ' '.$this->mappingFiltre['ns'].' LIKE '.$this->proteger($valeurMasque.' %');
725 gduche 161
						break;
162
					case 'tag' :
791 aurelien 163
						$where[] = ' '.$this->creerFiltreMotsCles($valeurMasque);
725 gduche 164
						break;
1365 aurelien 165
					case 'referentiel' :
166
						$where[] = ' referentiel_nom LIKE '.$this->proteger($valeurMasque.'%');
167
						break;
775 aurelien 168
					case 'ns' :
791 aurelien 169
						$where[] = ' nom_sel LIKE '.$this->proteger($valeurMasque.'%');
1326 gduche 170
						break;
1332 aurelien 171
					case 'nn' :
172
						$where[] = ' (nom_sel_nn LIKE '.$this->proteger($valeurMasque.'%').' OR '.
173
								   ' nom_ret_nn LIKE '.$this->proteger($valeurMasque.'%').') ';
174
						break;
791 aurelien 175
					case 'commune' :
176
						$where[] = ' '.$this->mappingFiltre[$idMasque].' LIKE '.$this->proteger(str_replace(array('-',' '), '_', $valeurMasque).'%');
1326 gduche 177
						break;
791 aurelien 178
					case 'masque' :
179
						$where[] = ' '.$this->creerFiltreMasqueGeneral($valeurMasque);
775 aurelien 180
						break;
725 gduche 181
					default:
182
						$where[] = ' '.$this->mappingFiltre[$idMasque].' = '.$this->proteger($valeurMasque);
1326 gduche 183
						break;
725 gduche 184
				}
1326 gduche 185
			}
709 gduche 186
		}
1326 gduche 187
 
709 gduche 188
		if (!empty($where)) {
189
			return ' WHERE '.implode('AND', $where);
190
		} else {
191
			return;
192
		}
193
	}
194
 
1326 gduche 195
	/**
196
	 * Créer un masque général lorsque l'on souhaite utiliser le passe partout
197
	 * @param la valeur du passe partout
198
	 * */
791 aurelien 199
	private function creerFiltreMasqueGeneral($valeurMasque) {
1326 gduche 200
		//TODO: affecter d'aborder les variables, puis les tester pour les
791 aurelien 201
		// ajouter à la chaine
202
		$whereAuteur = $this->creerFiltreAuteur($valeurMasque);
1326 gduche 203
		$whereIdZoneGeo = $this->creerFiltreIdZoneGeo($valeurMasque);
204
 
791 aurelien 205
		$masqueGeneral = '( '.
1326 gduche 206
				(($whereAuteur != '') ? $whereAuteur.' OR ' : '' ).
207
				(($whereIdZoneGeo != '') ? $whereIdZoneGeo.' OR ' : '' ).
208
				'zone_geo LIKE '.$this->proteger($this->remplacerParJokerCaractere($valeurMasque).'%').' OR '.
209
				$this->creerFiltreMotsCles($valeurMasque).' OR '.
210
				'nom_sel LIKE '.$this->proteger($valeurMasque.'%').' OR '.
211
				'famille LIKE '.$this->proteger($valeurMasque.'%').' OR '.
212
				'milieu LIKE '.$this->proteger($valeurMasque).' OR '.
213
				$this->mappingFiltre['ns'].' LIKE '.$this->proteger('%'.$valeurMasque.'% %').' OR '.
214
				$this->creerFiltreDate($valeurMasque).
215
				') ';
791 aurelien 216
 
217
		return $masqueGeneral;
218
	}
219
 
1326 gduche 220
	/**
221
	 * Créer le filtre auteur en recherchant dans son nom, prénom, adresse email en fonction
222
	 * de la chaine donnée
223
	 * @param la valeur recherchée
224
	 * */
791 aurelien 225
	private function creerFiltreAuteur($valeurMasque) {
226
		$masque = '';
227
		$auteurId = $valeurMasque;
228
		if (is_numeric($auteurId)) {
229
			$masque = ' ce_utilisateur = '.$auteurId;
230
		} else {
231
			if (strpos($auteurId, '@') === false) {
232
				$tableauNomPrenom = explode(' ',$auteurId, 2);
233
				if(count($tableauNomPrenom) == 2) {
234
					// on teste potentiellement un nom prenom ou bien un prénom nom
235
					$masque = '('.
1326 gduche 236
							'(nom LIKE '.$this->proteger($tableauNomPrenom[0].'%').' AND '.
237
							'prenom LIKE '.$this->proteger($tableauNomPrenom[1].'%').') OR '.
238
							'(nom LIKE '.$this->proteger($tableauNomPrenom[1].'%').' AND '.
239
							'prenom LIKE '.$this->proteger($tableauNomPrenom[0].'%').') OR '.
240
							'(dob.nom_utilisateur LIKE '.$this->proteger($tableauNomPrenom[0].'%').' AND '.
241
							'dob.prenom_utilisateur LIKE '.$this->proteger($tableauNomPrenom[1].'%').') OR '.
242
							'(dob.nom_utilisateur LIKE '.$this->proteger($tableauNomPrenom[1].'%').' AND '.
243
							'dob.prenom_utilisateur LIKE '.$this->proteger($tableauNomPrenom[0].'%').') OR '.
244
							'(nom LIKE '.$this->proteger($valeurMasque.'%').') OR '.
245
							'(prenom LIKE '.$this->proteger($valeurMasque.'%').') OR '.
246
							'(dob.nom_utilisateur LIKE '.$this->proteger($valeurMasque.'%').') OR '.
247
							'(dob.prenom_utilisateur LIKE '.$this->proteger($valeurMasque.'%').') '.
248
							')';
791 aurelien 249
				} else {
250
					$masque = '(
251
						            (nom LIKE '.$this->proteger($auteurId.'%').' OR '.
1326 gduche 252
							            'prenom LIKE '.$this->proteger($auteurId.'%').' OR '.
253
							            'dob.nom_utilisateur LIKE '.$this->proteger($auteurId.'%').' OR '.
254
							            'dob.prenom_utilisateur LIKE '.$this->proteger($auteurId.'%').')'.
255
							            ')';
791 aurelien 256
				}
257
			} else {
1333 gduche 258
				$masque = " ( courriel LIKE ".$this->proteger($valeurMasque.'%').
259
				" OR dob.courriel_utilisateur LIKE ".$this->proteger($valeurMasque.'%').")  ";
791 aurelien 260
			}
261
		}
262
		return $masque;
263
	}
264
 
265
 
1326 gduche 266
	/**
267
	 * Obtenir une chaine de caractère concaténant nom et prénom séparé par une virgule
268
	 * @param String $auteurId l'identifiant de l'auteur
269
	 * @return String la chaine de concaténation
270
	 * */
271
	private function getChaineNomPrenom($auteurId) {
272
		$nomPrenom = explode(' ', $auteurId);
273
		$nomPrenom = $this->proteger($nomPrenom);
274
		$chaineNomPrenom = implode(', ', $nomPrenom);
275
		return $chaineNomPrenom;
791 aurelien 276
	}
277
 
1326 gduche 278
	/**
279
	 * Créer le filtre de recherche par zone géographique en fonction du masque
280
	 * @param $valeurMasque le terme de la recherche
281
	 * */
791 aurelien 282
	private function creerFiltreIdZoneGeo($valeurMasque) {
283
		$masque = '';
284
		$dept = $valeurMasque;
285
		if (is_numeric($dept)) {
286
			$dept = sprintf('%02s', $dept);
287
			$dept = sprintf("%-'_5s", $dept);
288
			$masque = " ce_zone_geo LIKE ".$this->proteger('INSEE-C:'.$dept);
289
		} else {
290
			$deptId = $this->conteneur->getParametre($dept);
291
			if ($deptId != null) {
292
				$masque = " ce_zone_geo LIKE ".$this->proteger('INSEE-C:'.$deptId.'%');
293
			} else {
294
				$id = $this->obtenirIdDepartement($valeurMasque);
295
				$masque = " ce_zone_geo LIKE ".$this->proteger('INSEE-C:'.$id.'%');
296
			}
297
		}
298
		return $masque;
299
	}
300
 
1326 gduche 301
	/**
302
	 * Générer la chaine de recherche pour la date en  fonction du masque
303
	 * @param $valeurMasque la date recherchée (AAAA ou JJ/MM/AAAA)
304
	 * */
791 aurelien 305
	private function creerFiltreDate($valeurMasque) {
306
		//TODO: définir dans le fichier de config un tableau contenant plusieurs format de date
307
		// autorisés pour la recherche, qui seraient ajoutés au OR
308
		$masque = '(';
309
		$masque .= (is_numeric($valeurMasque)) ? ' YEAR(date_observation) = '.$this->proteger($valeurMasque).' OR ' : '';
310
		$masque .= " DATE_FORMAT(date_observation, '%d/%m/%Y') = ".$this->proteger($valeurMasque).' '.
1326 gduche 311
				')';
791 aurelien 312
		return $masque;
313
	}
801 aurelien 314
 
1326 gduche 315
	/**
316
	 * Générer la chaine de recherche dans les mots clés en fonction du masque
317
	 * @param $valeurMasque le mot clé recherché
318
	 * */
791 aurelien 319
	private function creerFiltreMotsCles($valeurMasque) {
1326 gduche 320
 
800 aurelien 321
		$mots_cles = explode(' ', $valeurMasque);
812 gduche 322
		$requeteMotsClesImg = array();
323
		$requeteMotsClesObs = array();
1180 aurelien 324
		$requeteMotsClesImgPublic = array();
1326 gduche 325
 
800 aurelien 326
		foreach($mots_cles as $mot_cle) {
1180 aurelien 327
			//TODO: rechercher sur les mots clés normalisés dans tous les cas ?
857 gduche 328
			$requeteMotsCles = $this->proteger('%'.$mot_cle.'%');
1180 aurelien 329
			$motsCleProtege = $this->proteger($this->normaliserMotCle('%'.$mot_cle.'%'));
1326 gduche 330
			$requeteMotsClesImgPublic[] = 'di.id_image IN (SELECT ce_image FROM del_image_tag WHERE tag_normalise LIKE '.$motsCleProtege.' AND actif = 1)';
857 gduche 331
			$requeteMotsClesImg[] = 'di.mots_cles_texte LIKE '.$requeteMotsCles;
332
			$requeteMotsClesObs[] = 'dob.mots_cles_texte LIKE '.$requeteMotsCles;
800 aurelien 333
		}
1326 gduche 334
 
1180 aurelien 335
		$requeteMotsClesImgPublic = implode(' AND ', $requeteMotsClesImgPublic);
857 gduche 336
		$requeteMotsClesImg = implode(' AND ', $requeteMotsClesImg);
337
		$requeteMotsClesObs = implode(' AND ', $requeteMotsClesObs);
1326 gduche 338
 
800 aurelien 339
		$masque = '('.
1326 gduche 340
				'('.$requeteMotsClesImgPublic.') OR '.
341
				'('.$requeteMotsClesImg.') OR '.
342
				'('.$requeteMotsClesObs.') '.
343
				')';
791 aurelien 344
 
1326 gduche 345
		return $masque;
709 gduche 346
	}
347
 
1326 gduche 348
	// ??
1168 aurelien 349
	private function assemblercomptageOccurencesMotsClesCel() {
350
		$chaineMotsClesAffiches = $this->conteneur->getParametre('mots_cles_cel_affiches');
351
		$tabMotsClesAffiches = explode(',',$chaineMotsClesAffiches);
352
		$chaineSql = '';
1326 gduche 353
 
1168 aurelien 354
		// Comptage du nombre de mots clés officiels présents dans la chaine mots clés texte
355
		foreach ($tabMotsClesAffiches as $motCle) {
356
			if($chaineSql != '') {
357
				$chaineSql .= ' + ';
358
			}
359
			$chaineSql .= 'IF(FIND_IN_SET('.$this->proteger($motCle).',di.mots_cles_texte) != 0, 1, 0)';
360
		}
1326 gduche 361
 
1168 aurelien 362
		return '('.$chaineSql.')';
363
	}
364
 
1326 gduche 365
	private function getTri() {
800 aurelien 366
		$order = '';
951 aurelien 367
		if($this->doitJoindreTableVotes()) {
368
			$order = ' GROUP BY dvote.ce_image, dob.id_observation ORDER BY total_votes '.$this->directionTri.', date_transmission desc ';
1168 aurelien 369
		} else if($this->doitJoindreTableTags()) {
370
			$order = ' GROUP BY doi.id_image ORDER BY total_tags '.$this->directionTri.', date_transmission desc ';
951 aurelien 371
		} else {
800 aurelien 372
			$order = ' ORDER BY '.$this->tri.' '.$this->directionTri.' ';
373
		}
374
		return $order;
375
	}
376
 
709 gduche 377
	/**
1326 gduche 378
	 * Compter le nombre total d'images dans la base pour affichage dans entete.
379
	 * */
380
	private function getFoundRows() {
709 gduche 381
		$requete = 'SELECT FOUND_ROWS() AS nbre ';
719 gduche 382
		$resultats = $this->bdd->recuperer($requete);
709 gduche 383
		return (int) $resultats['nbre'];
384
	}
385
 
1326 gduche 386
 
709 gduche 387
	/**
1326 gduche 388
	 * En fonction des paramètres, générer les conditions de recherche
389
	 * des observations
390
	 * */
391
	private function getConditionsObs() {
1327 gduche 392
		$conditionsObs = array();
1326 gduche 393
		$masques = $this->masque->getMasque();
394
		if (isset($masques['masque'])) {
395
			$passe = $masques['masque'];
396
 
397
			// Si on a saisi le masque passe partout, alors on doit chercher dans tous les champs
398
			// de la table observation (OR)
399
			$conditionLibre = array();
400
 
401
			if (!isset($masques['masque.ns'])) {
402
				$conditionsLibre[] = "nom_sel LIKE '$passe%'";
403
			}
404
 
405
			if (!isset($masques['masque.famille'])) {
406
				$conditionsLibre[] = "famille LIKE '$passe%'";
407
			}
408
 
409
			if (!isset($masques['masque.milieu'])) {
410
				$conditionsLibre[] = "nom_sel LIKE '$passe%'";
411
			}
412
 
413
			if (!isset($masques['masque.date'])) {
414
				$conditionsLibre[] = $this->creerFiltreDate($passe);
415
			}
416
 
417
			if (!isset($masques['masque.auteur'])) {
418
				$conditionsLibre[] = $this->creerFiltreAuteur($passe);
419
			}
420
 
421
			/*
422
			 * FIXME : remplacer par motcle projet !
423
			* if (!isset($masques['masque.tag'])) {
424
			$conditionsLibre[] = "mots_cles_texte LIKE '%$passe%'";
425
			}*/
426
 
427
			$conditionsObs[] = implode(' OR ', $conditionsLibre);
428
		}
429
 
430
		// nom sel
431
		if (isset($masques['masque.ns'])) {
432
			$nom_sel = $masques['masque.ns'];
433
			$conditionsObs[] = "nom_sel LIKE '$nom_sel%'";
434
		}
435
 
1332 aurelien 436
		// nom sel
437
		if (isset($masques['masque.nn'])) {
438
			$num_nom = $masques['masque.nn'];
439
			$conditionsObs[] = "(nom_sel_nn = '$num_nom' OR nom_ret_nn = '$num_nom') ";
440
		}
441
 
1326 gduche 442
		// famille
443
		if (isset($masques['masque.famille'])) {
444
			$famille = $masques['masque.famille'];
445
			$conditionsObs[] = "famille LIKE '$famille%'";
446
		}
447
 
448
		// genre
449
		if (isset($masques['masque.genre'])) {
450
			$genre = $masques['masque.genre'];
451
			$conditionsObs[] = "nom_sel LIKE '$genre%'";
452
		}
453
 
454
		// milieu
455
		if (isset($masques['masque.milieu'])) {
456
			$milieu = $masques['masque.milieu'];
457
			$conditionsObs[] = "nom_sel LIKE '$milieu%'";
458
		}
459
 
460
		// date
461
		if (isset($masques['masque.date'])) {
462
			$date = $masques['masque.date'];
463
			$conditionsObs[] = $this->creerFiltreDate($date);
464
		}
465
 
466
		// utilisateur
467
		if (isset($masques['masque.auteur'])) {
468
			$auteur = $masques['masque.auteur'];
469
			$conditionsObs[] = $this->creerFiltreAuteur($auteur);
470
		}
471
 
472
		// commune
473
		if (isset($masques['masque.commune'])) {
474
			$commune = $masques['masque.commune'];
475
			$conditionsObs[] = " zone_geo LIKE ".$this->proteger(str_replace(array('-',' '), '_', $commune).'%');
476
		}
477
 
478
		// commune
479
		if (isset($masques['masque.departement'])) {
480
			$dept = $masques['masque.departement'];
481
			$conditionsObs[] = $this->creerFiltreIdZoneGeo($dept);
482
		}
1332 aurelien 483
 
1326 gduche 484
		return $conditionsObs;
485
	}
486
 
487
	/**
488
	 * Obtenir le tableu de chaines de condition de requete images en fonction des masques
489
	 * */
490
	private function getConditionsImages() {
491
		$conditionsImg = array();
492
		$masques = $this->masque->getMasque();
493
		if (isset($masques['masque.tag'])) {
494
			$tag = $masques['masque.tag'];
495
			$conditionsImg[] = " dit.tag_normalise LIKE '$tag%' ";
496
			$conditionsImg[] = " di.mots_cles_texte LIKE '%$tag%' ";
497
		}
498
 
499
		return $conditionsImg;
500
	}
501
 
502
 
503
	/*-------------------------------------------------------------------------------
504
	 CHARGEMENT DES IMAGES
505
	--------------------------------------------------------------------------------*/
506
	/**
507
	 * Chargement depuis la bdd de toutes les liaisons entre images et observations
508
	 * Méthode appelée uniquement lorsque les paramètres sont vides
509
	 * */
510
	private function chargerLiaisonsSimple() {
511
 
512
		// On récupère d'abord les N images de del_obs_image, éventuellement triées,
513
		// Et on complète avec les informations des observations associées
514
		$requeteImages = ' SELECT *, di.mots_cles_texte as mots_cles_texte_image '.
515
						 ' FROM del_obs_image doi '.
516
						 ' INNER JOIN del_image di ON doi.id_image = di.id_image ';
517
 
518
		// Si le tri se fait par date d'observation, on récupère les identifiants de N observations triées
519
		if (isset($this->parametres['tri']) && $this->parametres['tri'] == 'date_observation') {
520
 
521
			$ordre = isset($this->parametres['ordre']) ? $this->parametres['ordre'] : 'DESC';
522
 
523
			$requeteIdObs = ' SELECT doi.id_image  as id_image '.
524
							' FROM del_obs_image doi '.
525
							' INNER JOIN del_observation dob ON dob.id_observation = doi.id_observation '.
526
							' INNER JOIN del_image di ON doi.id_image = di.id_image '.
527
							' ORDER BY date_observation '.$ordre.', dob.id_observation '.$ordre;
528
			$requeteIdObs .= $this->gestionBdd->getLimitSql();
529
 
530
			// Récupérer les N observations triées par date
531
			$observations = $this->bdd->recupererTous($requeteIdObs);
532
 
533
			$idsImages = array();
534
			foreach ($observations as $observation) {
535
				$idsImages[] = $observation['id_image'];
536
			}
537
 
538
			$chaineIdImages = implode(',', $idsImages);
539
			$requeteImages .= ' WHERE doi.id_image IN ('.$chaineIdImages.') '.
1328 aurelien 540
							  ' GROUP BY doi.id_image, doi.id_observation '.
1326 gduche 541
							  ' ORDER BY FIELD(doi.id_image,  '.$chaineIdImages.')'.
542
							  ' LIMIT '.$this->navigation->getLimite(); // On limite sur le nombre car les obs peuvent avoir plusieurs images
543
 
544
		} else {
1328 aurelien 545
			$requeteImages .= ' GROUP BY doi.id_image, doi.id_observation ';
1326 gduche 546
			$requeteImages .= ' ORDER BY id_observation DESC';
547
			$requeteImages .= $this->gestionBdd->getLimitSql();
548
		}
549
 
550
		$liaisons = $this->bdd->recupererTous($requeteImages);
551
 
552
		// Ce n'est pas la peine de continuer s'il n'y a pas eu de résultats dans la table del_obs_images
553
		if (!empty($liaisons)) {
554
 
555
			$idsObservations = array();
556
			foreach ($liaisons as $image) {
557
				$idObs = $image['id_observation'];
558
				$idsObservations[$idObs] = $idObs;
559
			}
560
 
561
			$chaineIdObs = implode(',', $idsObservations);
562
 
563
 
564
			// On récupère les observations qui complètent la requête précédente
565
			$requeteObservations = ' SELECT * '.
566
								   ' FROM del_observation dob '.
567
								   ' LEFT JOIN del_utilisateur du ON dob.ce_utilisateur = du.id_utilisateur '.
568
								   ' WHERE id_observation IN ('.$chaineIdObs.')';
569
 
570
			$resultatsObservations = $this->bdd->recupererTous($requeteObservations);
571
 
572
			// FIXME : Ca ne doit pas arriver, mais que se passe-t-il s'il n'y a pas d'observation pour l'image ?!
573
 
574
			// On range les observations dans un tableau pour pouvoir les retrouver par leur id :
575
			$observations = array();
576
			foreach ($resultatsObservations as $id => $observation) {
577
				$idObs = $observation['id_observation'];
578
				$observations[$idObs] = $observation;
579
			}
580
 
581
			// Enfin, pour chaque image préalablement récupérées, on complète avec les informations de l'observation
582
			// FIXME : peut-être peut-on utiliser un array_merge ici ?
583
			foreach ($liaisons as $id => $liaison) {
584
				$idObs = $liaison['id_observation'];
585
 
586
				$observation = $observations[$idObs];
587
				foreach ($observation as $cle => $valeur) {
588
					$liaisons[$id][$cle] = $valeur;
589
				}
590
			}
591
 
592
 
593
			// On compte à part les images issues de la jointure de del_obs_image et del_image car la fonction
594
			// SQL_CALC_FOUND_ROWS dans la fonction requete image fait passer le temps d'éxécution de 0.0011 à 15s !
1328 aurelien 595
			$requeteNbImages = 'SELECT SUM(t.nb) as nb FROM (SELECT count(DISTINCT doi.id_image) as nb '.
596
							   'FROM del_obs_image doi '.
597
							   'INNER JOIN del_image di '.
598
							   'ON di.id_image = doi.id_image '.
599
							   'GROUP BY doi.id_image, doi.id_observation) t ';
1326 gduche 600
			$resultatNbImages = $this->bdd->recupererTous($requeteNbImages);
1328 aurelien 601
 
1326 gduche 602
			$total = (int) $resultatNbImages[0]['nb'];
603
			$this->navigation->setTotal($total);
604
		}
1328 aurelien 605
 
1326 gduche 606
		return $liaisons;
607
	}
608
 
609
 
610
	/**
611
	 * Chargement depuis la bdd de toutes les liaisons entre images et observations
612
	 * Méthode appelée uniquement lorsque les paramètres concernent une observation
613
	 * */
614
	private function chargerLiaisonsObs() {
615
 
616
		// Récupérer les liaisons
1365 aurelien 617
		$requeteObs = ' SELECT SQL_CALC_FOUND_ROWS dob.id_observation as id_observation, nom_referentiel, nom_sel, nom_sel_nn, nt, famille, ce_zone_geo, zone_geo, lieudit, station, milieu, '.
1326 gduche 618
				 	  ' date_observation, dob.mots_cles_texte as mots_cles_texte, dob.commentaire as commentaire, di.mots_cles_texte as mots_cles_texte_image , date_transmission, '.
619
					  ' doi.id_image as id_image, di.ce_utilisateur as ce_utilisateur, prenom, nom, courriel, dob.prenom_utilisateur, dob.nom_utilisateur, dob.courriel_utilisateur, nom_original '.
620
					  'FROM del_observation dob '.
621
					  ' INNER JOIN del_obs_image doi ON dob.id_observation = doi.id_observation '.
622
					  ' INNER JOIN del_image di ON doi.id_image = di.id_image '.
623
					  ' LEFT JOIN del_utilisateur du ON dob.ce_utilisateur = du.id_utilisateur ';
624
 
625
		// Récupérer les conditions sous forme de tableau
626
		$conditionsObs = $this->getConditionsObs();
627
 
628
		if (!empty($conditionsObs)) {
629
			$where = ' WHERE '.implode(' AND ', $conditionsObs);
630
			$requeteObs .= $where;
631
		}
632
 
633
		// Gérer le tri (uniquement si c'est date_observation)
634
		if (isset($this->parametres['tri']) && $this->parametres['tri'] == 'date_observation') {
635
			$ordre = isset($this->parametres['ordre']) ? $this->parametres['ordre'] : 'DESC';
636
			$tri = ' ORDER BY '.$this->parametres['tri'].' '.$ordre.', doi.id_observation '.$ordre.' ';
637
			$requeteObs .= $tri;
638
		}
1332 aurelien 639
 
1326 gduche 640
		$requeteObs .= $this->gestionBdd->getLimitSql();
641
		$observations = $this->bdd->recupererTous($requeteObs);
1332 aurelien 642
 
1326 gduche 643
		$total = $this->getFoundRows();
644
		$this->navigation->setTotal($total);
645
 
646
		return $observations;
647
	}
648
 
649
	/**
650
	 * Chargement depuis la bdd de toutes les liaisons entre images et observations
651
	 * Méthode appelée uniquement lorsque les paramètres concernent les images
652
	 * */
653
	private function chargerLiaisonsImages() {
654
 
655
		// FIXME : si on faisait une requete à part pour compter, ca irait plus vite
656
		// Récupérer tous les ID d'image en fonction des paramètres de recherche
657
		$requeteImages = ' SELECT SQL_CALC_FOUND_ROWS '.
658
						 ' doi.id_image as id_image, dob.id_observation as id_observation, nom_sel, nom_sel_nn, nt, famille, ce_zone_geo, zone_geo, lieudit, station, milieu, '.
659
						 ' date_observation, dob.mots_cles_texte as mots_cles_texte, dob.commentaire as commentaire, di.mots_cles_texte as mots_cles_texte_image , date_transmission, '.
660
						 ' di.ce_utilisateur as ce_utilisateur, prenom, nom, courriel, dob.prenom_utilisateur, dob.nom_utilisateur, dob.courriel_utilisateur, nom_original '.
661
						 ' FROM del_obs_image doi '.
662
						 ' INNER JOIN del_image di ON doi.id_image = di.id_image '.
663
						 ' INNER JOIN del_observation dob ON dob.id_observation = doi.id_observation '.
664
						 ' LEFT JOIN del_image_tag dit ON dit.ce_image = di.id_image '.
665
						 ' LEFT JOIN del_utilisateur du ON du.id_utilisateur = di.ce_utilisateur ';
1328 aurelien 666
 
1326 gduche 667
		$conditionsImg = $this->getConditionsImages();
668
 
669
		if (!empty($conditionsImg)) {
670
			$where = ' WHERE ('.implode(' OR ', $conditionsImg).') ';
671
			$where .= ' AND dit.actif = 1 ';
672
			$requeteImages .= $where;
673
		}
674
 
675
		// Gérer le tri, sur les votes ou sur les tags
676
		if (isset($this->parametres['tri'])) {
677
 
678
			$chaineTri = '';
679
			$chaineOrdre = '';
680
 
681
			if ($this->parametres['tri'] == 'votes') {
682
 
683
				$protocole = isset($this->parametres['protocole']) ? $this->parametres['protocole'] : 1;
684
 
685
				$requeteVotes = ' SELECT doi.id_image as id_image, IF(divo.ce_protocole = '.$protocole.', AVG(divo.valeur), 0) as total_votes '.
1330 aurelien 686
								' FROM del_obs_image doi '.
687
								' INNER JOIN del_image di ON doi.id_image = di.id_image '.
688
								' INNER JOIN del_observation dob ON dob.id_observation = doi.id_observation '.
1331 gduche 689
								' LEFT JOIN del_image_vote divo ON doi.id_image = divo.ce_image '.
690
                                ' AND ce_protocole = '.$protocole.' ';
1326 gduche 691
 
692
				// Et si on a cherché par tag ?
693
				if (isset($this->parametres['masque.tag'])) {
694
					$tag = $this->parametres['masque.tag'];
695
					$requeteVotes .= ' LEFT JOIN del_image_tag dit ON dit.ce_image = di.id_image ';
696
					$requeteVotes .= " WHERE (dit.tag_normalise LIKE '$tag%' OR  di.mots_cles_texte LIKE '%$tag%') AND dit.actif = 1 ";
697
				}
698
 
699
				$requeteVotes .= ' GROUP BY doi.id_image, doi.id_observation '.
1330 aurelien 700
								 ' ORDER by total_votes '.$this->directionTri .', doi.id_observation '.$this->directionTri.' '.
1326 gduche 701
						$this->gestionBdd->getLimitSql();
702
 
1331 gduche 703
 
1326 gduche 704
				$resultatsVotes = $this->bdd->recupererTous($requeteVotes);
705
				$tabVotes = array();
706
				foreach ($resultatsVotes as $vote) {
707
					$tabVotes[] = $vote['id_image'];
708
				}
709
 
710
				$strVotes = empty($tabVotes) ? "''" : implode(',', $tabVotes);
711
 
712
				// Et si on a cherché par tag ?
713
				if (isset($this->parametres['masque.tag'])) {
714
					$chaineTri .= ' AND ';
715
				} else {
716
					$chaineTri .= ' WHERE ';
717
				}
718
 
719
 
720
				$chaineTri .= '  doi.id_image IN ('.$strVotes.') ';
721
				$chaineOrdre = ' ORDER BY FIELD(doi.id_image, '.$strVotes.')  ';
722
			}
1328 aurelien 723
 
1326 gduche 724
			if ($this->parametres['tri'] == 'tags') {
725
 
1328 aurelien 726
				$requetetags = ' SELECT SQL_CALC_FOUND_ROWS doi.id_image, COUNT(id_tag) as total_tags '.
1330 aurelien 727
						' FROM del_obs_image doi LEFT JOIN del_image_tag dit ON dit.ce_image = doi.id_image AND dit.actif = 1 '.
728
						 ' INNER JOIN del_image di ON doi.id_image = di.id_image '.
729
						 ' INNER JOIN del_observation dob ON dob.id_observation = doi.id_observation ';
730
 
1326 gduche 731
				if (isset($this->parametres['masque.tag'])) {
732
					// Et si on a cherché par tag ?
733
					$tag = $this->parametres['masque.tag'];
1328 aurelien 734
					$requetetags .= " WHERE (dit.tag_normalise LIKE '$tag%' OR  di.mots_cles_texte LIKE '%$tag%') ";
1326 gduche 735
				}
736
 
1330 aurelien 737
				$requetetags .= ' GROUP BY doi.id_image, doi.id_observation '.
738
						' ORDER by total_tags '.$this->directionTri.', doi.id_observation '.$this->directionTri.
1326 gduche 739
						$this->gestionBdd->getLimitSql();
1328 aurelien 740
 
1326 gduche 741
				$resultatstags = $this->bdd->recupererTous($requetetags);
742
				$tabtags = array();
743
				foreach ($resultatstags as $tag) {
744
					$tabtags[] = $tag['id_image'];
745
				}
746
				$strtags = empty($tabtags) ? "''" : implode(',', $tabtags);
747
 
748
 
749
				// Et si on a cherché par tag ?
750
				if (isset($this->parametres['masque.tag'])) {
751
					$chaineTri .= ' AND ';
752
				} else {
753
					$chaineTri .= ' WHERE ';
754
				}
755
 
756
				$chaineTri .= ' doi.id_image IN ('.$strtags.') ';
757
				$chaineOrdre = ' ORDER BY FIELD(doi.id_image, '.$strtags.') ';
758
			}
1328 aurelien 759
 
1326 gduche 760
			$requeteImages .= $chaineTri.' GROUP BY doi.id_image, doi.id_observation '.$chaineOrdre;
761
		} else {
762
			$requeteImages .= ' GROUP BY doi.id_image, doi.id_observation'; // des fois, on a plusieurs observations pour la même image ...
763
			$requeteImages .= $this->gestionBdd->getLimitSql();
764
		}
1328 aurelien 765
 
1326 gduche 766
		$retour  = $this->bdd->recupererTous($requeteImages);
767
		$total = $this->getFoundRows();
768
		$this->navigation->setTotal($total);
769
 
770
		return $retour;
771
	}
772
 
773
	/**
774
	 * Chargement depuis la bdd de toutes les liaisons entre images et observations
775
	 * */
776
	private function chargerLiaisons() {
777
 
778
		$champs = array('dob.id_observation as id_observation', 'nom_sel', 'nom_sel_nn', 'nt', 'famille', 'ce_zone_geo', 'zone_geo',
779
				'lieudit', 'station', 'milieu', 'date_observation', 'dob.mots_cles_texte as mots_cles_texte', 'dob.commentaire as commentaire',
780
				'di.mots_cles_texte as mots_cles_texte_image ', 'date_transmission', 'di.id_image as id_image', 'di.ce_utilisateur as ce_utilisateur',
781
				'prenom', 'nom', 'courriel', 'dob.prenom_utilisateur', 'dob.nom_utilisateur', 'dob.courriel_utilisateur', 'nom_original');
782
		// Attention le LEFT JOIN est indispensable pour ramener les images n'ayant pas de votes
783
		// en cas de tri par votes
784
		$requeteLiaisons = 'SELECT DISTINCT SQL_CALC_FOUND_ROWS '.implode(', ',$champs).' '.
785
				($this->doitJoindreTableVotes() ?
786
						', IF(dvote.ce_protocole = '.$this->parametres['protocole'].', AVG(dvote.valeur), 0) as total_votes ' :
787
						''
788
				).
789
				($this->doitJoindreTableTags() ?
790
						// attention le DISTINCT est indispensable !
791
						', (COUNT(DISTINCT dtag.id_tag) + '.$this->assemblercomptageOccurencesMotsClesCel().') as total_tags ' :
792
						''
793
				).
794
				'FROM '.$this->gestionBdd->formaterTable('del_obs_image', 'doi').
795
				'INNER JOIN del_image di '.
796
				'ON doi.id_image = di.id_image '.
797
				'INNER JOIN del_observation dob '.
798
				'ON doi.id_observation = dob.id_observation '.
799
				'LEFT JOIN del_utilisateur du '.
800
				'ON du.id_utilisateur = di.ce_utilisateur '.
801
				($this->doitJoindreTableTags() ?
802
						'LEFT JOIN del_image_tag dtag '.
803
						'ON doi.id_image = dtag.ce_image AND dtag.actif = 1 ' :
804
						''
805
								).
806
								($this->doitJoindreTableVotes() ?
807
								'LEFT JOIN del_image_vote dvote '.
808
								'ON doi.id_image = dvote.ce_image AND dvote.ce_protocole = '.$this->parametres['protocole'] :
809
								''
810
										);
1332 aurelien 811
				$requeteLiaisons .= $this->chargerClauseWhere();
812
				$requeteLiaisons .= $this->getTri();
813
				$requeteLiaisons .= $this->gestionBdd->getLimitSql();
814
				$retour = $this->bdd->recupererTous($requeteLiaisons);
815
				$total = $this->getFoundRows();
816
				$this->navigation->setTotal($total);
817
				return $retour;
1326 gduche 818
	}
819
 
820
 
821
	/**
822
	 * Retourner un tableau d'images formaté en fonction des liaisons trouvées
823
	 * @param $liaisons les liaisons de la table del_obs_images
824
	 * */
709 gduche 825
	private function chargerImage($liaisons) {
826
 
827
		$images = array();
828
		foreach ($liaisons as $liaison) {
736 gduche 829
			$idImage = $liaison['id_image'];
1326 gduche 830
 
1274 aurelien 831
			if($liaison['ce_utilisateur'] == 0) {
1326 gduche 832
				$liaison['prenom'] = $liaison['prenom_utilisateur'];
1274 aurelien 833
				$liaison['nom'] = $liaison['nom_utilisateur'];
834
			}
709 gduche 835
			// On enregistre l'ID de l'image pour n'effectuer qu'une seule requête par la suite
836
			$this->imageIds[] = $idImage;
800 aurelien 837
			$index = $liaison['id_image'].'-'.$liaison['id_observation'];
1025 aurelien 838
			$images[$index] = array('id_image' => $idImage, 'binaire.href' => $this->formaterLienImage($idImage),
1326 gduche 839
					'protocoles_votes' => array(),
840
					'mots_cles_texte' => $liaison['mots_cles_texte_image'], 'observation' => $this->formaterObservation($liaison));
709 gduche 841
		}
842
		return $images;
843
	}
844
 
845
	/**
1326 gduche 846
	 * Charger les votes pour chaque image
847
	 * */
709 gduche 848
	private function chargerVotes($images) {
1326 gduche 849
		$requeteVotes = 'SELECT v.*, p.* FROM '.
850
				$this->gestionBdd->formaterTable('del_image_vote', 'v').
851
				' INNER JOIN del_image_protocole p '.
852
				'ON v.ce_protocole = p.id_protocole '.
853
				$this->chargerClauseWhereVotes();
854
		$resultatsVotes = $this->bdd->recupererTous($requeteVotes);
709 gduche 855
 
857 gduche 856
 
1326 gduche 857
		//TODO : faire une méthode formater vote
858
		$votes = $this->formaterVotes($resultatsVotes);
709 gduche 859
 
1326 gduche 860
		foreach ($images as $id => $image) {
861
			if (isset($votes[$image['id_image']])) {
862
				$images[$id]['protocoles_votes'] = $votes[$image['id_image']];
709 gduche 863
			}
1326 gduche 864
		}
865
 
709 gduche 866
		return $images;
867
	}
868
 
759 delphine 869
	private function chargerClauseWhereVotes() {
870
		if (sizeof($this->imageIds) > 0) {
871
			$chaineImageIds = implode(',', $this->imageIds);
760 delphine 872
			$where[] = 'v.ce_image  IN ('.$chaineImageIds.')';
759 delphine 873
		}
874
		if (isset($this->parametres['protocole'])) {
760 delphine 875
			$where[] = 'v.ce_protocole = '.$this->proteger($this->parametres['protocole']);
759 delphine 876
		}
1326 gduche 877
 
775 aurelien 878
		$where = (!empty($where)) ? 'WHERE '.implode(' AND ', $where) : '';
879
		return $where;
759 delphine 880
	}
881
 
1326 gduche 882
	/**************************************************************************************
883
	 *		    			FONCTION DE CONFIGURATION ET UTILITAIRES					  *
884
	***************************************************************************************/
885
	/**
886
	 * Enregistrer dans les variables de classe les paramètres et ressources
887
	 * @param $ressources
888
	 * @param $parametres de recherche
889
	 * */
890
	private function initialiserRessourcesEtParametres($ressources, $parametres) {
891
		$this->ressources = $ressources;
892
		$this->parametres = $parametres;
893
	}
894
 
895
	/**
896
	 * Configuration du service en fonction du fichier de config config_del.ini
897
	 * */
898
	public function configurer() {
899
		$this->mappingFiltre = $this->conteneur->getParametre('mapping_masque');
900
		$this->mappingObservation = $this->conteneur->getParametre('mapping_observation');
901
		$this->mappingVotes = $this->conteneur->getParametre('mapping_votes');
902
	}
903
 
904
	/**
905
	 * Vérifier que le service est bien configuré
906
	 * */
907
	public function verifierConfiguration() {
908
 
909
		$erreurs = array();
910
		$tableauImages = $this->conteneur->getParametre('images');
911
		if (empty($tableauImages)) {
912
			$erreurs[] = '- le fichier de configuration ne contient pas le tableau [images] ou celui-ci est vide ;';
913
		} else {
914
			if ($this->conteneur->getParametre('url_service') == null) {
915
				$erreurs[] = '- paramètre "url_service" manquant ;';
916
			}
917
 
918
			if ($this->conteneur->getParametre('url_images') == null) {
919
				$erreurs[] = '- paramètre "url_images" manquant ;';
920
			}
921
 
922
		}
923
 
924
		if (empty($this->mappingObservation)) {
925
			$erreurs[] = '- le fichier de configuration ne contient pas le tableau [mapping_observation] ou celui-ci est vide ;';
926
		} else {
927
			$champsMappingObs = array('id_observation', 'date_observation', 'date_transmission', 'famille', 'nom_sel', 'nom_sel_nn', 'nt',
928
					'ce_zone_geo', 'lieudit', 'station', 'milieu', 'ce_utilisateur', 'nom', 'prenom');
929
			foreach ($champsMappingObs as $champ) {
930
				if (!isset($this->mappingObservation[$champ])) {
931
					$erreurs[] = '- le mapping du champ "'.$champ.'" pour l\'observation est manquant ;';
932
				}
933
			}
934
		}
935
 
936
		if (empty($this->mappingFiltre)) {
937
			$erreurs[] = '- le fichier de configuration ne contient pas le tableau [mapping_masque] ou celui-ci est vide ;';
938
		} else {
939
			$champsMappingFiltre = array('famille', 'ns', 'nn', 'date', 'tag', 'commune');
940
			foreach ($champsMappingFiltre as $champ) {
941
				if (!isset($this->mappingFiltre[$champ])) {
942
					$erreurs[] = '- le mapping du champ "'.$champ.'" pour l\'observation est manquant ;';
943
				}
944
			}
945
		}
946
 
947
		$tris_possibles = $this->conteneur->getParametre('tris_possibles');
948
		if (empty($tris_possibles)) {
949
			$erreurs[] = '- le fichier de configuration ne contient pas le parametre tris_possibles ou celui-ci est vide ;';
950
		}
951
 
952
		if (!empty($erreurs)) {
953
			$e = 'Erreur lors de la configuration : '."\n";
954
			$e .= implode("\n", $erreurs);
955
			throw new Exception($e, RestServeur::HTTP_CODE_ERREUR);
956
		}
957
	}
958
 
959
	/**
960
	 * Verifier que les paramètres de tri sont bien autorisés et qu'ils sont au bon format.
961
	 */
962
	private function verifierParametresTri() {
963
 
964
		$erreurs = array();
965
		$tris_possibles = $this->conteneur->getParametre('tris_possibles');
966
		$tris_possibles_tableau = explode(',', $tris_possibles);
967
		if(isset($this->parametres['tri']) && !in_array($this->parametres['tri'], $tris_possibles_tableau)) {
968
			$erreurs[] = '- le type de tri demandé est incorrect, les valeurs possibles sont '.$tris_possibles.' ;';
969
		}
970
 
971
		if(isset($this->parametres['tri']) && $this->parametres['tri'] == "votes") {
972
			if(!isset($this->parametres['protocole']) || !is_numeric($this->parametres['protocole'])) {
973
				$erreurs[] = '- Le paramètre protocole est obligatoire en cas de tri par vote et doit être un entier ;';
974
			}
975
		}
976
 
977
		$directions_tri = array('asc', 'desc');
978
		if(isset($this->parametres['ordre']) && !in_array($this->parametres['ordre'], $directions_tri)) {
979
			$erreurs[] = '- la direction du tri demandé est incorrecte, les valeurs supportées sont asc ou desc ;';
980
		}
981
 
982
		if (!empty($erreurs)) {
983
			$e = 'Erreur lors de l\'analyse des parametres du tri : '."\n";
984
			$e .= implode("\n", $erreurs);
985
			throw new Exception($e, RestServeur::HTTP_CODE_ERREUR);
986
		}
987
	}
1345 aurelien 988
 
989
	private function verifierParametreFormatRetour() {
990
		$erreurs = array();
991
		$formats_possibles_str = $this->conteneur->getParametre('formats_possibles');
992
		$formats_possibles = explode(',',$formats_possibles_str);
1326 gduche 993
 
1345 aurelien 994
		if(isset($this->parametres['format']) && !in_array($this->parametres['format'], $formats_possibles)) {
995
			$erreurs[] = "- le format d'image demandé n'est pas supporté ; Les format supportés sont : ".$formats_possibles_str;
996
		}
997
 
998
		if (!empty($erreurs)) {
999
			$e = 'Erreur lors de l\'analyse du format de retour demandé : '."\n";
1000
			$e .= implode("\n", $erreurs);
1001
			throw new Exception($e, RestServeur::HTTP_CODE_ERREUR);
1002
		}
1003
	}
1004
 
1005
	private function initialiserFormatRetour() {
1006
		$this->formatRetour = isset($this->parametres['format']) ? $this->parametres['format'] : $this->formatRetour;
1007
	}
1008
 
1326 gduche 1009
	/**
1010
	 * Initialiser les variables de tri depuis les paramètres
1011
	 * */
1012
	private function initialiserTri() {
1013
		$this->tri = isset($this->parametres['tri']) ? $this->parametres['tri'] : $this->tri;
1014
		$this->directionTri = isset($this->parametres['ordre']) ? $this->parametres['ordre'] : $this->directionTri;
1015
	}
1016
 
1017
	/** Pour eviter les requêtes trop gourmandes, on supprime les caractères passe-partout
1018
	 * @param les paramètres de l'application
1019
	 * */
1020
	public function nettoyerParametres($parametres) {
1021
		$parametresRetour = array();
1022
		foreach ($parametres as $cle => $valeur) {
1023
			$valSansPourcent = trim($valeur, "% ");
1024
			if ($valSansPourcent != '') {
1025
				$parametresRetour[$cle] = $valeur;
1026
			}
1027
		}
1028
 
1029
		return $parametresRetour;
1030
	}
1031
 
1032
	/**
1033
	 * Nettoyer les jokers
1034
	 * @param la valeur du masque
1035
	 * */
1036
	private function remplacerParJokerCaractere($valeurMasque) {
1037
		return str_replace(array('-',' '), '_', $valeurMasque);
1038
	}
1039
 
1040
	//TODO: déplacer les fonctions ci dessus et dessous dans une classe
1041
	// utilitaire
1042
 
1043
	/**
1044
	 * Supprimer les accents des chaines de caractères
1045
	 * */
1046
	function supprimerAccents($str, $charset='utf-8')
1047
	{
1048
		$str = htmlentities($str, ENT_NOQUOTES, $charset);
1049
 
1050
		$str = preg_replace('#&([A-za-z])(?:acute|cedil|circ|grave|orn|ring|slash|th|tilde|uml);#', '\1', $str);
1051
		$str = preg_replace('#&([A-za-z]{2})(?:lig);#', '\1', $str); // pour les ligatures e.g. '&oelig;'
1052
		$str = preg_replace('#&[^;]+;#', '', $str); // supprime les autres caractères
1053
 
1054
		return $str;
1055
	}
1056
 
1057
	/**
1058
	 * Normaliser en supprimant les accents et en mettant en minuscule
1059
	 * @param $mot_cle le mot recherché
1060
	 * */
1061
	private function normaliserMotCle($mot_cle) {
1062
		return mb_strtolower($this->supprimerAccents(trim($mot_cle)));
1063
	}
1064
 
1065
	/**
1066
	 * Récupérer le numéro du département d'un fichier de configuration
1067
	 * */
1068
	private function obtenirIdDepartement($nomDpt) {
1069
 
1070
		$nomDpt = $this->supprimerAccents($nomDpt);
1071
		$nomDpt = strtolower(str_replace(' ','-',$nomDpt));
1072
 
1073
		$idDpt = $this->conteneur->getParametre($nomDpt);
1074
		if($idDpt == null || $idDpt == ' ') {
1075
			$idDpt = ' ';
1076
		}
1077
		return $idDpt;
1078
	}
1079
 
1080
 
1081
	/**
1082
	 * Obtenir le type de requête à exécuter en fonction des paramètres de recherche
1083
	 * @param $parametres les paramètres de l'application
1084
	 * */
1085
	private function getTypeRequete($parametres) {
1086
 
1087
 
1088
		$typeRequete = 'simple';
1089
 
1090
		// Dans ce cas précis, les informations concernant le depart, la limite ou l'ordre ne
1091
		// rentre pas en compte dans le type de requête ; ce ne sont que des compléments.
1092
		unset($parametres['navigation.depart']);
1093
		unset($parametres['navigation.limite']);
1094
		unset($parametres['ordre']);
1095
 
1096
		// En revanche, chaque masque est associé à un type de requête particulier.
1332 aurelien 1097
		$masquesObservation = array('masque', 'masque.departement', 'masque.ns', 'masque.genre', 'masque.date', 'masque.commune', 'masque.famille', 'masque.auteur', 'masque.nn');
1326 gduche 1098
		$masquesImage = array('masque', 'masque.tag');
1099
 
1100
		// Le type de requête est défini par les tables qu'il doit inclure (observation, image, ou les deux)
1101
		$requeteSimple = false;
1102
		$pourObservation = false;
1103
		$pourImage = false;
1104
 
1105
		// S'il n'y a aucun paramètre, on lance une requête simple
1106
		if (empty($parametres)) {
1107
			$requeteSimple = true;
1108
		}
1109
 
1110
		// Si l'un des masques demandé concerne l'observation
1111
		foreach ($masquesObservation as $masque) {
1112
			if (isset($parametres[$masque])) {
1113
				$pourObservation = true;
1114
				break;
1115
			}
1116
		}
1117
 
1118
		// Si l'un des masques demandé concerne les images
1119
		foreach ($masquesImage as $masque) {
1120
			if (isset($parametres[$masque])) {
1121
				$pourImage = true;
1122
				break;
1123
			}
1124
		}
1125
 
1126
		// Selon les tri
1127
		if (isset($parametres['tri'])) {
1128
			switch ($parametres['tri']) {
1129
				case 'votes' :
1130
				case 'tags' :
1131
					$pourImage = true;
1132
					break;
1133
				default : //case 'date_observation' :
1134
					if (sizeof($parametres) > 1) {
1135
						$pourObservation = true;
1136
					}
1137
			}
1138
		}
1139
 
1140
		// Vérifier la combinaison des booléens pour en déduire le type de requête
1141
		if ($pourObservation && $pourImage) {
1142
			$typeRequete = 'obs-images';
1143
		} else {
1144
			if ($pourImage) {
1145
				$typeRequete = 'images';
1146
			} else if ($pourObservation) {
1147
				$typeRequete = 'obs';
1148
			} else { // if ($requeteSimple)
1149
				$typeRequete = 'simple';
1150
			}
1151
		}
1152
 
1153
		return $typeRequete;
1154
	}
1155
 
1156
 
1157
	private function doitJoindreTableVotes() {
1158
		return ($this->tri == 'votes');
1159
	}
1160
 
1161
	private function doitJoindreTableTags() {
1162
		return ($this->tri == 'tags');
1163
	}
1164
 
709 gduche 1165
	/*-------------------------------------------------------------------------------
1166
								FORMATER ET METTRE EN FORME
1167
	--------------------------------------------------------------------------------*/
1168
 
1169
	/**
1170
	*  Formater une observation depuis une ligne liaison
1171
	*  @param $liaison liaison issue de la recherche
1172
	*  @return $observation l'observation mise en forme
1173
	* */
1174
	private function formaterObservation($liaison) {
1175
		$observation = array();
1176
		foreach ($this->mappingObservation as $nomOriginal => $nomFinal) {
1177
			$observation[$nomFinal] = $liaison[$nomOriginal];
1178
		}
1179
 
1180
		return $observation;
1181
	}
1182
 
1183
	/**
1184
	*  Formater une observation depuis une ligne liaison
1185
	*  @param $liaison liaison issue de la recherche
1186
	*  @return $observation l'observation mise en forme
1187
	* */
1188
	private function formaterVotes($votes) {
1189
		$retour = array();
1190
		foreach ($votes as $vote) {
759 delphine 1191
			$retour_vote = array();
1192
			foreach ($vote as $param=>$valeur) {
760 delphine 1193
				if (strpos($this->mappingVotes[$param], 'protocole.') === 0) {
1194
					$retour_protocole[$this->mappingVotes[$param]] = $valeur;
1195
				} else {
1196
					$retour_vote[$this->mappingVotes[$param]] = $valeur;
1197
				}
759 delphine 1198
			}
760 delphine 1199
			if (!isset($retour[$vote['ce_image']][$vote['ce_protocole']])) {
1200
				$retour[$vote['ce_image']][$vote['ce_protocole']] = $retour_protocole;
1201
			}
1202
			$retour[$vote['ce_image']][$vote['ce_protocole']]['votes'][$vote['id_vote']] = $retour_vote;
709 gduche 1203
		}
1204
 
1205
		return $retour;
1206
	}
1207
 
1208
	/**
1209
	 * Formater le lien de l'image en fonction du fichier de config et de l'identifiant de l'image
1210
	 * */
1211
	private function formaterLienImage($idImage) {
1212
		$idImage = sprintf('%09s', $idImage);
719 gduche 1213
		$url = $this->conteneur->getParametre('url_images');
1345 aurelien 1214
		$urlImage = sprintf($url, $idImage, $this->formatRetour);
709 gduche 1215
		return $urlImage;
1216
	}
719 gduche 1217
 
1218
	private function proteger($valeur) {
1219
		if (is_array($valeur)) {
1220
			return $this->bdd->protegerTableau($valeur);
1221
		} else {
1222
			return $this->bdd->proteger($valeur);
709 gduche 1223
		}
1224
	}
1225
}
1226
?>