3516e9dc7c63031225a867a93fbea8d780ad6bb0
[ptitvelo/web/www.git] / www / ecrire / inc / utils.php
1 <?php
2
3 /***************************************************************************\
4 * SPIP, Systeme de publication pour l'internet *
5 * *
6 * Copyright (c) 2001-2012 *
7 * Arnaud Martin, Antoine Pitrou, Philippe Riviere, Emmanuel Saint-James *
8 * *
9 * Ce programme est un logiciel libre distribue sous licence GNU/GPL. *
10 * Pour plus de details voir le fichier COPYING.txt ou l'aide en ligne. *
11 \***************************************************************************/
12
13 if (!defined('_ECRIRE_INC_VERSION')) return;
14
15 //
16 // Utilitaires indispensables autour du serveur Http.
17 //
18
19 /**
20 * charge un fichier perso ou, a defaut, standard
21 * et retourne si elle existe le nom de la fonction homonyme (exec_$nom),
22 * ou de suffixe _dist
23 * Peut etre appelee plusieurs fois, donc optimiser
24 *
25 * http://doc.spip.org/@charger_fonction
26 *
27 * @param string $nom
28 * @param string $dossier
29 * @param bool $continue
30 * @return string
31 */
32 function charger_fonction($nom, $dossier='exec', $continue=false) {
33 static $echecs = array();
34
35 if (strlen($dossier) AND substr($dossier,-1) != '/') $dossier .= '/';
36 $f = str_replace('/','_',$dossier) . $nom;
37
38 if (function_exists($f))
39 return $f;
40 if (function_exists($g = $f . '_dist'))
41 return $g;
42
43 if (isset($echecs[$f])) return $echecs[$f];
44 // Sinon charger le fichier de declaration si plausible
45
46 if (!preg_match(',^\w+$,', $f)){
47 if ($continue) return false; //appel interne, on passe
48 die(htmlspecialchars($nom)." pas autorise");
49 }
50
51 // passer en minuscules (cf les balises de formulaires)
52 // et inclure le fichier
53 if (!$inc = include_spip($dossier.($d = strtolower($nom)))
54 // si le fichier truc/machin/nom.php n'existe pas,
55 // la fonction peut etre definie dans truc/machin.php qui regroupe plusieurs petites fonctions
56 AND strlen(dirname($dossier)) AND dirname($dossier)!='.')
57 include_spip(substr($dossier,0,-1));
58 if (function_exists($f)) return $f;
59 if (function_exists($g)) return $g;
60
61 if ($continue) return $echecs[$f] = false;
62
63 // Echec : message d'erreur
64 spip_log("fonction $nom ($f ou $g) indisponible" .
65 ($inc ? "" : " (fichier $d absent de $dossier)"));
66
67 include_spip('inc/minipres');
68 echo minipres(_T('forum_titre_erreur'),
69 _T('fichier_introuvable', array('fichier'=> '<b>'.htmlentities($d).'</b>')));
70 exit;
71 }
72
73 /**
74 * Inclusion unique avec verification d'existence du fichier + log en crash sinon
75 * @param string $file
76 * @return bool
77 */
78 function include_once_check($file){
79 if (file_exists($file)) {include_once $file;return true;}
80 $crash = (isset($GLOBALS['message_crash_plugins'])?unserialize($GLOBALS['message_crash_plugins']):'');
81 $crash = ($crash?$crash:array());
82 $crash[$file] = true;
83 ecrire_meta('message_crash_plugins',serialize($crash));
84 return false;
85 }
86
87 //
88 // la fonction cherchant un fichier PHP dans le SPIP_PATH
89 //
90 // http://doc.spip.org/@include_spip
91 function include_spip($f, $include = true) {
92 return find_in_path($f . '.php', '', $include);
93 }
94
95
96 function require_spip($f) {
97 return find_in_path($f . '.php', '', 'required');
98 }
99
100 // un pipeline est lie a une action et une valeur
101 // chaque element du pipeline est autorise a modifier la valeur
102 //
103 // le pipeline execute les elements disponibles pour cette action,
104 // les uns apres les autres, et retourne la valeur finale
105 //
106 // Cf. compose_filtres dans references.php, qui est la
107 // version compilee de cette fonctionnalite
108
109 // appel unitaire d'une fonction du pipeline
110 // utilisee dans le script pipeline precompile
111 // on passe $val par reference pour limiter les allocations memoire
112 // http://doc.spip.org/@minipipe
113 function minipipe($fonc,&$val){
114 // fonction
115 if (function_exists($fonc))
116 $val = call_user_func($fonc, $val);
117
118 // Class::Methode
119 else if (preg_match("/^(\w*)::(\w*)$/S", $fonc, $regs)
120 AND $methode = array($regs[1], $regs[2])
121 AND is_callable($methode))
122 $val = call_user_func($methode, $val);
123 else {
124 spip_log("Erreur - '$fonc' non definie !");
125 }
126 return $val;
127 }
128
129 // chargement du pipeline sous la forme d'un fichier php prepare
130 // http://doc.spip.org/@pipeline
131 function pipeline($action, $val=null) {
132 static $charger;
133
134 // chargement initial des fonctions mises en cache, ou generation du cache
135 if (!$charger) {
136 if (!($ok = @is_readable($charger = _CACHE_PIPELINES))) {
137 include_spip('inc/plugin');
138 // generer les fichiers php precompiles
139 // de chargement des plugins et des pipelines
140 actualise_plugins_actifs();
141 if (!($ok = @is_readable($charger)))
142 spip_log("fichier $charger pas cree");
143 }
144
145 if ($ok) {
146 include_once $charger;
147 }
148 }
149
150 // appliquer notre fonction si elle existe
151 $fonc = 'execute_pipeline_'.strtolower($action);
152 if (function_exists($fonc)) {
153 $val = $fonc($val);
154 }
155 // plantage ?
156 else {
157 spip_log("fonction $fonc absente : pipeline desactive",_LOG_ERREUR);
158 }
159
160 // si le flux est une table avec 2 cle args&data
161 // on ne ressort du pipe que les donnees dans 'data'
162 // array_key_exists pour php 4.1.0
163 if (is_array($val)
164 AND count($val)==2
165 AND (array_key_exists('data',$val)))
166 $val = $val['data'];
167 return $val;
168 }
169
170 /**
171 * Enregistrement des evenements
172 * spip_log($message)
173 * spip_log($message,'recherche')
174 * spip_log($message,_LOG_DEBUG)
175 * spip_log($message,'recherche.'._LOG_DEBUG)
176 * cette derniere notation est controversee mais le 3eme
177 * parametre est plante pour cause de compat ascendante.
178 * le niveau par defaut est _LOG_INFO
179 *
180 * http://doc.spip.org/@spip_log
181 *
182 * @param string $message
183 * @param string|int $name
184 * @param string $logdir ## inutile !! a supprimer ?
185 * @param string $logsuf ## inutile !! a supprimer ?
186 */
187 function spip_log($message=NULL, $name=NULL) {
188 static $pre = array();
189 static $log;
190 preg_match('/^([a-z_]*)\.?(\d)?$/iS', (string) $name, $regs);
191 if (!isset($regs[1]) OR !$logname = $regs[1])
192 $logname = null;
193 if (!isset($regs[2]) OR !$niveau = $regs[2])
194 $niveau = _LOG_INFO;
195
196 if ($niveau <= (defined('_LOG_FILTRE_GRAVITE') ? _LOG_FILTRE_GRAVITE : _LOG_INFO_IMPORTANTE)) {
197 if (!$pre){
198 $pre = array(
199 _LOG_HS=>'HS:',
200 _LOG_ALERTE_ROUGE=>'ALERTE:',
201 _LOG_CRITIQUE=>'CRITIQUE:',
202 _LOG_ERREUR=>'ERREUR:',
203 _LOG_AVERTISSEMENT=>'WARNING:',
204 _LOG_INFO_IMPORTANTE=>'!INFO:',
205 _LOG_INFO=>'info:',
206 _LOG_DEBUG=>'debug:');
207 $log = charger_fonction('log', 'inc');
208 }
209 if (!is_string($message)) $message = var_export($message, true);
210 $log($pre[$niveau].' '.$message, $logname);
211 }
212 }
213
214 //
215 // Enregistrement des journaux
216 //
217 function journal($phrase, $opt = array()) {
218 $journal = charger_fonction('journal', 'inc');
219 $journal($phrase, $opt);
220 }
221
222 // Renvoie le _GET ou le _POST emis par l'utilisateur
223 // ou pioche dans $c si c'est un array()
224 // http://doc.spip.org/@_request
225 function _request($var, $c=false) {
226
227 if (is_array($c))
228 return isset($c[$var]) ? $c[$var] : NULL;
229
230 if (isset($_GET[$var])) $a = $_GET[$var];
231 elseif (isset($_POST[$var])) $a = $_POST[$var];
232 else return NULL;
233
234 // Si on est en ajax et en POST tout a ete encode
235 // via encodeURIComponent, il faut donc repasser
236 // dans le charset local...
237 if (defined('_AJAX')
238 AND _AJAX
239 AND isset($GLOBALS['meta']['charset'])
240 AND $GLOBALS['meta']['charset'] != 'utf-8'
241 AND is_string($a)
242 AND preg_match(',[\x80-\xFF],', $a)) {
243 include_spip('inc/charsets');
244 return importer_charset($a, 'utf-8');
245 }
246
247 return $a;
248 }
249
250 // Methode set de la fonction _request()
251 // Attention au cas ou l'on fait set_request('truc', NULL);
252 // http://doc.spip.org/@set_request
253 function set_request($var, $val = NULL, $c=false) {
254 if (is_array($c)) {
255 unset($c[$var]);
256 if ($val !== NULL)
257 $c[$var] = $val;
258 return $c;
259 }
260
261 unset($_GET[$var]);
262 unset($_POST[$var]);
263 if ($val !== NULL)
264 $_GET[$var] = $val;
265
266 return false; # n'affecte pas $c
267 }
268
269
270 /**
271 * Tester si une url est absolue
272 * @param $url
273 * @return bool
274 */
275 function tester_url_absolue($url){
276 return preg_match(";^([a-z]+:)?//;Uims",trim($url))?true:false;
277 }
278
279 /**
280 * Prend une URL et lui ajoute/retire un parametre.
281 * Exemples : [(#SELF|parametre_url{suite,18})] (ajout)
282 * [(#SELF|parametre_url{suite,''})] (supprime)
283 * [(#SELF|parametre_url{suite})] (prend $suite dans la _request)
284 * [(#SELF|parametre_url{suite[],1})] (tableaux valeurs multiples)
285 *
286 * http://doc.spip.org/@parametre_url
287 *
288 * @param string $url
289 * @param string $c
290 * @param string|array $v
291 * @param string $sep
292 * @return string
293 */
294 function parametre_url($url, $c, $v=NULL, $sep='&amp;') {
295 // requete erronnee : plusieurs variable dans $c et aucun $v
296 if (strpos($c,"|")!==false AND is_null($v))
297 return null;
298
299 // lever l'#ancre
300 if (preg_match(',^([^#]*)(#.*)$,', $url, $r)) {
301 $url = $r[1];
302 $ancre = $r[2];
303 } else
304 $ancre = '';
305
306 // eclater
307 $url = preg_split(',[?]|&amp;|&,', $url);
308
309 // recuperer la base
310 $a = array_shift($url);
311 if (!$a) $a= './';
312
313 $regexp = ',^(' . str_replace('[]','\[\]',$c) . '[[]?[]]?)(=.*)?$,';
314 $ajouts = array_flip(explode('|',$c));
315 $u = is_array($v) ? $v : rawurlencode($v);
316 $testv = (is_array($v)?count($v):strlen($v));
317 // lire les variables et agir
318 foreach ($url as $n => $val) {
319 if (preg_match($regexp, urldecode($val), $r)) {
320 if ($v === NULL) {
321 return $r[2]?substr($r[2],1):'';
322 }
323 // suppression
324 elseif (!$testv) {
325 unset($url[$n]);
326 }
327 // Ajout. Pour une variable, remplacer au meme endroit,
328 // pour un tableau ce sera fait dans la prochaine boucle
329 elseif (substr($r[1],-2) != '[]') {
330 $url[$n] = $r[1].'='.$u;
331 unset($ajouts[$r[1]]);
332 }
333 }
334 }
335
336 // traiter les parametres pas encore trouves
337 if ($v === NULL
338 AND $args = func_get_args()
339 AND count($args)==2)
340 return $v;
341 elseif ($testv) {
342 foreach($ajouts as $k => $n) {
343 if (!is_array($v))
344 $url[] = $k .'=' . $u;
345 else {
346 $id = (substr($k,-2) == '[]') ? $k : ($k ."[]");
347 foreach ($v as $w) $url[]= $id .'=' . $w;
348 }
349 }
350 }
351
352 // eliminer les vides
353 $url = array_filter($url);
354
355 // recomposer l'adresse
356 if ($url)
357 $a .= '?' . join($sep, $url);
358
359 return $a . $ancre;
360 }
361
362 // Prend une URL et lui ajoute/retire une ancre apres l'avoir nettoyee
363 // pour l'ancre on translitere, vire les non alphanum du debut,
364 // et on remplace ceux a l'interieur ou au bout par -
365 // http://doc.spip.org/@ancre_url
366 function ancre_url($url, $ancre) {
367 // lever l'#ancre
368 if (preg_match(',^([^#]*)(#.*)$,', $url, $r)) {
369 $url = $r[1];
370 }
371 if (preg_match('/[^-_a-zA-Z0-9]+/S',$ancre)){
372 if (!function_exists('translitteration'))
373 include_spip('inc/charsets');
374 $ancre = preg_replace(array('/^[^-_a-zA-Z0-9]+/', '/[^-_a-zA-Z0-9]/'), array('', '-'),
375 translitteration($ancre));
376 }
377 return $url . (strlen($ancre) ? '#'. $ancre : '');
378 }
379
380 //
381 // pour le nom du cache, les types_urls et self
382 //
383 // http://doc.spip.org/@nettoyer_uri
384 function nettoyer_uri()
385 {
386 static $done = false;
387 static $propre = '';
388 if ($done) return $propre;
389 $done = true;
390
391 $uri1 = $GLOBALS['REQUEST_URI'];
392 do {
393 $uri = $uri1;
394 $uri1 = preg_replace
395 (',([?&])(PHPSESSID|(var_[^=&]*))=[^&]*(&|$),i',
396 '\1', $uri);
397 } while ($uri<>$uri1);
398
399 return $propre = (preg_replace(',[?&]$,', '', $uri1));
400 }
401
402 //
403 // donner l'URL de base d'un lien vers "soi-meme", modulo
404 // les trucs inutiles
405 //
406 // http://doc.spip.org/@self
407 function self($amp = '&amp;', $root = false) {
408 $url = nettoyer_uri();
409 if (!$root
410 AND (
411 // si pas de profondeur on peut tronquer
412 $GLOBALS['profondeur_url']<(_DIR_RESTREINT?1:2)
413 // sinon c'est OK si _SET_HTML_BASE a ete force a false
414 OR (defined('_SET_HTML_BASE') AND !_SET_HTML_BASE))
415 )
416 $url = preg_replace(',^[^?]*/,', '', $url);
417 // ajouter le cas echeant les variables _POST['id_...']
418 foreach ($_POST as $v => $c)
419 if (substr($v,0,3) == 'id_')
420 $url = parametre_url($url, $v, $c, '&');
421
422 // supprimer les variables sans interet
423 if (test_espace_prive()) {
424 $url = preg_replace (',([?&])('
425 .'lang|show_docs|'
426 .'changer_lang|var_lang|action)=[^&]*,i', '\1', $url);
427 $url = preg_replace(',([?&])[&]+,', '\1', $url);
428 $url = preg_replace(',[&]$,', '\1', $url);
429 }
430
431 // eviter les hacks
432 $url = htmlspecialchars($url);
433
434 // &amp; ?
435 if ($amp != '&amp;')
436 $url = str_replace('&amp;', $amp, $url);
437
438 // Si ca demarre par ? ou vide, donner './'
439 $url = preg_replace(',^([?].*)?$,', './\1', $url);
440
441 return $url;
442 }
443
444 // Indique si on est dans l'espace prive
445 // http://doc.spip.org/@test_espace_prive
446 function test_espace_prive() {
447 return defined('_ESPACE_PRIVE') ? _ESPACE_PRIVE : false;
448 }
449
450 /**
451 * Verifie la presence d'un plugin active, identifie par son prefix
452 *
453 *
454 * @param string $plugin
455 * @return bool
456 */
457 function test_plugin_actif($plugin){
458 return ($plugin AND defined('_DIR_PLUGIN_'.strtoupper($plugin)))? true:false;
459 }
460
461 /**
462 * Traduction des textes de SPIP
463 * http://doc.spip.org/@_T
464 *
465 * @param string $texte
466 * @param array $args
467 * @param array $options
468 * string class : nom d'une classe a ajouter sur un span pour encapsuler la chaine
469 * bool force : forcer un retour meme si la chaine n'a pas de traduction
470 * @return mixed|string
471 */
472 function _T($texte, $args=array(), $options=array()) {
473 static $traduire=false ;
474 $o = array('class'=>'', 'force'=>true);
475 if ($options){
476 // support de l'ancien argument $class
477 if (is_string($options))
478 $options = array('class'=>$options);
479 $o = array_merge($o,$options);
480 }
481
482 if (!$traduire) {
483 $traduire = charger_fonction('traduire', 'inc');
484 include_spip('inc/lang');
485 }
486
487 // On peut passer explicitement la langue dans le tableau
488 // On utilise le même nom de variable que la globale
489 if (isset($args['spip_lang'])){
490 $lang = $args['spip_lang'];
491 // On l'enleve pour ne pas le passer au remplacement
492 unset($args['spip_lang']);
493 }
494 // Sinon on prend la langue du contexte
495 else {
496 $lang = $GLOBALS['spip_lang'];
497 }
498 $text = $traduire($texte, $lang);
499
500 if (!strlen($text)){
501 if (!$o['force'])
502 return '';
503
504 $text = $texte;
505
506 // pour les chaines non traduites, assurer un service minimum
507 if (!$GLOBALS['test_i18n'])
508 $text = str_replace('_', ' ',
509 (($n = strpos($text,':')) === false ? $texte :
510 substr($texte, $n+1)));
511 $o['class'] = null;
512
513 }
514
515 return _L($text, $args, $o['class']);
516
517 }
518
519 // Remplacer les variables @....@ par leur valeur dans une chaine de langue.
520 // Aussi appelee quand une chaine n'est pas encore dans les fichiers de langue
521 // http://doc.spip.org/@_L
522 function _L($text, $args=array(), $class=null) {
523 $f = $text;
524 if (is_array($args)) {
525 foreach ($args as $name => $value) {
526 if ($class)
527 $value = "<span class='$class'>$value</span>";
528 $t = str_replace ("@$name@", $value, $text);
529 if ($text !== $t) {unset($args[$name]); $text = $t;}
530 }
531 // Si des variables n'ont pas ete inserees, le signaler
532 // (chaines de langues pas a jour)
533 if ($args) spip_log("$f: variables inutilisees " . join(', ', array_keys($args)),_LOG_DEBUG);
534 }
535
536 if ($GLOBALS['test_i18n'] AND $class===null)
537 return "<blink style='color:red;'>$text</blink>";
538 else
539 return $text;
540 }
541
542 // Afficher "ecrire/data/" au lieu de "data/" dans les messages
543 // ou tmp/ au lieu de ../tmp/
544 // http://doc.spip.org/@joli_repertoire
545 function joli_repertoire($rep) {
546 $a = substr($rep,0,1);
547 if ($a<>'.' AND $a<>'/')
548 $rep = (_DIR_RESTREINT?'':_DIR_RESTREINT_ABS).$rep;
549 $rep = preg_replace(',(^\.\.\/),', '', $rep);
550 return $rep;
551 }
552
553
554 //
555 // spip_timer : on l'appelle deux fois et on a la difference, affichable
556 //
557 // http://doc.spip.org/@spip_timer
558 function spip_timer($t='rien', $raw = false) {
559 static $time;
560 $a=time(); $b=microtime();
561 // microtime peut contenir les microsecondes et le temps
562 $b=explode(' ',$b);
563 if (count($b)==2) $a = end($b); // plus precis !
564 $b = reset($b);
565 if (!isset($time[$t])) {
566 $time[$t] = $a + $b;
567 } else {
568 $p = ($a + $b - $time[$t]) * 1000;
569 unset($time[$t]);
570 # echo "'$p'";exit;
571 if ($raw) return $p;
572 if ($p < 1000)
573 $s = '';
574 else {
575 $s = sprintf("%d ", $x = floor($p/1000));
576 $p -= ($x*1000);
577 }
578 return $s . sprintf($s?"%07.3f ms":"%.3f ms", $p);
579 }
580 }
581
582
583 // Renvoie False si un fichier n'est pas plus vieux que $duree secondes,
584 // sinon renvoie True et le date sauf si ca n'est pas souhaite
585 // http://doc.spip.org/@spip_touch
586 function spip_touch($fichier, $duree=0, $touch=true) {
587 if ($duree) {
588 clearstatcache();
589 if ((@$f=filemtime($fichier)) AND ($f >= time() - $duree))
590 return false;
591 }
592 if ($touch!==false) {
593 if (!@touch($fichier)) { spip_unlink($fichier); @touch($fichier); };
594 @chmod($fichier, _SPIP_CHMOD & ~0111);
595 }
596 return true;
597 }
598
599 // Ce declencheur de tache de fond, de l'espace prive (cf inc_presentation)
600 // et de l'espace public (cf #SPIP_CRON dans inc_balise), est appelee
601 // par un background-image car contrairement a un iframe vide,
602 // les navigateurs ne diront pas qu'ils n'ont pas fini de charger,
603 // c'est plus rassurant.
604 // C'est aussi plus discret qu'un <img> sous un navigateur non graphique.
605
606 // http://doc.spip.org/@action_cron
607 function action_cron() {
608 include_spip('inc/headers');
609 http_status(204); // No Content
610 header("Connection: close");
611 define('_DIRECT_CRON_FORCE',true);
612 cron();
613 }
614
615 /**
616 * cron() : execution des taches de fond
617 * On peut lui passer en 1er (ou 2e arg pour compat)
618 * le tableau de taches attendu par inc_genie()
619 * Retourne Vrai si un tache a pu etre effectuee
620 * pas de verrou ici : les verrous sont geres sur chaque tache
621 * a chaque execution
622 *
623 * http://doc.spip.org/@cron
624 *
625 * @param array $taches
626 * taches forcees
627 * @param array $taches_old
628 * taches forcees, pour compat avec ancienne syntaxe
629 * @return bool
630 */
631 function cron ($taches=array(), $taches_old= array()) {
632 // si pas en mode cron force
633 // ou si base inaccessible, laisser tomber.
634 if (!defined('_DIRECT_CRON_FORCE') OR !spip_connect()) return false;
635 spip_log("cron !",'jq'._LOG_DEBUG);
636 if (!is_array($taches)) $taches = $taches_old; // compat anciens appels
637 if ($genie = charger_fonction('genie', 'inc', true)) {
638 $genie($taches);
639 return true;
640 }
641 return false;
642 }
643
644 /**
645 * Ajout d'une tache dans la file d'attente
646 *
647 * @param $function
648 * The function name to call.
649 * @param $description
650 * A human-readable description of the queued job.
651 * @param $arguments
652 * Optional array of arguments to pass to the function.
653 * @param $file
654 * Optional file path which needs to be included for $function.
655 * if ends with '/', will do charger_fonction($function,$file);
656 * @param $no_duplicate
657 * If TRUE, do not add the job to the queue if one with the same function and
658 * arguments already exists.
659 * @param $time
660 * time for starting the job. If 0, job will start as soon as possible
661 * @param $priority
662 * -10 (low priority) to +10 (high priority), 0 is the default
663 * @return int
664 * id of job
665 */
666 function job_queue_add($function, $description, $arguments = array(), $file = '', $no_duplicate = FALSE, $time=0, $priority=0) {
667 include_spip('inc/queue');
668 return queue_add_job($function, $description, $arguments, $file, $no_duplicate, $time, $priority);
669 }
670
671 /**
672 * Supprimer une tache de la file d'attente
673 * @param int $id_job
674 * id of jonb to delete
675 * @return bool
676 */
677 function job_queue_remove($id_job){
678 include_spip('inc/queue');
679 return queue_remove_job($id_job);
680 }
681
682 /**
683 * Associer une tache a un/des objets de SPIP
684 * @param int $id_job
685 * id of job to link
686 * @param array $objets
687 * can be a simple array('objet'=>'article','id_objet'=>23)
688 * or an array of simple array to link multiples objet in one time
689 */
690 function job_queue_link($id_job,$objets){
691 include_spip('inc/queue');
692 return queue_link_job($id_job,$objets);
693 }
694
695
696 /**
697 * Renvoyer le temps de repos restant jusqu'au prochain job
698 * 0 si un job est a traiter
699 * null si la queue n'est pas encore initialise
700 * $force est utilisee par queue_set_next_job_time() pour maj la valeur
701 * - si true, force la relecture depuis le fichier
702 * - si int, affecte la static directement avec la valeur
703 *
704 * @staticvar int $queue_next_job_time
705 * @param int/bool $force_next
706 * @return int
707 */
708 function queue_sleep_time_to_next_job($force=null) {
709 static $queue_next_job_time = -1;
710 if ($force===true)
711 $queue_next_job_time = -1;
712 elseif ($force)
713 $queue_next_job_time = $force;
714
715 if ($queue_next_job_time==-1) {
716 define('_JQ_NEXT_JOB_TIME_FILENAME',_DIR_TMP . "job_queue_next.txt");
717 // utiliser un cache memoire si dispo
718 if (include_spip('inc/memoization') AND defined('_MEMOIZE_MEMORY') AND _MEMOIZE_MEMORY) {
719 $queue_next_job_time = cache_get(_JQ_NEXT_JOB_TIME_FILENAME);
720 }
721 else {
722 $queue_next_job_time = null;
723 if (lire_fichier(_JQ_NEXT_JOB_TIME_FILENAME, $contenu))
724 $queue_next_job_time = intval($contenu);
725 }
726 }
727
728 if (is_null($queue_next_job_time))
729 return null;
730 if (!$_SERVER['REQUEST_TIME'])
731 $_SERVER['REQUEST_TIME'] = time();
732 return max(0,$queue_next_job_time-$_SERVER['REQUEST_TIME']);
733 }
734
735
736 // transformation XML des "&" en "&amp;"
737 // http://doc.spip.org/@quote_amp
738 function quote_amp($u) {
739 return preg_replace(
740 "/&(?![a-z]{0,4}\w{2,3};|#x?[0-9a-f]{2,5};)/i",
741 "&amp;",$u);
742 }
743
744 // Production d'une balise Script valide
745 // http://doc.spip.org/@http_script
746 function http_script($script, $src='', $noscript='') {
747 static $done = array();
748
749 if ($src && !isset($done[$src])){
750 $done[$src] = true;
751 $src = find_in_path($src, _JAVASCRIPT);
752 $src = " src='$src'";
753 }
754 else $src = '';
755 if ($script)
756 $script = ("/*<![CDATA[*/\n" .
757 preg_replace(',</([^>]*)>,','<\/\1>', $script) .
758 "/*]]>*/");
759 if ($noscript)
760 $noscript = "<noscript>\n\t$noscript\n</noscript>\n";
761
762 return ($src OR $script OR $noscript)
763 ? "<script type='text/javascript'$src>$script</script>$noscript"
764 : '';
765 }
766
767 // Transforme n'importe quel champ en une chaine utilisable
768 // en PHP ou Javascript en toute securite
769 // < ? php $x = '[(#TEXTE|texte_script)]'; ? >
770 // http://doc.spip.org/@texte_script
771 function texte_script($texte) {
772 return str_replace('\'', '\\\'', str_replace('\\', '\\\\', $texte));
773 }
774
775 // Chaque appel a cette fonction ajoute un repertoire en tete du chemin courant (path)
776 // si un repertoire lui est passe en parametre
777 // retourne le chemin courant sinon, sous forme de array.
778 // Si l'argument est de la forme dir1:dir2:dir3, ces 3 chemins sont places en tete
779 // du path, dans cet ordre.
780 // Exception: si un $dossier_squelette est defini, il reste en tete, pour raison historique
781 // http://doc.spip.org/@_chemin
782 function _chemin($dir_path=NULL){
783 static $path_base = NULL;
784 static $path_full = NULL;
785 if ($path_base==NULL){
786 // Chemin standard depuis l'espace public
787 $path = defined('_SPIP_PATH') ? _SPIP_PATH :
788 _DIR_RACINE.':'.
789 _DIR_RACINE.'squelettes-dist/:'.
790 _DIR_RACINE.'prive/:'.
791 _DIR_RESTREINT;
792 // Ajouter squelettes/
793 if (@is_dir(_DIR_RACINE.'squelettes'))
794 $path = _DIR_RACINE.'squelettes/:' . $path;
795 foreach (explode(':', $path) as $dir) {
796 if (strlen($dir) AND substr($dir,-1) != '/')
797 $dir .= "/";
798 $path_base[] = $dir;
799 }
800 $path_full = $path_base;
801 // Et le(s) dossier(s) des squelettes nommes
802 if (strlen($GLOBALS['dossier_squelettes']))
803 foreach (array_reverse(explode(':', $GLOBALS['dossier_squelettes'])) as $d)
804 array_unshift($path_full, ($d[0] == '/' ? '' : _DIR_RACINE) . $d . '/');
805 $GLOBALS['path_sig'] = md5(serialize($path_full));
806 }
807 if ($dir_path===NULL) return $path_full;
808
809 if (strlen($dir_path)){
810 $tete = "";
811 if (reset($path_base)==_DIR_RACINE.'squelettes/')
812 $tete = array_shift($path_base);
813 $dirs = array_reverse(explode(':',$dir_path));
814 foreach($dirs as $dir_path){
815 #if ($dir_path{0}!='/')
816 # $dir_path = $dir_path;
817 if (substr($dir_path,-1) != '/')
818 $dir_path .= "/";
819 if (!in_array($dir_path,$path_base))
820 array_unshift($path_base,$dir_path);
821 }
822 if (strlen($tete))
823 array_unshift($path_base,$tete);
824 }
825 $path_full = $path_base;
826 // Et le(s) dossier(s) des squelettes nommes
827 if (strlen($GLOBALS['dossier_squelettes']))
828 foreach (array_reverse(explode(':', $GLOBALS['dossier_squelettes'])) as $d)
829 array_unshift($path_full, ($d[0] == '/' ? '' : _DIR_RACINE) . $d . '/');
830
831 $GLOBALS['path_sig'] = md5(serialize($path_full));
832 return $path_full;
833 }
834
835 // http://doc.spip.org/@creer_chemin
836 function creer_chemin() {
837 $path_a = _chemin();
838 static $c = '';
839
840 // on calcule le chemin si le dossier skel a change
841 if ($c != $GLOBALS['dossier_squelettes']) {
842 // assurer le non plantage lors de la montee de version :
843 $c = $GLOBALS['dossier_squelettes'];
844 $path_a = _chemin(''); // forcer un recalcul du chemin
845 }
846 return $path_a;
847 }
848
849
850 function lister_themes_prives(){
851 static $themes = null;
852 if (is_null($themes)){
853 // si pas encore definie
854 if (!defined('_SPIP_THEME_PRIVE'))
855 define('_SPIP_THEME_PRIVE', 'spip');
856 $themes = array(_SPIP_THEME_PRIVE);
857 // lors d'une installation neuve, prefs n'est pas definie.
858 if (isset($GLOBALS['visiteur_session']['prefs'])) {
859 $prefs = isset($GLOBALS['visiteur_session']['prefs']);
860 } else {
861 $prefs = array();
862 }
863 if (is_string($prefs))
864 $prefs = unserialize($GLOBALS['visiteur_session']['prefs']);
865 if (
866 ((isset($prefs['theme']) AND $theme = $prefs['theme'])
867 OR (isset($GLOBALS['theme_prive_defaut']) AND $theme = $GLOBALS['theme_prive_defaut']))
868 AND $theme != _SPIP_THEME_PRIVE)
869 array_unshift($themes,$theme); // placer le theme choisi en tete
870 }
871 return $themes;
872 }
873
874 function find_in_theme($file, $subdir='', $include=false){
875 static $themefiles=array();
876 if (isset($themefiles["$subdir$file"])) return $themefiles["$subdir$file"];
877 $themes = lister_themes_prives();
878 foreach($themes as $theme){
879 if ($f = find_in_path($file,"prive/themes/$theme/$subdir",$include))
880 return $themefiles["$subdir$file"] = $f;
881 }
882 spip_log("$file introuvable dans le theme prive ".reset($themes),'theme');
883 return $themefiles["$subdir$file"] = "";
884 }
885
886 // Cherche une image dans les dossiers images
887 // gere le renommage des icones de facon temporaire (le temps de la migration)
888 // definis par _NOM_IMG_PACK et _DIR_IMG_PACK
889 // peut se trouver dans un dossier plugin, donc on passe par un find_in_path si elle n'est pas
890 // dans _DIR_IMG_PACK
891 // http://doc.spip.org/@chemin_image
892 function chemin_image($icone){
893 static $icone_renommer;
894 // gerer le cas d'un double appel en evitant de refaire le travail inutilement
895 if (strpos($icone,"/")!==false AND file_exists($icone)) return $icone;
896
897 // si c'est un nom d'image complet (article-24.png) essayer de le renvoyer direct
898 if (preg_match(',[.](png|gif|jpg)$,',$icone) AND $f = find_in_theme("images/$icone"))
899 return $f;
900 // sinon passer par le module de renommage
901 if (is_null($icone_renommer))
902 $icone_renommer = charger_fonction('icone_renommer','inc',true);
903 if ($icone_renommer){
904 list($icone,$fonction) = $icone_renommer($icone,"");
905 if (file_exists($icone))
906 return $icone;
907 }
908 return find_in_path ($icone, _NOM_IMG_PACK);
909 }
910
911 //
912 // chercher un fichier $file dans le SPIP_PATH
913 // si on donne un sous-repertoire en 2e arg optionnel, il FAUT le / final
914 // si 3e arg vrai, on inclut si ce n'est fait.
915 $GLOBALS['path_sig'] = '';
916 $GLOBALS['path_files'] = null;
917
918 // http://doc.spip.org/@find_in_path
919 function find_in_path ($file, $dirname='', $include=false) {
920 static $dirs=array();
921 static $inc = array(); # cf http://trac.rezo.net/trac/spip/changeset/14743
922 static $c = '';
923
924 // on calcule le chemin si le dossier skel a change
925 if ($c != $GLOBALS['dossier_squelettes']){
926 // assurer le non plantage lors de la montee de version :
927 $c = $GLOBALS['dossier_squelettes'];
928 creer_chemin(); // forcer un recalcul du chemin et la mise a jour de path_sig
929 }
930
931 if (isset($GLOBALS['path_files'][$GLOBALS['path_sig']][$dirname][$file])) {
932 if (!$GLOBALS['path_files'][$GLOBALS['path_sig']][$dirname][$file])
933 return false;
934 if ($include AND !isset($inc[$dirname][$file])) {
935 include_once _ROOT_CWD . $GLOBALS['path_files'][$GLOBALS['path_sig']][$dirname][$file];
936 $inc[$dirname][$file] = $inc[''][$dirname . $file] = true;
937 }
938 return $GLOBALS['path_files'][$GLOBALS['path_sig']][$dirname][$file];
939 }
940
941 $a = strrpos($file,'/');
942 if ($a !== false) {
943 $dirname .= substr($file, 0, ++$a);
944 $file = substr($file, $a);
945 }
946
947 foreach(creer_chemin() as $dir) {
948 if (!isset($dirs[$a = $dir . $dirname]))
949 $dirs[$a] = (is_dir(_ROOT_CWD . $a) || !$a) ;
950 if ($dirs[$a]) {
951 if (file_exists(_ROOT_CWD . ($a .= $file))) {
952 if ($include AND !isset($inc[$dirname][$file])) {
953 include_once _ROOT_CWD . $a;
954 $inc[$dirname][$file] = $inc[''][$dirname . $file] = true;
955 }
956 if (!defined('_SAUVER_CHEMIN'))
957 define('_SAUVER_CHEMIN',true);
958 return $GLOBALS['path_files'][$GLOBALS['path_sig']][$dirname][$file] = $GLOBALS['path_files'][$GLOBALS['path_sig']][''][$dirname . $file] = $a;
959 }
960 }
961 }
962
963 if ($include){
964 spip_log("include_spip $dirname$file non trouve");
965 if ($include==='required'){
966 echo '<pre>',
967 "<strong>Erreur Fatale</strong><br />";
968 if (function_exists('debug_print_backtrace'))
969 echo debug_print_backtrace();
970 echo '</pre>';
971 die("Erreur interne: ne peut inclure $dirname$file");
972 }
973 }
974
975 if (!defined('_SAUVER_CHEMIN'))
976 define('_SAUVER_CHEMIN',true);
977 return $GLOBALS['path_files'][$GLOBALS['path_sig']][$dirname][$file] = $GLOBALS['path_files'][$GLOBALS['path_sig']][''][$dirname . $file] = false;
978 }
979
980 function clear_path_cache(){
981 $GLOBALS['path_files'] = array();
982 spip_unlink(_CACHE_CHEMIN);
983 }
984 function load_path_cache(){
985 // charger le path des plugins
986 if (@is_readable(_CACHE_PLUGINS_PATH)){
987 include_once(_CACHE_PLUGINS_PATH);
988 }
989 $GLOBALS['path_files'] = array();
990 // si le visiteur est admin,
991 // on ne recharge pas le cache pour forcer sa mise a jour
992 if (
993 // la session n'est pas encore chargee a ce moment, on ne peut donc pas s'y fier
994 //AND (!isset($GLOBALS['visiteur_session']['statut']) OR $GLOBALS['visiteur_session']['statut']!='0minirezo')
995 // utiliser le cookie est un pis aller qui marche 'en general'
996 // on blinde par un second test au moment de la lecture de la session
997 // !isset($_COOKIE[$GLOBALS['cookie_prefix'].'_admin'])
998 // et en ignorant ce cache en cas de recalcul explicite
999 !_request('var_mode')
1000 ){
1001 // on essaye de lire directement sans verrou pour aller plus vite
1002 if ($contenu = spip_file_get_contents(_CACHE_CHEMIN)){
1003 // mais si semble corrompu on relit avec un verrou
1004 if (!$GLOBALS['path_files']=unserialize($contenu)){
1005 lire_fichier(_CACHE_CHEMIN,$contenu);
1006 if (!$GLOBALS['path_files']=unserialize($contenu))
1007 $GLOBALS['path_files'] = array();
1008 }
1009 }
1010 }
1011 }
1012
1013 function save_path_cache(){
1014 if (defined('_SAUVER_CHEMIN')
1015 AND _SAUVER_CHEMIN)
1016 ecrire_fichier(_CACHE_CHEMIN,serialize($GLOBALS['path_files']));
1017 }
1018
1019
1020 /**
1021 * Trouve tous les fichiers du path correspondants a un pattern
1022 * pour un nom de fichier donne, ne retourne que le premier qui sera trouve
1023 * par un find_in_path
1024 *
1025 * @param string $dir
1026 * @param string $pattern
1027 * @param bool $recurs
1028 * @return array
1029 */
1030 // http://doc.spip.org/@find_all_in_path
1031 function find_all_in_path($dir,$pattern, $recurs=false){
1032 $liste_fichiers=array();
1033 $maxfiles = 10000;
1034
1035 // Parcourir le chemin
1036 foreach (creer_chemin() as $d) {
1037 $f = $d.$dir;
1038 if (@is_dir($f)){
1039 $liste = preg_files($f,$pattern,$maxfiles-count($liste_fichiers),$recurs===true?array():$recurs);
1040 foreach($liste as $chemin){
1041 $nom = basename($chemin);
1042 // ne prendre que les fichiers pas deja trouves
1043 // car find_in_path prend le premier qu'il trouve,
1044 // les autres sont donc masques
1045 if (!isset($liste_fichiers[$nom]))
1046 $liste_fichiers[$nom] = $chemin;
1047 }
1048 }
1049 }
1050 return $liste_fichiers;
1051 }
1052
1053 // predicat sur les scripts de ecrire qui n'authentifient pas par cookie
1054
1055 // http://doc.spip.org/@autoriser_sans_cookie
1056 function autoriser_sans_cookie($nom)
1057 {
1058 static $autsanscookie = array('install', 'base_repair');
1059 $nom = preg_replace('/.php[3]?$/', '', basename($nom));
1060 return in_array($nom, $autsanscookie);
1061 }
1062
1063 /**
1064 * Fonction codant et decodant les URLS des objets SQL mis en page par SPIP
1065 *
1066 * http://doc.spip.org/@generer_url_entite
1067 *
1068 *
1069 * @param string $id
1070 * numero de la cle primaire si nombre, URL a decoder si pas numerique
1071 * @param string $entite
1072 * surnom de la table SQL (donne acces au nom de cle primaire)
1073 * @param string $args
1074 * query_string a placer apres cle=$id&....
1075 * @param string $ancre
1076 * ancre a mettre a la fin de l'URL a produire
1077 * @param bool|string $public
1078 * produire l'URL publique ou privee (par defaut: selon espace)
1079 * si string : serveur de base de donnee (nom du connect)
1080 * @param string $type
1081 * fichier dans le repertoire ecrire/urls determinant l'apparence
1082 * @return string|array
1083 * url codee ou fonction de decodage
1084 * array : derogatoire, la fonction d'url retourne (objet,id_objet) utilises par nettoyer_raccourcis_typo() pour generer un lien titre
1085 * (cas des raccourcis personalises [->spip20] : il faut implementer une fonction generer_url_spip et une fonction generer_url_ecrire_spip)
1086 */
1087 function generer_url_entite($id='', $entite='', $args='', $ancre='', $public=NULL, $type=NULL)
1088 {
1089 if ($public === NULL) $public = !test_espace_prive();
1090 $entite = objet_type($entite); // cas particulier d'appels sur objet/id_objet...
1091
1092 if (!$public) {
1093 if (!$entite) return '';
1094 if (!function_exists('generer_url_ecrire_objet'))
1095 include_spip('inc/urls');
1096 $res = generer_url_ecrire_objet($entite,$id, $args, $ancre, false);
1097 } else {
1098 if ($type === NULL) {
1099 $type = ($GLOBALS['type_urls'] === 'page'
1100 AND $GLOBALS['meta']['type_urls'])
1101 ? $GLOBALS['meta']['type_urls']
1102 : $GLOBALS['type_urls']; // pour SPIP <2
1103 }
1104
1105 $f = charger_fonction($type, 'urls', true);
1106 // se rabattre sur les urls page si les urls perso non dispo
1107 if (!$f) $f = charger_fonction('page', 'urls', true);
1108
1109 // si $entite='', on veut la fonction de passage URL ==> id
1110 // sinon on veut effectuer le passage id ==> URL
1111 if (!$entite) return $f;
1112
1113 // mais d'abord il faut tester le cas des urls sur une
1114 // base distante
1115 if (is_string($public)
1116 AND $g = charger_fonction('connect', 'urls', true))
1117 $f = $g;
1118
1119 $res = $f(intval($id), $entite, $args, $ancre, $public);
1120
1121 }
1122 if ($res) return $res;
1123 // Sinon c'est un raccourci ou compat SPIP < 2
1124 if (!function_exists($f = 'generer_url_' . $entite)) {
1125 if (!function_exists($f .= '_dist')) $f = '';
1126 }
1127 if ($f) {
1128 $url = $f($id, $args, $ancre);
1129 if (strlen($args))
1130 $url .= strstr($url, '?')
1131 ? '&amp;'.$args
1132 : '?'.$args;
1133 return $url;
1134 }
1135 // On a ete gentil mais la ....
1136 spip_log("generer_url_entite: entite $entite ($f) inconnue $type $public");
1137 return '';
1138 }
1139
1140 function generer_url_ecrire_entite_edit($id, $entite, $args='', $ancre=''){
1141 $exec = objet_info($entite,'url_edit');
1142 $url = generer_url_ecrire($exec,$args);
1143 if (intval($id))
1144 $url = parametre_url($url,id_table_objet($entite),$id);
1145 else
1146 $url = parametre_url($url,'new','oui');
1147 if ($ancre)
1148 $url = ancre_url($url,$ancre);
1149 return $url;
1150 }
1151
1152 // http://doc.spip.org/@urls_connect_dist
1153 function urls_connect_dist($i, &$entite, $args='', $ancre='', $public=null) {
1154 include_spip('base/connect_sql');
1155 $id_type = id_table_objet($entite,$public);
1156 return _DIR_RACINE . get_spip_script('./')
1157 . "?"._SPIP_PAGE."=$entite&$id_type=$i&connect=$public"
1158 . (!$args ? '' : "&$args")
1159 . (!$ancre ? '' : "#$ancre");
1160 }
1161
1162
1163 // Transformer les caracteres utf8 d'une URL (farsi par ex) selon la RFC 1738
1164 function urlencode_1738($url) {
1165 if (preg_match(',[^\x00-\x7E],sS', $url)){
1166 $uri = '';
1167 for ($i=0; $i < strlen($url); $i++) {
1168 if (ord($a = $url[$i]) > 127)
1169 $a = rawurlencode($a);
1170 $uri .= $a;
1171 }
1172 $url = $uri;
1173 }
1174 return quote_amp($url);
1175 }
1176
1177 // http://doc.spip.org/@generer_url_entite_absolue
1178 function generer_url_entite_absolue($id='', $entite='', $args='', $ancre='', $connect=NULL)
1179 {
1180 if (!$connect) $connect = true;
1181 $h = generer_url_entite($id, $entite, $args, $ancre, $connect);
1182 if (!preg_match(',^\w+:,', $h)) {
1183 include_spip('inc/filtres_mini');
1184 $h = url_absolue($h);
1185 }
1186 return $h;
1187 }
1188
1189 // Sur certains serveurs, la valeur 'Off' tient lieu de false dans certaines
1190 // variables d'environnement comme $_SERVER[HTTPS] ou ini_get(register_globals)
1191 // http://doc.spip.org/@test_valeur_serveur
1192 function test_valeur_serveur($truc) {
1193 if (!$truc) return false;
1194 return (strtolower($truc) !== 'off');
1195 }
1196
1197 //
1198 // Fonctions de fabrication des URL des scripts de Spip
1199 //
1200
1201 // l'URL de base du site, sans se fier a meta(adresse_site) qui
1202 // peut etre fausse (sites a plusieurs noms d'hotes, deplacements, erreurs)
1203 // Note : la globale $profondeur_url doit etre initialisee de maniere a
1204 // indiquer le nombre de sous-repertoires de l'url courante par rapport a la
1205 // racine de SPIP : par exemple, sur ecrire/ elle vaut 1, sur sedna/ 1, et a
1206 // la racine 0. Sur url/perso/ elle vaut 2
1207 // http://doc.spip.org/@url_de_base
1208 function url_de_base() {
1209
1210 static $url = array();
1211
1212 if (isset($url[$GLOBALS['profondeur_url']]))
1213 return $url[$GLOBALS['profondeur_url']];
1214
1215 $http = (
1216 (isset($_SERVER["SCRIPT_URI"]) AND
1217 substr($_SERVER["SCRIPT_URI"],0,5) == 'https')
1218 OR (isset($_SERVER['HTTPS']) AND
1219 test_valeur_serveur($_SERVER['HTTPS']))
1220 ) ? 'https' : 'http';
1221 # note : HTTP_HOST contient le :port si necessaire
1222 $host = $_SERVER['HTTP_HOST'];
1223 if (isset($_SERVER['SERVER_PORT'])
1224 AND $port=$_SERVER['SERVER_PORT']
1225 AND strpos($host,":")==false){
1226 if ($http=="http" AND $port!=80) $host.=":$port";
1227 if ($http=="https" AND $port!=443) $host.=":$port";
1228 }
1229 if (!$GLOBALS['REQUEST_URI']){
1230 if (isset($_SERVER['REQUEST_URI'])) {
1231 $GLOBALS['REQUEST_URI'] = $_SERVER['REQUEST_URI'];
1232 } else {
1233 $GLOBALS['REQUEST_URI'] = $_SERVER['PHP_SELF'];
1234 if ($_SERVER['QUERY_STRING']
1235 AND !strpos($_SERVER['REQUEST_URI'], '?'))
1236 $GLOBALS['REQUEST_URI'] .= '?'.$_SERVER['QUERY_STRING'];
1237 }
1238 }
1239
1240 $url[$GLOBALS['profondeur_url']] = url_de_($http,$host,$GLOBALS['REQUEST_URI'],$GLOBALS['profondeur_url']);
1241
1242 return $url[$GLOBALS['profondeur_url']];
1243 }
1244 /**
1245 * fonction testable de construction d'une url appelee par url_de_base()
1246 * @param string $http
1247 * @param string $host
1248 * @param string $request
1249 * @param int $prof
1250 * @return string
1251 */
1252 function url_de_($http,$host,$request,$prof=0){
1253 $prof = max($prof,0);
1254
1255 $myself = ltrim($request,'/');
1256 # supprimer la chaine de GET
1257 list($myself) = explode('?', $myself);
1258 $url = join('/', array_slice(explode('/', $myself), 0, -1-$prof)).'/';
1259
1260 $url = $http.'://'.rtrim($host,'/').'/'.ltrim($url,'/');
1261 return $url;
1262 }
1263
1264
1265 // Pour une redirection, la liste des arguments doit etre separee par "&"
1266 // Pour du code XHTML, ca doit etre &amp;
1267 // Bravo au W3C qui n'a pas ete capable de nous eviter ca
1268 // faute de separer proprement langage et meta-langage
1269
1270 // Attention, X?y=z et "X/?y=z" sont completement differents!
1271 // http://httpd.apache.org/docs/2.0/mod/mod_dir.html
1272
1273 // http://doc.spip.org/@generer_url_ecrire
1274 function generer_url_ecrire($script='', $args="", $no_entities=false, $rel=false) {
1275 if (!$rel)
1276 $rel = url_de_base() . _DIR_RESTREINT_ABS . _SPIP_ECRIRE_SCRIPT;
1277 else if (!is_string($rel))
1278 $rel = _DIR_RESTREINT ? _DIR_RESTREINT :
1279 ('./' . _SPIP_ECRIRE_SCRIPT);
1280
1281 @list($script, $ancre) = explode('#', $script);
1282 if ($script AND ($script<>'accueil' OR $rel))
1283 $args = "?exec=$script" . (!$args ? '' : "&$args");
1284 elseif ($args)
1285 $args ="?$args";
1286 if ($ancre) $args .= "#$ancre";
1287 return $rel . ($no_entities ? $args : str_replace('&', '&amp;', $args));
1288 }
1289
1290 // http://doc.spip.org/@generer_url_retour
1291 function generer_url_retour($script, $args="")
1292 {
1293 return rawurlencode(generer_url_ecrire($script, $args, true, true));
1294 }
1295
1296 //
1297 // Adresse des scripts publics (a passer dans inc-urls...)
1298 //
1299
1300 // Detecter le fichier de base, a la racine, comme etant spip.php ou ''
1301 // dans le cas de '', un $default = './' peut servir (comme dans urls/page.php)
1302 // http://doc.spip.org/@get_spip_script
1303 function get_spip_script($default='') {
1304 # cas define('_SPIP_SCRIPT', '');
1305 if (_SPIP_SCRIPT)
1306 return _SPIP_SCRIPT;
1307 else
1308 return $default;
1309 }
1310
1311 // http://doc.spip.org/@generer_url_public
1312 function generer_url_public($script='', $args="", $no_entities=false, $rel=true, $action='') {
1313 // si le script est une action (spip_pass, spip_inscription),
1314 // standardiser vers la nouvelle API
1315
1316 if (!$action) $action = get_spip_script();
1317 if ($script)
1318 $action = parametre_url($action, _SPIP_PAGE, $script, '&');
1319
1320 if ($args) {
1321 if (is_array($args)) {
1322 $r = '';
1323 foreach($args as $k => $v) $r .= '&' . $k . '=' . $v;
1324 $args = substr($r,1);
1325 }
1326 $action .=
1327 (strpos($action, '?') !== false ? '&' : '?') . $args;
1328 }
1329 if (!$no_entities)
1330 $action = quote_amp($action);
1331
1332 // ne pas generer une url avec /./?page= en cas d'url absolue et de _SPIP_SCRIPT vide
1333 return ($rel ? _DIR_RACINE . $action : rtrim(url_de_base(),'/') . preg_replace(",^/[.]/,","/","/$action"));
1334 }
1335
1336 // http://doc.spip.org/@generer_url_prive
1337 function generer_url_prive($script, $args="", $no_entities=false) {
1338
1339 return generer_url_public($script, $args, $no_entities, false, _DIR_RESTREINT_ABS . 'prive.php');
1340 }
1341
1342 // Pour les formulaires en methode POST,
1343 // mettre le nom du script a la fois en input-hidden et dans le champ action:
1344 // 1) on peut ainsi memoriser le signet comme si c'etait un GET
1345 // 2) ca suit http://en.wikipedia.org/wiki/Representational_State_Transfer
1346
1347 // http://doc.spip.org/@generer_form_ecrire
1348 function generer_form_ecrire($script, $corps, $atts='', $submit='') {
1349 global $spip_lang_right;
1350
1351 $script1 = explode('&', $script);
1352 $script1 = reset($script1);
1353
1354 return "<form action='"
1355 . ($script ? generer_url_ecrire($script) : '')
1356 . "' "
1357 . ($atts ? $atts : " method='post'")
1358 . "><div>\n"
1359 . "<input type='hidden' name='exec' value='$script1' />"
1360 . $corps
1361 . (!$submit ? '' :
1362 ("<div style='text-align: $spip_lang_right'><input class='fondo' type='submit' value=\"".entites_html($submit)."\" /></div>"))
1363 . "</div></form>\n";
1364 }
1365
1366 /**
1367 * Generer un formulaire pour lancer une action vers $script
1368 *
1369 * Attention, JS/Ajax n'aime pas le melange de param GET/POST
1370 * On n'applique pas la recommandation ci-dessus pour les scripts publics
1371 * qui ne sont pas destines a etre mis en signets
1372 * http://doc.spip.org/@generer_form_action
1373 *
1374 * @param string $script
1375 * @param string $corps
1376 * @param string $atts
1377 * @param bool $public
1378 * @return string
1379 */
1380 function generer_form_action($script, $corps, $atts='', $public=false) {
1381 // si l'on est dans l'espace prive, on garde dans l'url
1382 // l'exec a l'origine de l'action, qui permet de savoir si il est necessaire
1383 // ou non de proceder a l'authentification (cas typique de l'install par exemple)
1384 $h = (_DIR_RACINE AND !$public)
1385 ? generer_url_ecrire(_request('exec'))
1386 : generer_url_public();
1387
1388 return "\n<form action='" .
1389 $h .
1390 "'" .
1391 $atts .
1392 ">\n" .
1393 "<div>" .
1394 "\n<input type='hidden' name='action' value='$script' />" .
1395 $corps .
1396 "</div></form>";
1397 }
1398
1399 // http://doc.spip.org/@generer_url_action
1400 function generer_url_action($script, $args="", $no_entities=false , $public = false) {
1401 // si l'on est dans l'espace prive, on garde dans l'url
1402 // l'exec a l'origine de l'action, qui permet de savoir si il est necessaire
1403 // ou non de proceder a l'authentification (cas typique de l'install par exemple)
1404 $url = (_DIR_RACINE AND !$public)
1405 ? generer_url_ecrire(_request('exec'))
1406 : generer_url_public('','',false,false);
1407 $url = parametre_url($url,'action',$script);
1408 if ($args) $url .= quote_amp('&'.$args);
1409
1410 if ($no_entities) $url = str_replace('&amp;','&',$url);
1411 return $url;
1412 }
1413
1414
1415 /**
1416 * Fonction d'initialisation groupee pour compatibilite ascendante
1417 *
1418 * @param string $pi
1419 * @param string $pa
1420 * @param string $ti
1421 * @param string $ta
1422 */
1423 function spip_initialisation($pi=NULL, $pa=NULL, $ti=NULL, $ta=NULL) {
1424 spip_initialisation_core($pi,$pa,$ti,$ta);
1425 spip_initialisation_suite();
1426 }
1427
1428 /**
1429 * Fonction d'initialisation, appellee dans inc_version ou mes_options
1430 * Elle definit les repertoires et fichiers non partageables
1431 * et indique dans $test_dirs ceux devant etre accessibles en ecriture
1432 * mais ne touche pas a cette variable si elle est deja definie
1433 * afin que mes_options.php puisse en specifier d'autres.
1434 * Elle definit ensuite les noms des fichiers et les droits.
1435 * Puis simule un register_global=on securise.
1436 *
1437 * @param string $pi
1438 * @param string $pa
1439 * @param string $ti
1440 * @param string $ta
1441 */
1442 function spip_initialisation_core($pi=NULL, $pa=NULL, $ti=NULL, $ta=NULL) {
1443 static $too_late = 0;
1444 if ($too_late++) return;
1445
1446 // Declaration des repertoires
1447
1448 // le nom du repertoire plugins/ activables/desactivables
1449 if (!defined('_DIR_PLUGINS')) define('_DIR_PLUGINS', _DIR_RACINE . "plugins/");
1450
1451 // le nom du repertoire des extensions/ permanentes du core, toujours actives
1452 if (!defined('_DIR_PLUGINS_DIST')) define('_DIR_PLUGINS_DIST', _DIR_RACINE . "plugins-dist/");
1453
1454 // le nom du repertoire des librairies
1455 if (!defined('_DIR_LIB')) define('_DIR_LIB', _DIR_RACINE . "lib/");
1456
1457 if (!defined('_DIR_IMG')) define('_DIR_IMG', $pa);
1458 if (!defined('_DIR_LOGOS')) define('_DIR_LOGOS', $pa);
1459 if (!defined('_DIR_IMG_ICONES')) define('_DIR_IMG_ICONES', _DIR_LOGOS . "icones/");
1460
1461 if (!defined('_DIR_DUMP')) define('_DIR_DUMP', $ti . "dump/");
1462 if (!defined('_DIR_SESSIONS')) define('_DIR_SESSIONS', $ti . "sessions/");
1463 if (!defined('_DIR_TRANSFERT')) define('_DIR_TRANSFERT', $ti . "upload/");
1464 if (!defined('_DIR_CACHE')) define('_DIR_CACHE', $ti . "cache/");
1465 if (!defined('_DIR_CACHE_XML')) define('_DIR_CACHE_XML', _DIR_CACHE . "xml/");
1466 if (!defined('_DIR_SKELS')) define('_DIR_SKELS', _DIR_CACHE . "skel/");
1467 if (!defined('_DIR_AIDE')) define('_DIR_AIDE', _DIR_CACHE . "aide/");
1468 if (!defined('_DIR_TMP')) define('_DIR_TMP', $ti);
1469
1470 if (!defined('_DIR_VAR')) define('_DIR_VAR', $ta);
1471
1472 if (!defined('_DIR_ETC')) define('_DIR_ETC', $pi);
1473 if (!defined('_DIR_CONNECT')) define('_DIR_CONNECT', $pi);
1474 if (!defined('_DIR_CHMOD')) define('_DIR_CHMOD', $pi);
1475
1476 if (!isset($GLOBALS['test_dirs']))
1477 // Pas $pi car il est bon de le mettre hors ecriture apres intstall
1478 // il sera rajoute automatiquement si besoin a l'etape 2 de l'install
1479 $GLOBALS['test_dirs'] = array($pa, $ti, $ta);
1480
1481 // Declaration des fichiers
1482
1483 if (!defined('_CACHE_PLUGINS_PATH')) define('_CACHE_PLUGINS_PATH', _DIR_CACHE . "charger_plugins_chemins.php");
1484 if (!defined('_CACHE_PLUGINS_OPT')) define('_CACHE_PLUGINS_OPT', _DIR_CACHE . "charger_plugins_options.php");
1485 if (!defined('_CACHE_PLUGINS_FCT')) define('_CACHE_PLUGINS_FCT', _DIR_CACHE . "charger_plugins_fonctions.php");
1486 if (!defined('_CACHE_PIPELINES')) define('_CACHE_PIPELINES', _DIR_CACHE."charger_pipelines.php");
1487 if (!defined('_CACHE_CHEMIN')) define('_CACHE_CHEMIN', _DIR_CACHE."chemin.txt");
1488
1489 # attention .php obligatoire pour ecrire_fichier_securise
1490 if (!defined('_FILE_META')) define('_FILE_META', $ti . 'meta_cache.php');
1491 if (!defined('_DIR_LOG')) define('_DIR_LOG', _DIR_TMP . 'log/');
1492 if (!defined('_FILE_LOG')) define('_FILE_LOG', 'spip');
1493 if (!defined('_FILE_LOG_SUFFIX')) define('_FILE_LOG_SUFFIX', '.log');
1494
1495 // Le fichier de connexion a la base de donnees
1496 // tient compte des anciennes versions (inc_connect...)
1497 if (!defined('_FILE_CONNECT_INS')) define('_FILE_CONNECT_INS', 'connect');
1498 if (!defined('_FILE_CONNECT')) define('_FILE_CONNECT',
1499 (@is_readable($f = _DIR_CONNECT . _FILE_CONNECT_INS . '.php') ? $f
1500 : (@is_readable($f = _DIR_RESTREINT . 'inc_connect.php') ? $f
1501 : false)));
1502
1503 // Le fichier de reglages des droits
1504 if (!defined('_FILE_CHMOD_INS')) define('_FILE_CHMOD_INS', 'chmod');
1505 if (!defined('_FILE_CHMOD')) define('_FILE_CHMOD',
1506 (@is_readable($f = _DIR_CHMOD . _FILE_CHMOD_INS . '.php') ? $f
1507 : false));
1508
1509 if (!defined('_FILE_LDAP')) define('_FILE_LDAP', 'ldap.php');
1510
1511 if (!defined('_FILE_TMP_SUFFIX')) define('_FILE_TMP_SUFFIX', '.tmp.php');
1512 if (!defined('_FILE_CONNECT_TMP')) define('_FILE_CONNECT_TMP', _DIR_CONNECT . _FILE_CONNECT_INS . _FILE_TMP_SUFFIX);
1513 if (!defined('_FILE_CHMOD_TMP')) define('_FILE_CHMOD_TMP', _DIR_CHMOD . _FILE_CHMOD_INS . _FILE_TMP_SUFFIX);
1514
1515 // Definition des droits d'acces en ecriture
1516 if (!defined('_SPIP_CHMOD') AND _FILE_CHMOD)
1517 include_once _FILE_CHMOD;
1518
1519 // Se mefier des fichiers mal remplis!
1520 if (!defined('_SPIP_CHMOD')) define('_SPIP_CHMOD', 0777);
1521
1522 // Le charset par defaut lors de l'installation
1523 if (!defined('_DEFAULT_CHARSET')) define('_DEFAULT_CHARSET', 'utf-8');
1524 if (!defined('_ROOT_PLUGINS')) define('_ROOT_PLUGINS', _ROOT_RACINE . "plugins/");
1525 if (!defined('_ROOT_PLUGINS_DIST')) define('_ROOT_PLUGINS_DIST', _ROOT_RACINE . "plugins-dist/");
1526 if (!defined('_ROOT_PLUGINS_SUPPL') && defined('_DIR_PLUGINS_SUPPL') && _DIR_PLUGINS_SUPPL) define('_ROOT_PLUGINS_SUPPL', _ROOT_RACINE . str_replace(_DIR_RACINE,'',_DIR_PLUGINS_SUPPL));
1527
1528 // La taille des Log
1529 if (!defined('_MAX_LOG')) define('_MAX_LOG', 100);
1530
1531 // Sommes-nous dans l'empire du Mal ?
1532 // (ou sous le signe du Pingouin, ascendant GNU ?)
1533 if (strpos($_SERVER['SERVER_SOFTWARE'], '(Win') !== false){
1534 if (!defined('_OS_SERVEUR')) define('_OS_SERVEUR', 'windows');
1535 if (!defined('_SPIP_LOCK_MODE')) define('_SPIP_LOCK_MODE',1); // utiliser le flock php
1536 }
1537 else {
1538 if (!defined('_OS_SERVEUR')) define('_OS_SERVEUR', '');
1539 if (!defined('_SPIP_LOCK_MODE')) define('_SPIP_LOCK_MODE',1); // utiliser le flock php
1540 #if (!defined('_SPIP_LOCK_MODE')) define('_SPIP_LOCK_MODE',2); // utiliser le nfslock de spip mais link() est tres souvent interdite
1541 }
1542
1543 //
1544 // Module de lecture/ecriture/suppression de fichiers utilisant flock()
1545 // (non surchargeable en l'etat ; attention si on utilise include_spip()
1546 // pour le rendre surchargeable, on va provoquer un reecriture
1547 // systematique du noyau ou une baisse de perfs => a etudier)
1548 include_once _ROOT_RESTREINT . 'inc/flock.php';
1549
1550 // charger tout de suite le path et son cache
1551 load_path_cache();
1552
1553 // *********** traiter les variables ************
1554
1555 //
1556 // Securite
1557 //
1558
1559 // Ne pas se faire manger par un bug php qui accepte ?GLOBALS[truc]=toto
1560 if (isset($_REQUEST['GLOBALS'])) die();
1561 // nettoyer les magic quotes \' et les caracteres nuls %00
1562 spip_desinfecte($_GET);
1563 spip_desinfecte($_POST);
1564 spip_desinfecte($_COOKIE);
1565 spip_desinfecte($_REQUEST);
1566
1567 // Par ailleurs on ne veut pas de magic_quotes au cours de l'execution
1568 @set_magic_quotes_runtime(0);
1569
1570 // Si les variables sont passees en global par le serveur,
1571 // il faut faire quelques verifications de base
1572 $avertir_register_globals = false;
1573 if (test_valeur_serveur(@ini_get('register_globals'))) {
1574 // ne pas desinfecter les globales en profondeur car elle contient aussi les
1575 // precedentes, qui seraient desinfectees 2 fois.
1576 spip_desinfecte($GLOBALS,false);
1577 if (include_spip('inc/php3'))
1578 spip_register_globals(true);
1579
1580 $avertir_register_globals = true;
1581 }
1582
1583 // appliquer le cookie_prefix
1584 if ($GLOBALS['cookie_prefix'] != 'spip') {
1585 include_spip('inc/cookie');
1586 recuperer_cookies_spip($GLOBALS['cookie_prefix']);
1587 }
1588
1589 //
1590 // Capacites php (en fonction de la version)
1591 //
1592 $GLOBALS['flag_ob'] = (function_exists("ob_start")
1593 && function_exists("ini_get")
1594 && !strstr(@ini_get('disable_functions'), 'ob_'));
1595 $GLOBALS['flag_sapi_name'] = function_exists("php_sapi_name");
1596 $GLOBALS['flag_get_cfg_var'] = (@get_cfg_var('error_reporting') != "");
1597 $GLOBALS['flag_upload'] = (!$GLOBALS['flag_get_cfg_var'] ||
1598 (get_cfg_var('upload_max_filesize') > 0));
1599
1600
1601 // Compatibilite avec serveurs ne fournissant pas $REQUEST_URI
1602 if (isset($_SERVER['REQUEST_URI'])) {
1603 $GLOBALS['REQUEST_URI'] = $_SERVER['REQUEST_URI'];
1604 } else {
1605 $GLOBALS['REQUEST_URI'] = $_SERVER['PHP_SELF'];
1606 if ($_SERVER['QUERY_STRING']
1607 AND !strpos($_SERVER['REQUEST_URI'], '?'))
1608 $GLOBALS['REQUEST_URI'] .= '?'.$_SERVER['QUERY_STRING'];
1609 }
1610
1611 // Duree de validite de l'alea pour les cookies et ce qui s'ensuit.
1612 if (!defined('_RENOUVELLE_ALEA')) define('_RENOUVELLE_ALEA', 12 * 3600);
1613
1614 // charger les meta si possible et renouveller l'alea au besoin
1615 // charge aussi effacer_meta et ecrire_meta
1616 $inc_meta = charger_fonction('meta', 'inc');
1617 $inc_meta();
1618
1619 // on a pas pu le faire plus tot
1620 if ($avertir_register_globals)
1621 avertir_auteurs("register_globals",_L("Probl&egrave;me de s&eacute;curit&eacute; : register_globals=on; dans php.ini &agrave; corriger."));
1622
1623 // nombre de repertoires depuis la racine
1624 // on compare a l'adresse de spip.php : $_SERVER["SCRIPT_NAME"]
1625 // ou a defaut celle donnee en meta ; (mais si celle-ci est fausse
1626 // le calcul est faux)
1627 if (!_DIR_RESTREINT)
1628 $GLOBALS['profondeur_url'] = 1;
1629 else {
1630 $uri = isset($_SERVER['REQUEST_URI']) ? explode('?', $_SERVER['REQUEST_URI']) : '';
1631 $uri_ref = $_SERVER["SCRIPT_NAME"];
1632 if (!$uri_ref
1633 // si on est appele avec un autre ti, on est sans doute en mutu
1634 // si jamais c'est de la mutu avec sous rep, on est perdu si on se fie
1635 // a spip.php qui est a la racine du spip, et vue qu'on sait pas se reperer
1636 // s'en remettre a l'adresse du site. alea jacta est.
1637 OR $ti!==_NOM_TEMPORAIRES_INACCESSIBLES){
1638
1639 if (isset($GLOBALS['meta']['adresse_site'])) {
1640 $uri_ref = parse_url($GLOBALS['meta']['adresse_site']);
1641 $uri_ref = $uri_ref['path'].'/';
1642 }
1643 else
1644 $uri_ref = "";
1645 }
1646 if (!$uri OR !$uri_ref)
1647 $GLOBALS['profondeur_url'] = 0;
1648 else {
1649 $GLOBALS['profondeur_url'] = max(0,
1650 substr_count($uri[0], '/')
1651 - substr_count($uri_ref,'/'));
1652 }
1653 }
1654 // s'il y a un cookie ou PHP_AUTH, initialiser visiteur_session
1655 if (_FILE_CONNECT) {
1656 if (verifier_visiteur()=='0minirezo'
1657 // si c'est un admin sans cookie admin, il faut ignorer le cache chemin !
1658 AND !isset($_COOKIE['spip_admin']))
1659 clear_path_cache();
1660 }
1661
1662 }
1663
1664 /**
1665 * Complements d'initialisation non critiques pouvant etre realises
1666 * par les plugins
1667 *
1668 */
1669 function spip_initialisation_suite() {
1670 static $too_late = 0;
1671 if ($too_late++) return;
1672
1673 // taille mini des login
1674 if (!defined('_LOGIN_TROP_COURT')) define('_LOGIN_TROP_COURT', 4);
1675
1676 // la taille maxi des logos (0 : pas de limite)
1677 if (!defined('_LOGO_MAX_SIZE')) define('_LOGO_MAX_SIZE', 0); # poids en ko
1678 if (!defined('_LOGO_MAX_WIDTH')) define('_LOGO_MAX_WIDTH', 0); # largeur en pixels
1679 if (!defined('_LOGO_MAX_HEIGHT')) define('_LOGO_MAX_HEIGHT', 0); # hauteur en pixels
1680
1681 if (!defined('_DOC_MAX_SIZE')) define('_DOC_MAX_SIZE', 0); # poids en ko
1682
1683 if (!defined('_IMG_MAX_SIZE')) define('_IMG_MAX_SIZE', 0); # poids en ko
1684 if (!defined('_IMG_MAX_WIDTH')) define('_IMG_MAX_WIDTH', 0); # largeur en pixels
1685 if (!defined('_IMG_MAX_HEIGHT')) define('_IMG_MAX_HEIGHT', 0); # hauteur en pixels
1686 if (!defined('_PASS_LONGUEUR_MINI')) define('_PASS_LONGUEUR_MINI',6);
1687
1688
1689 // Qualite des images calculees automatiquement. C'est un nombre entre 0 et 100, meme pour imagick (on ramene a 0..1 par la suite)
1690 if (!defined('_IMG_QUALITE')) define('_IMG_QUALITE', 85); # valeur par defaut
1691 if (!defined('_IMG_GD_QUALITE')) define('_IMG_GD_QUALITE', _IMG_QUALITE); # surcharge pour la lib GD
1692 if (!defined('_IMG_CONVERT_QUALITE')) define('_IMG_CONVERT_QUALITE', _IMG_QUALITE); # surcharge pour imagick en ligne de commande
1693 // Historiquement la valeur pour imagick semble differente. Si ca n'est pas necessaire, il serait preferable de garder _IMG_QUALITE
1694 if (!defined('_IMG_IMAGICK_QUALITE')) define('_IMG_IMAGICK_QUALITE', 75); # surcharge pour imagick en PHP
1695
1696 if (!defined('_COPIE_LOCALE_MAX_SIZE')) define('_COPIE_LOCALE_MAX_SIZE',16777216); // poids en octet
1697
1698 // qq chaines standard
1699 if (!defined('_ACCESS_FILE_NAME')) define('_ACCESS_FILE_NAME', '.htaccess');
1700 if (!defined('_AUTH_USER_FILE')) define('_AUTH_USER_FILE', '.htpasswd');
1701 if (!defined('_SPIP_DUMP')) define('_SPIP_DUMP', 'dump@nom_site@@stamp@.xml');
1702 if (!defined('_CACHE_RUBRIQUES')) define('_CACHE_RUBRIQUES', _DIR_TMP.'menu-rubriques-cache.txt');
1703 if (!defined('_CACHE_RUBRIQUES_MAX')) define('_CACHE_RUBRIQUES_MAX', 500);
1704
1705 if (!defined('_EXTENSION_SQUELETTES')) define('_EXTENSION_SQUELETTES', 'html');
1706
1707 if (!defined('_DOCTYPE_ECRIRE')) define('_DOCTYPE_ECRIRE',
1708 // "<!DOCTYPE HTML PUBLIC '-//W3C//DTD HTML 4.01 Transitional//EN' 'http://www.w3.org/TR/html4/loose.dtd'>\n");
1709 //"<!DOCTYPE html PUBLIC '-//W3C//DTD XHTML 1.0 Transitional//EN' 'http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd'>\n");
1710 //"<!DOCTYPE html PUBLIC '-//W3C//DTD XHTML 1.0 Strict//EN' 'http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd'>\n");
1711 // "<!DOCTYPE html PUBLIC '-//W3C//DTD XHTML 1.1 //EN' 'http://www.w3.org/TR/xhtml11/DTD/xhtml11.dtd'>\n");
1712 "<!DOCTYPE html>\n");
1713 if (!defined('_DOCTYPE_AIDE')) define('_DOCTYPE_AIDE',
1714 "<!DOCTYPE html PUBLIC '-//W3C//DTD HTML 4.01 Frameset//EN' 'http://www.w3.org/TR/1999/REC-html401-19991224/frameset.dtd'>");
1715
1716 // L'adresse de base du site ; on peut mettre '' si la racine est geree par
1717 // le script de l'espace public, alias index.php
1718 if (!defined('_SPIP_SCRIPT')) define('_SPIP_SCRIPT', 'spip.php');
1719 // argument page, personalisable en cas de conflit avec un autre script
1720 if (!defined('_SPIP_PAGE')) define('_SPIP_PAGE', 'page');
1721
1722 // le script de l'espace prive
1723 // Mettre a "index.php" si DirectoryIndex ne le fait pas ou pb connexes:
1724 // les anciens IIS n'acceptent pas les POST sur ecrire/ (#419)
1725 // meme pb sur thttpd cf. http://forum.spip.org/fr_184153.html
1726
1727 if (!defined('_SPIP_ECRIRE_SCRIPT')) define('_SPIP_ECRIRE_SCRIPT', // true ? #decommenter ici et commenter la
1728 preg_match(',IIS|thttpd,',$_SERVER['SERVER_SOFTWARE']) ?
1729 'index.php' : '');
1730
1731
1732 if (!defined('_SPIP_AJAX'))
1733 define('_SPIP_AJAX', ((!isset($_COOKIE['spip_accepte_ajax']))
1734 ? 1
1735 : (($_COOKIE['spip_accepte_ajax'] != -1) ? 1 : 0)));
1736
1737 // La requete est-elle en ajax ?
1738 if (!defined('_AJAX')) define('_AJAX',
1739 (isset($_SERVER['HTTP_X_REQUESTED_WITH']) # ajax jQuery
1740 OR @$_REQUEST['var_ajax_redir'] # redirection 302 apres ajax jQuery
1741 OR @$_REQUEST['var_ajaxcharset'] # compat ascendante pour plugins
1742 OR @$_REQUEST['var_ajax'] # forms ajax & inclure ajax de spip
1743 )
1744 AND !@$_REQUEST['var_noajax'] # horrible exception, car c'est pas parce que la requete est ajax jquery qu'il faut tuer tous les formulaires ajax qu'elle contient
1745 );
1746
1747 # nombre de pixels maxi pour calcul de la vignette avec gd
1748 # au dela de 5500000 on considere que php n'est pas limite en memoire pour cette operation
1749 # les configurations limitees en memoire ont un seuil plutot vers 1MPixel
1750 if (!defined('_IMG_GD_MAX_PIXELS')) define('_IMG_GD_MAX_PIXELS',
1751 (isset($GLOBALS['meta']['max_taille_vignettes'])&&$GLOBALS['meta']['max_taille_vignettes']<5500000)
1752 ? $GLOBALS['meta']['max_taille_vignettes']
1753 : 0);
1754
1755 if (!defined('_MEMORY_LIMIT_MIN')) define('_MEMORY_LIMIT_MIN',10); // en Mo
1756 // si on est dans l'espace prive et si le besoin est superieur a 8Mo (qui est vraiment le standard)
1757 // on verifie que la memoire est suffisante pour le compactage css+js pour eviter la page blanche
1758 // il y aura d'autres problemes et l'utilisateur n'ira pas tres loin, mais ce sera plus comprehensible qu'une page blanche
1759 if (test_espace_prive() AND _MEMORY_LIMIT_MIN>8){
1760 if ($memory = trim(ini_get('memory_limit'))){
1761 $unit = strtolower(substr($memory,strlen($memory/1),1));
1762 switch($unit) {
1763 // Le modifieur 'G' est disponible depuis PHP 5.1.0
1764 case 'g': $memory *= 1024;
1765 case 'm': $memory *= 1024;
1766 case 'k': $memory *= 1024;
1767 }
1768 if ($memory<_MEMORY_LIMIT_MIN*1024*1024){
1769 ini_set('memory_limit',$m=_MEMORY_LIMIT_MIN.'M');
1770 if (trim(ini_get('memory_limit'))!=$m){
1771 if (!defined('_INTERDIRE_COMPACTE_HEAD_ECRIRE')) define('_INTERDIRE_COMPACTE_HEAD_ECRIRE',true); // evite une page blanche car on ne saura pas calculer la css dans ce hit
1772 }
1773 }
1774 }
1775 else
1776 if (!defined('_INTERDIRE_COMPACTE_HEAD_ECRIRE')) define('_INTERDIRE_COMPACTE_HEAD_ECRIRE',true); // evite une page blanche car on ne saura pas calculer la css dans ce hit
1777 }
1778
1779 init_var_mode();
1780 }
1781
1782 // Reperer les variables d'URL qui conditionnent la perennite du cache, des urls
1783 // ou d'autres petit caches (trouver_table, css et js compactes ...)
1784 // http://doc.spip.org/@init_var_mode
1785 function init_var_mode(){
1786 static $done = false;
1787 if (!$done) {
1788
1789 if (isset($_GET['var_mode'])) {
1790 // tout le monde peut calcul/recalcul
1791 if ($_GET['var_mode'] == 'calcul'
1792 OR $_GET['var_mode'] == 'recalcul') {
1793 if (!defined('_VAR_MODE')) define('_VAR_MODE',$_GET['var_mode']);
1794 }
1795 // preview, debug, blocs, urls et images necessitent une autorisation
1796 else if (in_array($_GET['var_mode'],array('preview','debug','inclure','urls','images'))) {
1797 include_spip('inc/autoriser');
1798 if (autoriser(
1799 ($_GET['var_mode'] == 'preview')
1800 ? 'previsualiser'
1801 : 'debug'
1802 )) {
1803 switch($_GET['var_mode']){
1804 case 'preview':
1805 // basculer sur les criteres de preview dans les boucles
1806 if (!defined('_VAR_PREVIEW')) define('_VAR_PREVIEW',true);
1807 // forcer le calcul
1808 if (!defined('_VAR_MODE')) define('_VAR_MODE','calcul');
1809 // et ne pas enregistrer de cache
1810 if (!defined('_VAR_NOCACHE')) define('_VAR_NOCACHE',true);
1811 break;
1812 case 'inclure':
1813 // forcer le compilo et ignorer les caches existants
1814 if (!defined('_VAR_MODE')) define('_VAR_MODE','calcul');
1815 if (!defined('_VAR_INCLURE')) define('_VAR_INCLURE',true);
1816 // et ne pas enregistrer de cache
1817 if (!defined('_VAR_NOCACHE')) define('_VAR_NOCACHE',true);
1818 break;
1819 case 'urls':
1820 // forcer le compilo et ignorer les caches existants
1821 if (!defined('_VAR_MODE')) define('_VAR_MODE','calcul');
1822 if (!defined('_VAR_URLS')) define('_VAR_URLS',true);
1823 break;
1824 case 'images':
1825 // forcer le compilo et ignorer les caches existants
1826 if (!defined('_VAR_MODE')) define('_VAR_MODE','calcul');
1827 // indiquer qu'on doit recalculer les images
1828 if (!defined('_VAR_IMAGES')) define('_VAR_IMAGES',true);
1829 break;
1830 case 'debug':
1831 if (!defined('_VAR_MODE')) define('_VAR_MODE','debug');
1832 // et ne pas enregistrer de cache
1833 if (!defined('_VAR_NOCACHE')) define('_VAR_NOCACHE',true);
1834 break;
1835 default :
1836 if (!defined('_VAR_MODE')) define('_VAR_MODE',$_GET['var_mode']);
1837 break;
1838 }
1839 spip_log($GLOBALS['visiteur_session']['nom']
1840 . " "._VAR_MODE);
1841 }
1842 // pas autorise ?
1843 else {
1844 // si on n'est pas connecte on se redirige
1845 if (!$GLOBALS['visiteur_session']) {
1846 include_spip('inc/headers');
1847 redirige_par_entete(generer_url_public('login',
1848 'url='.rawurlencode(
1849 parametre_url(self(), 'var_mode', $_GET['var_mode'], '&')
1850 ), true));
1851 }
1852 // sinon tant pis
1853 }
1854 }
1855 if (!defined('_VAR_MODE')) define('_VAR_MODE',false);
1856 }
1857 $done = true;
1858 }
1859 }
1860
1861 // Annuler les magic quotes \' sur GET POST COOKIE et GLOBALS ;
1862 // supprimer aussi les eventuels caracteres nuls %00, qui peuvent tromper
1863 // la commande is_readable('chemin/vers/fichier/interdit%00truc_normal')
1864 // http://doc.spip.org/@spip_desinfecte
1865 function spip_desinfecte(&$t,$deep = true) {
1866 static $magic_quotes;
1867 if (!isset($magic_quotes))
1868 $magic_quotes = @get_magic_quotes_gpc();
1869
1870 foreach ($t as $key => $val) {
1871 if (is_string($t[$key])) {
1872 if ($magic_quotes)
1873 $t[$key] = stripslashes($t[$key]);
1874 $t[$key] = str_replace(chr(0), '-', $t[$key]);
1875 }
1876 // traiter aussi les "texte_plus" de article_edit
1877 else if ($deep AND is_array($t[$key]) AND $key!=='GLOBALS')
1878 spip_desinfecte($t[$key],$deep);
1879 }
1880 }
1881
1882 // retourne le statut du visiteur s'il s'annonce
1883
1884 // http://doc.spip.org/@verifier_visiteur
1885 function verifier_visiteur() {
1886 // Rq: pour que cette fonction marche depuis mes_options
1887 // il faut forcer l'init si ce n'est fait
1888 // mais on risque de perturber des plugins en initialisant trop tot
1889 // certaines constantes
1890 @spip_initialisation_core(
1891 (_DIR_RACINE . _NOM_PERMANENTS_INACCESSIBLES),
1892 (_DIR_RACINE . _NOM_PERMANENTS_ACCESSIBLES),
1893 (_DIR_RACINE . _NOM_TEMPORAIRES_INACCESSIBLES),
1894 (_DIR_RACINE . _NOM_TEMPORAIRES_ACCESSIBLES)
1895 );
1896
1897 // Demarrer une session NON AUTHENTIFIEE si on donne son nom
1898 // dans un formulaire sans login (ex: #FORMULAIRE_FORUM)
1899 // Attention on separe bien session_nom et nom, pour eviter
1900 // les melanges entre donnees SQL et variables plus aleatoires
1901 $variables_session = array('session_nom', 'session_email');
1902 foreach($variables_session as $var) {
1903 if (_request($var) !== null) {
1904 $init = true;
1905 break;
1906 }
1907 }
1908 if (isset($init)) {
1909 #@spip_initialisation_suite();
1910 $session = charger_fonction('session', 'inc');
1911 $session();
1912 include_spip('inc/texte');
1913 foreach($variables_session as $var)
1914 if (($a = _request($var)) !== null)
1915 $GLOBALS['visiteur_session'][$var] = safehtml($a);
1916 if (!isset($GLOBALS['visiteur_session']['id_auteur']))
1917 $GLOBALS['visiteur_session']['id_auteur'] = 0;
1918 $session($GLOBALS['visiteur_session']);
1919 return 0;
1920 }
1921
1922 $h = (isset($_SERVER['PHP_AUTH_USER']) AND !$GLOBALS['ignore_auth_http']);
1923 if ($h OR isset($_COOKIE['spip_session']) OR isset($_COOKIE[$GLOBALS['cookie_prefix'].'_session'])) {
1924
1925 $session = charger_fonction('session', 'inc');
1926 if ($session()) {
1927 return $GLOBALS['visiteur_session']['statut'];
1928 }
1929 if ($h AND isset($_SERVER['PHP_AUTH_PW'])) {
1930 include_spip('inc/auth');
1931 $h = lire_php_auth($_SERVER['PHP_AUTH_USER'], $_SERVER['PHP_AUTH_PW']);
1932 }
1933 if ($h) {
1934 $GLOBALS['visiteur_session'] = $h;
1935 return $GLOBALS['visiteur_session']['statut'];
1936 }
1937 }
1938
1939 // au moins son navigateur nous dit la langue preferee de cet inconnu
1940 include_spip('inc/lang');
1941 utiliser_langue_visiteur();
1942
1943 return false;
1944 }
1945
1946 // selectionne la langue donnee en argument et memorise la courante
1947 // ou restaure l'ancienne si appel sans argument
1948 // On pourrait economiser l'empilement en cas de non changemnt
1949 // et lui faire retourner False pour prevenir l'appelant
1950 // Le noyau de Spip sait le faire, mais pour assurer la compatibilite
1951 // cette fonction retourne toujours non False
1952
1953 // http://doc.spip.org/@lang_select
1954 function lang_select ($lang=NULL) {
1955 static $pile_langues = array();
1956 if (!function_exists('changer_langue'))
1957 include_spip('inc/lang');
1958 if ($lang === NULL)
1959 $lang = array_pop($pile_langues);
1960 else {
1961 array_push($pile_langues, $GLOBALS['spip_lang']);
1962 }
1963 if (isset($GLOBALS['spip_lang']) AND $lang == $GLOBALS['spip_lang'])
1964 return $lang;
1965 changer_langue($lang);
1966 return $lang;
1967 }
1968
1969
1970 // Renvoie une chaine qui decrit la session courante pour savoir si on peut
1971 // utiliser un cache enregistre pour cette session.
1972 // Par convention cette chaine ne doit pas contenir de caracteres [^0-9A-Za-z]
1973 // Attention on ne peut *pas* inferer id_auteur a partir de la session, qui
1974 // est une chaine arbitraire
1975 // Cette chaine est courte (8 cars) pour pouvoir etre utilisee dans un nom
1976 // de fichier cache
1977 // http://doc.spip.org/@spip_session
1978 function spip_session($force = false) {
1979 static $session;
1980 if ($force OR !isset($session)) {
1981 $s = pipeline('definir_session',
1982 $GLOBALS['visiteur_session']
1983 ? serialize($GLOBALS['visiteur_session'])
1984 . '_' . @$_COOKIE['spip_session']
1985 : ''
1986 );
1987 $session = $s ? substr(md5($s), 0, 8) : '';
1988 }
1989 #spip_log('session: '.$session);
1990 return $session;
1991 }
1992
1993
1994 /**
1995 * Aide, aussi depuis l'espace prive a present.
1996 * Surchargeable mais pas d'erreur fatale si indisponible.
1997 *
1998 * @param string $aide
1999 * Cle d'identification de l'aide desiree
2000 * @param bool $distante
2001 * Generer une url locale (par defaut)
2002 * ou une url distante [directement sur spip.net]
2003 * @return Lien sur une icone d'aide
2004 **/
2005 // http://doc.spip.org/@aide
2006 function aide($aide='', $distante = false) {
2007 $aider = charger_fonction('aider', 'inc', true);
2008 return $aider ? $aider($aide, '', array(), $distante) : '';
2009 }
2010
2011 // normalement il faudrait creer exec/info.php, mais pour mettre juste ca:
2012 // http://doc.spip.org/@exec_info_dist
2013 function exec_info_dist() {
2014 global $connect_statut;
2015 if ($connect_statut == '0minirezo')
2016 phpinfo();
2017 else
2018 echo "pas admin";
2019 }
2020
2021 /**
2022 * Génère une erreur de squelette
2023 *
2024 * Génère une erreur de squelette qui sera bien visible par un
2025 * administrateur authentifié lors d'une visite de la page en erreur
2026 *
2027 * @param bool|string|array $message
2028 * - Message d'erreur (string|array)
2029 * - false pour retourner le texte des messages d'erreurs
2030 * - vide pour afficher les messages d'erreurs
2031 * @param string|array|object $lieu
2032 * Lieu d'origine de l'erreur
2033 * @return null|string
2034 * Rien dans la plupart des cas
2035 * - string si $message à false.
2036 **/
2037 function erreur_squelette($message='', $lieu='') {
2038 $debusquer = charger_fonction('debusquer', 'public');
2039 if (is_array($lieu)) {
2040 include_spip('public/compiler');
2041 $lieu = reconstruire_contexte_compil($lieu);
2042 }
2043 return $debusquer($message, $lieu);
2044 }
2045
2046 /**
2047 * Calcule un squelette avec un contexte et retourne son contenu
2048 *
2049 * La fonction de base de SPIP : un squelette + un contexte => une page.
2050 * $fond peut etre un nom de squelette, ou une liste de squelette au format array.
2051 * Dans ce dernier cas, les squelettes sont tous evalues et mis bout a bout
2052 * $options permet de selectionner les options suivantes :
2053 * trim => true (valeur par defaut) permet de ne rien renvoyer si le fond ne produit que des espaces ;
2054 * raw => true permet de recuperer la strucure $page complete avec entetes et invalideurs
2055 * pour chaque $fond fourni.
2056 *
2057 * @api
2058 * @param string/array $fond
2059 * Le ou les squelettes à utiliser, sans l'extension, {@example prive/liste/auteurs}
2060 * Le fichier sera retrouvé dans la liste des chemins connus de SPIP (squelettes, plugins, spip)
2061 * @param array $contexte
2062 * Informations de contexte envoyées au squelette, {@example array('id_rubrique' => 8)}
2063 * La langue est transmise automatiquement (sauf option étoile).
2064 * @param array $options
2065 * Options complémentaires :
2066 * - trim : applique un trim sur le résultat (true par défaut)
2067 * - raw : retourne un tableau d'information sur le squelette (false par défaut)
2068 * - etoile : ne pas transmettre la langue au contexte automatiquement (false par défaut),
2069 * équivalent de INCLURE*
2070 * - ajax : gere les liens internes du squelette en ajax (équivalent du paramètre {ajax})
2071 * @param string $connect
2072 * Non du connecteur de bdd a utiliser
2073 * @return string|array
2074 * Contenu du squelette calculé
2075 * ou tableau d'information sur le squelette.
2076 */
2077 function recuperer_fond($fond, $contexte=array(), $options = array(), $connect='') {
2078 if (!function_exists('evaluer_fond'))
2079 include_spip('public/assembler');
2080 // assurer la compat avec l'ancienne syntaxe
2081 // (trim etait le 3eme argument, par defaut a true)
2082 if (!is_array($options)) $options = array('trim'=>$options);
2083 if (!isset($options['trim'])) $options['trim']=true;
2084
2085 if (isset($contexte['connect'])){
2086 $connect = ($connect ? $connect : $contexte['connect']);
2087 unset($contexte['connect']);
2088 }
2089
2090 $texte = "";
2091 $pages = array();
2092 $lang_select = '';
2093 if (!isset($options['etoile']) OR !$options['etoile']){
2094 // Si on a inclus sans fixer le critere de lang, on prend la langue courante
2095 if (!isset($contexte['lang']))
2096 $contexte['lang'] = $GLOBALS['spip_lang'];
2097
2098 if ($contexte['lang'] != $GLOBALS['meta']['langue_site']) {
2099 $lang_select = lang_select($contexte['lang']);
2100 }
2101 }
2102
2103 @$GLOBALS['_INC_PUBLIC']++;
2104
2105 foreach(is_array($fond) ? $fond : array($fond) as $f){
2106 $page = evaluer_fond($f, $contexte, $connect);
2107 if ($page === '') {
2108 $c = isset($options['compil']) ? $options['compil'] :'';
2109 $a = array('fichier'=>$fond);
2110 $erreur = _T('info_erreur_squelette2', $a); // squelette introuvable
2111 erreur_squelette($erreur, $c);
2112 // eviter des erreurs strictes ensuite sur $page['cle'] en PHP >= 5.4
2113 $page = array('texte' => '', 'erreur' => $erreur);
2114 }
2115
2116 $page = pipeline('recuperer_fond',array(
2117 'args'=>array('fond'=>$fond,'contexte'=>$contexte,'options'=>$options,'connect'=>$connect),
2118 'data'=>$page
2119 ));
2120 if (isset($options['ajax']) AND $options['ajax']){
2121 if (!function_exists('encoder_contexte_ajax'))
2122 include_spip('inc/filtres');
2123 $page['texte'] = encoder_contexte_ajax(array_merge($contexte,array('fond'=>$f)),'',$page['texte'], $options['ajax']);
2124 }
2125
2126 if (isset($options['raw']) AND $options['raw'])
2127 $pages[] = $page;
2128 else
2129 $texte .= $options['trim'] ? rtrim($page['texte']) : $page['texte'];
2130 }
2131
2132 $GLOBALS['_INC_PUBLIC']--;
2133
2134 if ($lang_select) lang_select();
2135 if (isset($options['raw']) AND $options['raw'])
2136 return is_array($fond)?$pages:reset($pages);
2137 else
2138 return $options['trim'] ? ltrim($texte) : $texte;
2139 }
2140
2141 /**
2142 * Trouve un squelette dans le repertoire modeles/
2143 *
2144 * @param $nom
2145 * @return string
2146 */
2147 function trouve_modele($nom) {
2148 return trouver_fond($nom,'modeles/');
2149 }
2150
2151 /**
2152 * Trouver un squelette dans le chemin
2153 * on peut specifier un sous-dossier dans $dir
2154 * si $pathinfo est a true, retourne un tableau avec
2155 * les composantes du fichier trouve
2156 * + le chemin complet sans son extension dans fond
2157 *
2158 * @param string $nom
2159 * @param string $dir
2160 * @param bool $pathinfo
2161 * @return array|string
2162 */
2163 function trouver_fond($nom, $dir='', $pathinfo = false) {
2164 $f = find_in_path($nom.'.'. _EXTENSION_SQUELETTES, $dir?rtrim($dir,'/').'/':'');
2165 if (!$pathinfo) return $f;
2166 // renvoyer un tableau detaille si $pathinfo==true
2167 $p = pathinfo($f);
2168 if (!isset($p['extension']) OR !$p['extension']) {
2169 $p['extension'] = _EXTENSION_SQUELETTES;
2170 }
2171 if (!isset($p['extension']) OR !$p['filename']) {
2172 $p['filename'] = ($p['basename']?substr($p['basename'],0,-strlen($p['extension'])-1):'');
2173 }
2174 $p['fond'] = ($f?substr($f,0,-strlen($p['extension'])-1):'');
2175 return $p;
2176 }
2177
2178 function tester_url_ecrire($nom){
2179 static $exec=array();
2180 if (isset($exec[$nom])) return $exec[$nom];
2181 // tester si c'est une page en squelette
2182 if (trouver_fond($nom, 'prive/squelettes/contenu/'))
2183 return $exec[$nom] = 'fond';
2184 // compat skels orthogonaux version precedente
2185 elseif (trouver_fond($nom, 'prive/exec/'))
2186 return $exec[$nom] = 'fond_monobloc';
2187 // echafaudage d'un fond !
2188 elseif(include_spip('public/styliser_par_z') AND z_echafaudable($nom))
2189 return $exec[$nom] = 'fond';
2190 // attention, il ne faut pas inclure l'exec ici
2191 // car sinon #URL_ECRIRE provoque des inclusions
2192 // et des define intrusifs potentiels
2193 return $exec[$nom] = ((find_in_path("{$nom}.php",'exec/') OR charger_fonction($nom,'exec',true))?$nom:'');
2194 }
2195
2196 // Charger dynamiquement une extension php
2197 // http://doc.spip.org/@charger_php_extension
2198 function charger_php_extension($module) {
2199 if (extension_loaded($module)) {
2200 return true;
2201 } else {
2202 $charger_php_extension = charger_fonction('charger_php_extension','inc');
2203 return $charger_php_extension($module);
2204 }
2205 }
2206
2207 // Renvoie TRUE si et seulement si la configuration autorise
2208 // le code HTML5 sur le site public
2209 function html5_permis() {
2210 return (isset($GLOBALS['meta']['version_html_max'])
2211 AND ('html5' == $GLOBALS['meta']['version_html_max']));
2212 }
2213
2214 /*
2215 * Bloc de compatibilite : quasiment tous les plugins utilisent ces fonctions
2216 * desormais depreciees ; plutot que d'obliger tout le monde a charger
2217 * vieilles_defs, on va assumer l'histoire de ces 3 fonctions ubiquitaires
2218 */
2219 // Fonction depreciee
2220 // http://doc.spip.org/@lire_meta
2221 function lire_meta($nom) {
2222 return $GLOBALS['meta'][$nom];
2223 }
2224
2225 // Fonction depreciee
2226 // http://doc.spip.org/@ecrire_metas
2227 function ecrire_metas() {}
2228
2229 // Fonction depreciee, cf. http://doc.spip.org/@sql_fetch
2230 // http://doc.spip.org/@spip_fetch_array
2231 function spip_fetch_array($r, $t=NULL) {
2232 if (!isset($t)) {
2233 if ($r) return sql_fetch($r);
2234 } else {
2235 if ($t=='SPIP_NUM') $t = MYSQL_NUM;
2236 if ($t=='SPIP_BOTH') $t = MYSQL_BOTH;
2237 if ($t=='SPIP_ASSOC') $t = MYSQL_ASSOC;
2238 spip_log("appel deprecie de spip_fetch_array(..., $t)", 'vieilles_defs');
2239 if ($r) return mysql_fetch_array($r, $t);
2240 }
2241 }
2242
2243 /**
2244 * Poser une alerte qui sera affiche aux auteurs de bon statut ('' = tous)
2245 * au prochain passage dans l'espace prive
2246 * chaque alerte doit avoir un nom pour eviter duplication a chaque hit
2247 * les alertes affichees une fois sont effacees
2248 *
2249 * @param string $nom
2250 * @param string $message
2251 * @param string $statut
2252 */
2253 function avertir_auteurs($nom,$message, $statut=''){
2254 $alertes = $GLOBALS['meta']['message_alertes_auteurs'];
2255 if (!$alertes
2256 OR !is_array($alertes = unserialize($alertes)))
2257 $alertes = array();
2258
2259 if (!isset($alertes[$statut]))
2260 $alertes[$statut] = array();
2261 $alertes[$statut][$nom] = $message;
2262 ecrire_meta("message_alertes_auteurs",serialize($alertes));
2263 }
2264 ?>