<?php /***************************************************************************\ * SPIP, Systeme de publication pour l'internet * * * * Copyright (c) 2001-2006 * * Arnaud Martin, Antoine Pitrou, Philippe Riviere, Emmanuel Saint-James * * * * Ce programme est un logiciel libre distribue sous licence GNU/GPL. * * Pour plus de details voir le fichier COPYING.txt ou l'aide en ligne. * \***************************************************************************/ if (!defined("_ECRIRE_INC_VERSION")) return; // // Gestion des inclusions et infos repertoires // # fonction obsolete, assurant la compatilibite ascendante. cf include_spip // http://doc.spip.org/@include_ecrire function include_ecrire($file, $silence=false) { preg_match('/^((inc_)?([^.]*))(\.php[3]?)?$/', $file, $r); // Version new style, surchargeable # cas speciaux if ($r[3] == 'index') return include_spip('inc/indexation'); if ($r[3] == 'db_mysql') return include_spip('base/db_mysql'); if ($r[3] == 'connect') { spip_connect(); return; } # cas general if ($f=include_spip('inc/'.$r[3])) return $f; // fichiers old-style, ecrire/inc_truc.php if (is_readable($f = _DIR_RESTREINT . $r[1] . '.php')) return include_once($f); } // charge un fichier perso ou, a defaut, standard // et retourne si elle existe le nom de la fonction homonyme (exec_$nom), // ou de suffixe _dist // http://doc.spip.org/@charger_fonction function charger_fonction($nom, $dossier='exec', $continue=false) { // Securite de base if (!preg_match(',^\w+$,', $nom)) redirige_par_entete('./'); if (substr($dossier,-1) == '/') $dossier = substr($dossier,0,-1); // Si la fonction existe deja (definie par mes_options, par exemple) if (function_exists($f = $dossier.'_'.$nom)) return $f; if (function_exists($g = $f . '_dist')) return $g; // Sinon charger le fichier de declaration // passer en minuscules (cf les balises de formulaires) $inc = include_spip($dossier.'/'. strtolower($nom)); if (function_exists($f)) return $f; if (function_exists($g)) return $g; if ($continue) return false; // Echec : message d'erreur spip_log("fonction $nom indisponible" . ($inc ? "" : "(fichier $dossier/$nom absent)")); include_spip('inc/minipres'); minipres(_T('forum_titre_erreur'), _T('fichier_introuvable', array('fichier'=> '<b>'.htmlentities("$dossier/$nom").'</b>'))); } // // une fonction cherchant un fichier dans une liste de repertoires // // http://doc.spip.org/@include_spip function include_spip($f, $include = true) { // Dans le noyau ? if (isset($GLOBALS['noyau'][$f])) { $s = $GLOBALS['noyau'][$f]; } // Sinon le chercher et mettre a jour le noyau else { if (!$s = find_in_path($f . '.php') AND (!_EXTENSION_PHP OR !$s = find_in_path($f . '.php3'))) { return $GLOBALS['noyau'][$f] = false; } else $GLOBALS['noyau'][$f] = $s; } // On charge le fichier (sauf si on ne voulait que son chemin) if ($include && $s) { include_once $s; } return $s; } // un pipeline est lie a une action et une valeur // chaque element du pipeline est autorise a modifier la valeur // // le pipeline execute les elements disponibles pour cette action, // les uns apres les autres, et retourne la valeur finale // // Cf. compose_filtres dans references.php, qui est la // version compilee de cette fonctionnalite // appel unitaire d'une fonction du pipeline // utilisee dans le script pipeline precompile // http://doc.spip.org/@minipipe function minipipe($fonc,$val){ // fonction if (function_exists($fonc)) $val = call_user_func($fonc, $val); // Class::Methode else if (preg_match("/^(\w*)::(\w*)$/", $fonc, $regs) AND $methode = array($regs[1], $regs[2]) AND is_callable($methode)) $val = call_user_func($methode, $val); else spip_log("Erreur - '$fonc' non definie !"); return $val; } // chargement du pipeline sous la forme d'un fichier php prepare // http://doc.spip.org/@pipeline function pipeline($action,$val) { static $charger; // chargement initial des fonctions mises en cache, ou generation du cache if (!$charger) { if (!($ok = @is_readable($charger = _DIR_TMP."charger_pipelines.php"))) { include_spip('inc/plugin'); // generer les fichiers php precompiles // de chargement des plugins et des pipelines verif_plugin(); if (!($ok = @is_readable($charger))) spip_log("fichier $charger pas cree"); } if ($ok) include_once $charger; } // appliquer notre fonction si elle existe $fonc = 'execute_pipeline_'.$action; if (function_exists($fonc)) { $val = $fonc($val); } // plantage ? else { include_spip('inc/plugin'); // on passe $action en arg pour creer la fonction meme si le pipe // n'est defini nul part ; vu qu'on est la c'est qu'il existe ! verif_plugin($action); spip_log("fonction $f absente : pipeline desactive"); } // si le flux est une table qui encapsule donnees et autres // on ne ressort du pipe que les donnees if (is_array($val) && isset($val['data'])) $val = $val['data']; return $val; } // // Enregistrement des evenements // // http://doc.spip.org/@spip_log function spip_log($message, $logname='spip') { static $compteur; if ($compteur++ > 100) return; $pid = '(pid '.@getmypid().')'; $message = date("M d H:i:s").' '.$GLOBALS['ip'].' '.$pid.' ' .preg_replace("/\n*$/", "\n", $message); $logfile = _DIR_TMP . $logname . '.log'; if (@is_readable($logfile) AND (!$s = @filesize($logfile) OR $s > 10*1024)) { $rotate = true; $message .= "[-- rotate --]\n"; } else $rotate = ''; $f = @fopen($logfile, "ab"); if ($f) { fputs($f, htmlspecialchars($message)); fclose($f); } if ($rotate) { @unlink($logfile.'.3'); @rename($logfile.'.2',$logfile.'.3'); @rename($logfile.'.1',$logfile.'.2'); @rename($logfile,$logfile.'.1'); } // recopier les spip_log mysql (ce sont uniquement des erreurs) // dans le spip_log general if ($logname == 'mysql') spip_log($message); } // API d'appel a la base de donnees: // on charge le fichier du repertoire base/ donne en argument // et on execute la fonction homonyme censee initaliser la connexion // et renvoyer le nom de la fonction a connexion persistante. // On memorise ce nom dans une statique pour n'appeler qu'une fois. // http://doc.spip.org/@spip_connect function spip_connect($serveur='') { static $t = array(); // Assimiler spip_connect() et spip_connect('') [PHP les distingue]. // Tous deux designent le serveur SQL std "db_mysql" (obscur mais historique) if (!$serveur) $serveur = 'db_mysql'; if (isset($t[$serveur])) return $t[$serveur]; $f = charger_fonction($serveur, 'base', true); return ($t[$serveur] = ($f ? $f() : false)); } // http://doc.spip.org/@spip_query function spip_query($query, $serveur='') { if (!($f = spip_connect($serveur))) return; // Erreur de connexion // executer la requete return $f($query); } // a demenager dans base/abstract_sql a terme // http://doc.spip.org/@spip_abstract_quote function spip_abstract_quote($arg_sql) { return (is_int($arg_sql)) ? $arg_sql : ("'" . addslashes($arg_sql) . "'"); } // Renvoie le _GET ou le _POST emis par l'utilisateur // // S'il y a un second argument, on SET la valeur dans $_GET -- qui est // la premiere a etre retournee par _request() // http://doc.spip.org/@_request function _request($var) { // set ? if (func_num_args()>1) { unset($_GET[$var]); // au cas ou l'on fait _request('truc', NULL); unset($_POST[$var]); if (func_get_arg(1) !== NULL) $_GET[$var] = func_get_arg(1); } if (isset($_GET[$var])) $a = $_GET[$var]; elseif (isset($_POST[$var])) $a = $_POST[$var]; else return NULL; // temporaire: si on est en ajax et en POST tout a ete encode // via encodeURIComponent, il faut donc repasser // dans le charset local.... on le connait grace // a la variable var_ajaxcharset ajoutee dans layer.js if (isset($_POST['var_ajaxcharset']) AND isset($GLOBALS['meta']['charset']) AND $GLOBALS['meta']['charset'] != $_POST['var_ajaxcharset'] AND is_string($a) AND preg_match(',[\x80-\xFF],', $a)) { include_spip('inc/charsets'); return importer_charset($a, $_POST['var_ajaxcharset']); } return $a; } // // Prend une URL et lui ajoute/retire un parametre. // Exemples : [(#SELF|parametre_url{suite,18})] (ajout) // [(#SELF|parametre_url{suite,''})] (supprime) // [(#SELF|parametre_url{suite})] (prend $suite dans la _request) // http://www.spip.net/@parametre_url // // http://doc.spip.org/@parametre_url function parametre_url($url, $c, $v=NULL, $sep='&') { // lever l'#ancre if (preg_match(',^([^#]*)(#.*)$,', $url, $r)) { $url = $r[1]; $ancre = $r[2]; } else $ancre = ''; // eclater $url = preg_split(',[?]|&|&,', $url); // recuperer la base $a = array_shift($url); if (!$a) $a= './'; // ajout de la globale ? if ($v === NULL) $v = _request($c); // lire les variables et agir foreach ($url as $n => $val) { if (preg_match(',^'.$c.'(=.*)?$,', $val)) { // suppression if (!$v) { unset($url[$n]); } else { $url[$n] = $c.'='.rawurlencode($v); $v = ''; } } } // ajouter notre parametre si on ne l'a pas encore trouve if ($v) $url[] = $c.'='.rawurlencode($v); // eliminer les vides $url = array_filter($url); // recomposer l'adresse if ($url) $a .= '?' . join($sep, $url); return $a . $ancre; } // // Prend une URL et lui ajoute/retire une ancre. // http://doc.spip.org/@ancre_url function ancre_url($url, $ancre) { // lever l'#ancre if (preg_match(',^([^#]*)(#.*)$,', $url, $r)) { $url = $r[1]; } return $url .'#'. $ancre; } // // pour calcul du nom du fichier cache et autres // // http://doc.spip.org/@nettoyer_uri function nettoyer_uri() { $uri1 = $GLOBALS['REQUEST_URI']; do { $uri = $uri1; $uri1 = preg_replace (',([?&])(PHPSESSID|(var_[^=&]*))=[^&]*(&|$),i', '\1', $uri); } while ($uri<>$uri1); return preg_replace(',[?&]$,', '', $uri1); } // // donner l'URL de base d'un lien vers "soi-meme", modulo // les trucs inutiles // // http://doc.spip.org/@self function self($root = false) { $url = nettoyer_uri(); if (!$root) $url = preg_replace(',^[^?]*/,', '', $url); // ajouter le cas echeant les variables _POST['id_...'] foreach ($_POST as $v => $c) if (substr($v,0,3) == 'id_') $url = parametre_url($url, $v, $c, '&'); // supprimer les variables sans interet if (!_DIR_RESTREINT) { $url = preg_replace (',([?&])(' .'lang|set_options|set_couleur|set_disp|set_ecran|show_docs|' .'changer_lang|var_lang|action)=[^&]*,i', '\1', $url); $url = preg_replace(',([?&])[&]+,', '\1', $url); $url = preg_replace(',[&]$,', '\1', $url); } // eviter les hacks $url = htmlspecialchars($url); // Si c'est vide, donner './' $url = preg_replace(',^$,', './', $url); return $url; } // Fonction abandonnee. . Ne plus utiliser, remplacer par son contenu // http://doc.spip.org/@lire_meta function lire_meta($nom) { global $meta; return $meta[$nom];} // // Traduction des textes de SPIP // // http://doc.spip.org/@_T function _T($texte, $args=array()) { # petite optimisation pour ne passer qu'une fois dans include_spip static $c; $c OR $c = include_spip('inc/lang'); $text = traduire_chaine($texte); if (!empty($GLOBALS['xhtml'])) { include_spip('inc/charsets'); $text = html2unicode($text, true /* secure */); } if (!$text) // pour les chaines non traduites $text = str_replace('_', ' ', (($n = strpos($texte,':')) === false ? $texte : substr($texte, $n+1))); while (list($name, $value) = @each($args)) $text = str_replace ("@$name@", $value, $text); return $text; } // chaines en cours de traduction // http://doc.spip.org/@_L function _L($text, $args=array()) { while (list($name, $value) = @each($args)) $text = str_replace ("@$name@", $value, $text); if ($GLOBALS['test_i18n']) return "<span style='color:red;'>$text</span>"; else return $text; } // Afficher "ecrire/data/" au lieu de "data/" dans les messages // http://doc.spip.org/@joli_repertoire function joli_repertoire($rep) { $a = substr($rep,0,1); if ($a<>'.' AND $a<>'/') $rep = (_DIR_RESTREINT?'':_DIR_RESTREINT_ABS).$rep; return $rep; } // Nommage bizarre des tables d'objets // http://doc.spip.org/@table_objet function table_objet($type) { static $surnoms = array( 'article' => 'articles', 'auteur' => 'auteurs', 'breve' => 'breves', 'document' => 'documents', 'doc' => 'documents', # pour les modeles 'img' => 'documents', 'emb' => 'documents', 'forum' => 'forum', # hum 'groupe_mots' => 'groupes_mots', # hum 'message' => 'messages', 'mot' => 'mots', 'petition' => 'petitions', 'rubrique' => 'rubriques', 'signature' => 'signatures', 'syndic' => 'syndic', 'site' => 'syndic', # hum hum 'syndic_article' => 'syndic_articles', 'type_document' => 'types_documents' # hum ); return $surnoms[$type]; } // http://doc.spip.org/@id_table_objet function id_table_objet($type) { if ($type == 'site' OR $type == 'syndic') return 'id_syndic'; else if ($type == 'forum') return 'id_forum'; else if ($type=='doc' OR $type=='img' OR $type=='emb') # pour les modeles return 'id_document'; else return 'id_'.$type; } // // spip_timer : on l'appelle deux fois et on a la difference, affichable // // http://doc.spip.org/@spip_timer function spip_timer($t='rien') { static $time; $a=time(); $b=microtime(); if (isset($time[$t])) { $p = $a + $b - $time[$t]; unset($time[$t]); return sprintf("%.2fs", $p); } else $time[$t] = $a + $b; } // spip_touch : verifie si un fichier existe et n'est pas vieux (duree en s) // et le cas echeant le touch() ; renvoie true si la condition est verifiee // et fait touch() sauf si ca n'est pas souhaite // (regle aussi le probleme des droits sur les fichiers touch()) // http://doc.spip.org/@spip_touch function spip_touch($fichier, $duree=0, $touch=true) { if (!($exists = @is_readable($fichier)) || ($duree == 0) || (@filemtime($fichier) < time() - $duree)) { if ($touch) { if (!@touch($fichier)) { @unlink($fichier); @touch($fichier); }; if (!$exists) @chmod($fichier, 0666); } return true; } return false; } // Pour executer des taches de fond discretement, on utilise background-image // car contrairement a un iframe vide, les navigateurs ne diront pas qu'ils // n'ont pas fini de charger, c'est plus rassurant. // C'est aussi plus discret qu'un <img> sous un navigateur non graphique. // Cette fonction est utilisee pour l'espace prive (cf inc_presentation) // et pour l'espace public (cf #SPIP_CRON dans inc_balise) // http://doc.spip.org/@generer_spip_cron function generer_spip_cron() { return '<div style="background-image: url(\'' . generer_url_action('cron') . '\');"></div>'; } // envoi de l'image demandee dans le code ci-dessus // http://doc.spip.org/@envoie_image_vide function envoie_image_vide() { $image = pack("H*", "47494638396118001800800000ffffff00000021f90401000000002c0000000018001800000216848fa9cbed0fa39cb4da8bb3debcfb0f86e248965301003b"); header("Content-Type: image/gif"); header("Content-Length: ".strlen($image)); header("Cache-Control: no-cache,no-store"); header("Pragma: no-cache"); header("Connection: close"); echo $image; flush(); } // http://doc.spip.org/@action_cron function action_cron() { envoie_image_vide(); cron (1); } // // cron() : execution des taches de fond // quand il est appele par public.php il n'est pas gourmand; // quand il est appele par ?action=cron, il est gourmand // http://doc.spip.org/@cron function cron ($gourmand=false) { // Si on est gourmand, ou si le fichier gourmand n'existe pas // (ou est trop vieux -> 60 sec), on va voir si un cron est necessaire. // Au passage si on est gourmand on le dit aux autres if (spip_touch(_FILE_CRON_LOCK.'-gourmand', 60, $gourmand) OR $gourmand) { // Faut-il travailler ? Pas tous en meme temps svp // Au passage si on travaille on bloque les autres if (spip_touch(_FILE_CRON_LOCK, 2)) { include_spip('inc/cron'); spip_cron(); } } } // envoyer le navigateur sur une nouvelle adresse // en evitant les attaques par la redirection (souvent indique par 1 $_GET) // http://doc.spip.org/@redirige_par_entete function redirige_par_entete($url) { # en theorie on devrait faire ca tout le temps, mais quand la chaine # commence par ? c'est imperatif, sinon l'url finale n'est pas la bonne if ($url[0]=='?') $url = url_de_base().$url; @header("Location: " . strtr($url, "\n\r", " ")); echo '<!DOCTYPE HTML PUBLIC "-//IETF//DTD HTML 2.0//EN"> <html><head> <title>302 Found</title> </head> <body> <h1>302 Found</h1> <a href="' .quote_amp($url) .'">Click here</a>. </body></html>'; exit; } // transformation XML des "&" en "&" // http://doc.spip.org/@quote_amp function quote_amp($u) { return preg_replace( "/&(?![a-z]{0,4}\w{2,3};|#x?[0-9a-f]{2,5};)/i", "&",$u); } // Transforme n'importe quel champ en une chaine utilisable // en PHP ou Javascript en toute securite // < ? php $x = '[(#TEXTE|texte_script)]'; ? > // http://doc.spip.org/@texte_script function texte_script($texte) { return str_replace('\'', '\\\'', str_replace('\\', '\\\\', $texte)); } // // find_in_path() : chercher un fichier nomme x selon le chemin rep1:rep2:rep3 // // http://doc.spip.org/@creer_chemin function creer_chemin() { static $path_a = array(); static $c = ''; // on calcule le chemin si le nombre de plugins a change if ($c != count($GLOBALS['plugins']).$GLOBALS['dossier_squelettes']) { $c = count($GLOBALS['plugins']).$GLOBALS['dossier_squelettes']; // Chemin standard depuis l'espace public $path = defined('_SPIP_PATH') ? _SPIP_PATH : _DIR_RACINE.':'. _DIR_RACINE.'dist/:'. _DIR_RESTREINT; // Ajouter les repertoires des plugins if ($GLOBALS['plugins']) $path = _DIR_PLUGINS . join(':'._DIR_PLUGINS, $GLOBALS['plugins']) . ':' . $path; // Ajouter squelettes/ if (@is_dir(_DIR_RACINE.'squelettes')) $path = _DIR_RACINE.'squelettes/:' . $path; // Et le(s) dossier(s) des squelettes nommes if ($GLOBALS['dossier_squelettes']) foreach (array_reverse(explode(':', $GLOBALS['dossier_squelettes'])) as $d) $path = ($d[0] == '/' ? '' : _DIR_RACINE) . $d . '/:' . $path; // nettoyer les / du path $path_a = array(); foreach (explode(':', $path) as $dir) { if (strlen($dir) AND substr($dir,-1) != '/') $dir .= "/"; $path_a[] = $dir; } } return $path_a; } // http://doc.spip.org/@find_in_path function find_in_path ($filename) { // Parcourir le chemin foreach (creer_chemin() as $dir) { if (@is_readable($f = "$dir$filename")) { # spip_log("find_in_path trouve $f"); return $f; } } # spip_log("find_in_path n'a pas vu '$filename' dans " . join(':',creer_chemin())); return false; } // predicat sur les scripts de ecrire qui n'authentifient pas par cookie // http://doc.spip.org/@autoriser_sans_cookie function autoriser_sans_cookie($nom) { static $autsanscookie = array('aide_index', 'install', 'admin_repair'); $nom = preg_replace('/.php[3]?$/', '', basename($nom)); return in_array($nom, $autsanscookie); } // Cette fonction charge le bon inc-urls selon qu'on est dans l'espace // public ou prive, la presence d'un (old style) inc-urls.php3, etc. // http://doc.spip.org/@charger_generer_url function charger_generer_url() { static $ok; // espace prive if (!_DIR_RESTREINT) include_spip('inc/urls'); // espace public else { if ($ok++) return; # fichier deja charge // fichier inc-urls ? (old style) if (@is_readable($f = _DIR_RACINE.'inc-urls.php3') OR @is_readable($f = _DIR_RACINE.'inc-urls.php') OR $f = find_in_path('inc-urls-'.$GLOBALS['type_urls'].'.php3') OR $f = include_spip('urls/'.$GLOBALS['type_urls'], false) ) include_once($f); } } // Sur certains serveurs, la valeur 'Off' tient lieu de false dans certaines // variables d'environnement comme $_SERVER[HTTPS] ou ini_get(register_globals) // http://doc.spip.org/@test_valeur_serveur function test_valeur_serveur($truc) { if (!$truc) return false; if (strtolower($truc) == 'off') return false; return true; } // // Fonctions de fabrication des URL des scripts de Spip // // l'URL de base du site, sans se fier a meta(adresse_site) qui // peut etre fausse (sites a plusieurs noms d'hotes, deplacements, erreurs) // Note : la globale $profondeur_url doit etre initialisee de maniere a // indiquer le nombre de sous-repertoires de l'url courante par rapport a la // racine de SPIP : par exemple, sur ecrire/ elle vaut 1, sur sedna/ 1, et a // la racine 0. Sur url/perso/ elle vaut 2 // http://doc.spip.org/@url_de_base function url_de_base() { global $REQUEST_URI; static $url; if ($url) return $url; // cas particulier des sites filtres par un proxy entrant // cf. http://trac.rezo.net/trac/spip/ticket/401 if (isset($_SERVER['HTTP_X_FORWARDED_HOST'])) { $a = $GLOBALS['meta']['adresse_site']; return (substr($a,-1) == '/')?$a:$a.'/'; } $http = ( (isset($_SERVER["SCRIPT_URI"]) AND substr($_SERVER["SCRIPT_URI"],0,5) == 'https') OR (isset($_SERVER['HTTPS']) AND test_valeur_serveur($_SERVER['HTTPS'])) ) ? 'https' : 'http'; # note : HTTP_HOST contient le :port si necessaire $myself = $http.'://'.$_SERVER['HTTP_HOST'].$REQUEST_URI; # supprimer la chaine de GET $myself = preg_replace(',\?.*$,','', $myself); # supprimer n sous-repertoires $supprime_preg = '/+'; for ($i=0; $i<$GLOBALS['profondeur_url']; $i++) $supprime_preg .= '[^/]+/+'; $url = preg_replace(','.$supprime_preg.'[^/]*$,', '/', $myself); return $url; } // Pour une redirection, la liste des arguments doit etre separee par "&" // Pour du code XHTML, ca doit etre & // Bravo au W3C qui n'a pas ete capable de nous eviter ca // faute de separer proprement langage et meta-langage // Attention, X?y=z et "X/?y=z" sont completement differents! // http://httpd.apache.org/docs/2.0/mod/mod_dir.html // http://doc.spip.org/@generer_url_ecrire function generer_url_ecrire($script, $args="", $no_entities=false, $rel=false) { if (!$rel) $rel = url_de_base() . _DIR_RESTREINT_ABS; else if (!is_string($rel)) $rel = _DIR_RESTREINT ? _DIR_RESTREINT : './'; // Les anciens IIS n'acceptent pas les POST sur ecrire/ (#419) // meme pb sur thttpd cf. http://forum.spip.org/fr_184153.html if (preg_match(',IIS|thttpd,',$_SERVER['SERVER_SOFTWARE'])) $rel .= 'index.php'; if ($script AND $script<>'accueil') $args = "?exec=$script" . (!$args ? '' : "&$args"); elseif ($args) $args ="?$args"; return $rel . ($no_entities ? $args : str_replace('&', '&', $args)); } // http://doc.spip.org/@generer_url_retour function generer_url_retour($script, $args="") { return rawurlencode(generer_url_ecrire($script, $args, true, true)); } // // Adresse des scripts publics (a passer dans inc-urls...) // // Detecter le fichier de base, a la racine, comme etant spip.php ou '' // dans le cas de '', un $default = './' peut servir (comme dans urls/page.php) // http://doc.spip.org/@get_spip_script function get_spip_script($default='') { # cas define('_SPIP_SCRIPT', ''); if (_SPIP_SCRIPT) return _SPIP_SCRIPT; else return $default; } // http://doc.spip.org/@generer_url_public function generer_url_public($script, $args="", $no_entities=false) { // si le script est une action (spip_pass, spip_inscription), // standardiser vers la nouvelle API // [hack temporaire pour faire fonctionner #URL_PAGE{spip_pass} ] if (preg_match(',^spip_(.*),', $script, $regs)) { $args = "action=" . $regs[1] .($args ? "&$args" :''); $script = ""; } $action = get_spip_script(); if ($script) $action = parametre_url($action, 'page', $script, '&'); if ($args) $action .= (strpos($action, '?') !== false ? '&' : '?') . $args; if (!$no_entities) $action = quote_amp($action); return url_de_base() . $action; } // http://doc.spip.org/@generer_url_prive function generer_url_prive($script, $args="", $no_entities=false) { $action = 'prive.php'; if ($script) $action = parametre_url($action, 'page', $script, '&'); if ($args) $action .= (strpos($action, '?') !== false ? '&' : '?') . $args; if (!$no_entities) $action = quote_amp($action); return url_de_base() . _DIR_RESTREINT_ABS . $action; } // http://doc.spip.org/@generer_url_action function generer_url_action($script, $args="", $no_entities=false) { return generer_url_public('', "action=$script" .($args ? "&$args" : ''), $no_entities); } // Dirty hack contre le register_globals a 'Off' (PHP 4.1.x) // A remplacer (bientot ?) par une gestion propre des variables admissibles ;-) // Attention pour compatibilite max $_GET n'est pas superglobale // NB: c'est une fonction de maniere a ne pas pourrir $GLOBALS // http://doc.spip.org/@spip_register_globals function spip_register_globals() { // Liste des variables dont on refuse qu'elles puissent provenir du client $refuse_gpc = array ( # inc-public 'fond', 'delais', # ecrire/inc_auth 'REMOTE_USER', 'PHP_AUTH_USER', 'PHP_AUTH_PW' ); // Liste des variables (contexte) dont on refuse qu'elles soient cookie // (histoire que personne ne vienne fausser le cache) $refuse_c = array ( # inc-calcul 'id_parent', 'id_rubrique', 'id_article', 'id_auteur', 'id_breve', 'id_forum', 'id_secteur', 'id_syndic', 'id_syndic_article', 'id_mot', 'id_groupe', 'id_document', 'date', 'lang' ); // Si les variables sont passees en global par le serveur, il faut // faire quelques verifications de base if (test_valeur_serveur(@ini_get('register_globals'))) { foreach ($refuse_gpc as $var) { if (isset($GLOBALS[$var])) { if ( // demande par le client $_REQUEST[$var] !== NULL // et pas modifie par les fichiers d'appel AND $GLOBALS[$var] == $_REQUEST[$var] ) // Alors on ne sait pas si c'est un hack die ("register_globals: $var interdite"); } } foreach ($refuse_c as $var) { if (isset($GLOBALS[$var])) { if ( isset ($_COOKIE[$var]) AND $_COOKIE[$var] == $GLOBALS[$var] ) define ('spip_interdire_cache', true); } } } // sinon il faut les passer nous-memes, a l'exception des interdites. // (A changer en une liste des variables admissibles...) else { foreach (array('_SERVER', '_COOKIE', '_POST', '_GET') as $_table) { foreach ($GLOBALS[$_table] as $var => $val) { if (!isset($GLOBALS[$var]) # indispensable securite AND isset($GLOBALS[$_table][$var]) AND ($_table == '_SERVER' OR !in_array($var, $refuse_gpc)) AND ($_table <> '_COOKIE' OR !in_array($var, $refuse_c))) $GLOBALS[$var] = $val; } } } } // http://doc.spip.org/@spip_initialisation function spip_initialisation() { // la taille maxi des logos (0 : pas de limite) define('_LOGO_MAX_SIZE', 0); # poids en ko define('_LOGO_MAX_WIDTH', 0); # largeur en pixels define('_LOGO_MAX_HEIGHT', 0); # hauteur en pixels define('_DOC_MAX_SIZE', 0); # poids en ko define('_IMG_MAX_SIZE', 0); # poids en ko define('_IMG_MAX_WIDTH', 0); # largeur en pixels define('_IMG_MAX_HEIGHT', 0); # hauteur en pixels define('_IMG_GD_MAX_PIXELS', 0); # nombre de pixels maxi pour calcul de la vignette avec gd // Icones # le chemin http (relatif) vers les images standard define('_DIR_IMG_PACK', (_DIR_RESTREINT . 'img_pack/')); # vers les logos de type de document, standard & perso define('_DIR_IMG_ICONES_DIST', _DIR_IMG_PACK . "icones/"); # les icones de la barre d'edition des formulaires define('_DIR_IMG_ICONES_BARRE', _DIR_IMG_PACK . "icones_barre/"); # le chemin php (absolu) vers les images standard (pour hebergement centralise) define('_ROOT_IMG_PACK', (dirname(dirname(__FILE__)) . '/img_pack/')); define('_ROOT_IMG_ICONES_DIST', (dirname(dirname(__FILE__)) . '/img_pack/icones/')); // Le charset par defaut lors de l'installation define('_DEFAULT_CHARSET', 'utf-8'); // qq chaines standard define('_ACCESS_FILE_NAME', '.htaccess'); define('_AUTH_USER_FILE', '.htpasswd'); define('_SPIP_DUMP', 'dump.xml'); define('_DOCTYPE_ECRIRE', "<!DOCTYPE HTML PUBLIC '-//W3C//DTD HTML 4.01 Transitional//EN' 'http://www.w3.org/TR/html4/loose.dtd'>\n"); // L'adresse de base du site ; on peut mettre '' si la racine est geree par // le script index.php define('_SPIP_SCRIPT', 'spip.php'); // *********** traiter les variables ************ // Recuperer les superglobales $_GET si non definies // (en theorie c'est impossible depuis PHP 4.0.3, cf. track_vars) // et les identifier aux $HTTP_XX_VARS foreach (array('_GET', '_POST', '_COOKIE', '_SERVER') as $_table) { $http_table_vars = 'HTTP'.$_table.'_VARS'; if (!is_array($GLOBALS[$_table])) { $GLOBALS[$_table] = array(); if (is_array($GLOBALS[$http_table_vars])) $GLOBALS[$_table] = & $GLOBALS[$http_table_vars]; } $GLOBALS[$http_table_vars] = & $GLOBALS[$_table]; } // // Securite // // Ne pas se faire manger par un bug php qui accepte ?GLOBALS[truc]=toto if (isset($_REQUEST['GLOBALS'])) die(); // nettoyer les magic quotes \' et les caracteres nuls %00 spip_desinfecte($_GET); spip_desinfecte($_POST); spip_desinfecte($_COOKIE); spip_desinfecte($_REQUEST); spip_desinfecte($GLOBALS); // Par ailleurs on ne veut pas de magic_quotes au cours de l'execution @set_magic_quotes_runtime(0); // Remplir $GLOBALS avec $_GET et $_POST (methode a revoir pour fonctionner // completement en respectant register_globals = off) spip_register_globals(); // appliquer le cookie_prefix if ($GLOBALS['cookie_prefix'] != 'spip') { include_spip('inc/cookie'); recuperer_cookies_spip($GLOBALS['cookie_prefix']); } // // Capacites php (en fonction de la version) // $GLOBALS['flag_gz'] = function_exists("gzencode"); #php 4.0.4 $GLOBALS['flag_ob'] = (function_exists("ob_start") && function_exists("ini_get") && (@ini_get('max_execution_time') > 0) && !strstr(ini_get('disable_functions'), 'ob_')); $GLOBALS['flag_sapi_name'] = function_exists("php_sapi_name"); $GLOBALS['flag_revisions'] = function_exists("gzcompress"); $GLOBALS['flag_get_cfg_var'] = (@get_cfg_var('error_reporting') != ""); $GLOBALS['flag_upload'] = (!$GLOBALS['flag_get_cfg_var'] || (get_cfg_var('upload_max_filesize') > 0)); // // Sommes-nous dans l'empire du Mal ? // (ou sous le signe du Pingouin, ascendant GNU ?) // if (strpos($_SERVER['SERVER_SOFTWARE'], '(Win') !== false) define ('os_serveur', 'windows'); else define ('os_serveur', ''); // // Infos sur le fichier courant // // Compatibilite avec serveurs ne fournissant pas $REQUEST_URI if (!isset($GLOBALS['REQUEST_URI'])) { if (isset($_SERVER['REQUEST_URI'])) { $GLOBALS['REQUEST_URI'] = $_SERVER['REQUEST_URI']; } else { $GLOBALS['REQUEST_URI'] = $_SERVER['PHP_SELF']; if ($_SERVER['QUERY_STRING'] AND !strpos($_SERVER['REQUEST_URI'], '?')) $GLOBALS['REQUEST_URI'] .= '?'.$_SERVER['QUERY_STRING']; } } // // Module de lecture/ecriture/suppression de fichiers utilisant flock() // (non surchargeable en l'etat ; attention si on utilise include_spip() // pour le rendre surchargeable, on va provoquer un reecriture // systematique du noyau ou une baisse de perfs => a etudier) include_once _DIR_RESTREINT . 'inc/flock.php'; // Lire les meta cachees et initier le noyau (espace public uniquement) $GLOBALS['noyau'] = array(); if (lire_fichier(_FILE_META, $meta)) { $GLOBALS['meta'] = @unserialize($meta); if (_DIR_RESTREINT AND isset($GLOBALS['meta']['noyau']) AND is_array($GLOBALS['meta']['noyau'])) { $GLOBALS['noyau'] = $GLOBALS['meta']['noyau']; unset ($GLOBALS['meta']['noyau']); } } // en cas d'echec refaire le fichier if (!is_array($GLOBALS['meta']) AND _FILE_CONNECT) { include_spip('inc/meta'); ecrire_metas(); } // supprimer le noyau si on recalcule if (isset($_REQUEST['var_mode'])) $GLOBALS['noyau'] = array(); // Langue principale du site $GLOBALS['langue_site'] = $GLOBALS['meta']['langue_site']; if (!$GLOBALS['langue_site']) include_spip('inc/lang'); $GLOBALS['spip_lang'] = $GLOBALS['langue_site']; // Verifier le visiteur if (_FILE_CONNECT) verifier_visiteur(); } // Annuler les magic quotes \' sur GET POST COOKIE et GLOBALS ; // supprimer aussi les eventuels caracteres nuls %00, qui peuvent tromper // la commande is_readable('chemin/vers/fichier/interdit%00truc_normal') // http://doc.spip.org/@spip_desinfecte function spip_desinfecte(&$t) { static $magic_quotes; if (!isset($magic_quotes)) $magic_quotes = @get_magic_quotes_gpc(); foreach ($t as $key => $val) { if (is_string($t[$key])) { if ($magic_quotes) $t[$key] = stripslashes($t[$key]); $t[$key] = str_replace(chr(0), '-', $t[$key]); } // traiter aussi les "texte_plus" de articles_edit else if ($key == 'texte_plus' AND is_array($t[$key])) spip_desinfecte($t[$key]); } } // retourne le statut du visiteur s'il s'annonce // http://doc.spip.org/@verifier_visiteur function verifier_visiteur() { // Rq: pour que cette fonction marche depuis mes_options elle a besoin // que les constantes principales soient initialisees spip_initialisation_parametree(_DIR_RACINE, _DIR_RESTREINT) ; if (isset($_COOKIE['spip_session']) OR (isset($_SERVER['PHP_AUTH_USER']) AND !$GLOBALS['ignore_auth_http'])) { $var_f = charger_fonction('session', 'inc'); if ($var_f()) return $GLOBALS['auteur_session']['statut']; include_spip('inc/actions'); return verifier_php_auth(); } return false; } // selectionner une langue // http://doc.spip.org/@lang_select function lang_select ($lang='') { if (!is_array($GLOBALS['pile_langues'])) $GLOBALS['pile_langues'] = array(); array_push($GLOBALS['pile_langues'], $GLOBALS['spip_lang']); if ($lang != $GLOBALS['spip_lang']) { include_spip('inc/lang'); changer_langue($lang); } } // revenir a la langue precedente // http://doc.spip.org/@lang_dselect function lang_dselect ($rien='') { $lang = array_pop($GLOBALS['pile_langues']); if ($lang != $GLOBALS['spip_lang']) { include_spip('inc/lang'); changer_langue($lang); } } ?>