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