diff --git a/.gitattributes b/.gitattributes index dc13bfe108f8de43f24103cbf2aaa3b2a4ec17dc..4be2fc5af21c6b7f136bb467d16d1b54b3742922 100644 --- a/.gitattributes +++ b/.gitattributes @@ -324,6 +324,7 @@ ecrire/inc_statistiques_svg.php -text ecrire/inc_statistiques_visites.php -text ecrire/inc_synchro.php -text ecrire/inc_upgrade.php -text +ecrire/inc_utils.php -text ecrire/lang/ecrire_br.php3 -text ecrire/lang/ecrire_ru.php3 -text ecrire/lang/public_br.php3 -text diff --git a/ecrire/inc_magicquotes.php b/ecrire/inc_magicquotes.php index bdbced5bc970de7d6b746efdff1f68861973281e..caf8c007254b9e77a7e2df1c7e70cc191222b27c 100644 --- a/ecrire/inc_magicquotes.php +++ b/ecrire/inc_magicquotes.php @@ -12,6 +12,101 @@ if (!defined("_ECRIRE_INC_VERSION")) return; +// Dirty hack contre le register_globals a 'Off' (PHP 4.1.x) +// A remplacer (un jour!) 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 +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', + + # ecrire/inc_texte + 'debut_intertitre', 'fin_intertitre', 'ligne_horizontale', + 'ouvre_ref', 'ferme_ref', 'ouvre_note', 'ferme_note', + 'les_notes', 'compt_note', 'nombre_surligne', + 'url_glossaire_externe', 'puce', 'puce_rtl' + ); + + // 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 (@ini_get('register_globals')) { + foreach ($refuse_gpc as $var) { + if (isset($GLOBALS[$var])) { + foreach (array('_GET', '_POST', '_COOKIE') as $_table) { + if ( + // demande par le client + isset ($GLOBALS[$_table][$var]) + // et pas modifie par les fichiers d'appel + AND $GLOBALS[$_table][$var] == $GLOBALS[$var] + ) // On ne sait pas si c'est un hack + { + # REMOTE_USER ou fond, c'est grave ; + # pour le reste (cookie 'lang', par exemple), simplement + # interdire la mise en cache de la page produite + switch ($var) { + case 'REMOTE_USER': + die ("$var interdite"); + break; + case 'fond': + if (!defined('_SPIP_PAGE')) + die ("$var interdite"); + break; + default: + define ('spip_interdire_cache', true); + } + } + } + } + } + foreach ($refuse_c as $var) { + if (isset($GLOBALS[$var])) { + foreach (array('_COOKIE') as $_table) { + if ( + // demande par le client + isset ($GLOBALS[$_table][$var]) + // et pas modifie par les fichiers d'appel + AND $GLOBALS[$_table][$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]) + 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; + } + } + } +} + + // Magic quotes : on n'en veut pas sur la base // et on nettoie les GET/POST/COOKIE le cas echeant function magic_unquote(&$t) { @@ -25,16 +120,4 @@ function magic_unquote(&$t) { $t = stripslashes($t); } -magic_unquote($_GET); -magic_unquote($_POST); -magic_unquote($_COOKIE); -# if (@ini_get('register_globals')) // pas fiable - magic_unquote($GLOBALS); - -# et a la fin supprimer la variable anti-recursion devenue inutile (et meme nuisible, notamment si on teste $_POST) -unset($_GET['spip_recursions']); -unset($_POST['spip_recursions']); -unset($_COOKIE['spip_recursions']); -unset($_GLOBALS['spip_recursions']); - ?> diff --git a/ecrire/inc_utils.php b/ecrire/inc_utils.php new file mode 100644 index 0000000000000000000000000000000000000000..37ffc96d4923cd3245dc578d7a8b3957516e70ce --- /dev/null +++ b/ecrire/inc_utils.php @@ -0,0 +1,541 @@ +<?php + +/***************************************************************************\ + * SPIP, Systeme de publication pour l'internet * + * * + * Copyright (c) 2001-2005 * + * 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 +// + +$included_files = array(); + +function include_local($file) { + if (@$GLOBALS['included_files'][$file]++) return; + + if (is_readable($file)) + include($file); + else spip_log($file . " illisible"); +} + +function include_ecrire($file) { +# Hack pour etre compatible avec les mes_options qui appellent cette fonction + define_once('_DIR_INCLUDE', _DIR_RESTREINT); + $file = _DIR_INCLUDE . $file; + if ($GLOBALS['included_files'][$file]++) return; + if (is_readable($file)) + include($file); + else spip_log($file . " illisible"); +} + +// charge un fichier perso ou, a defaut, standard +// et retourne si elle existe le nom de la fonction homonyme, ou de prefixe _dist + +function include_fonction($nom) { +# Hack pour etre compatible avec les mes_options qui appellent cette fonction + define_once('_DIR_INCLUDE', _DIR_RESTREINT); + spip_log($nom); + $nom = preg_replace("/\.php3?$/",'', basename($nom)); + $inc = ('inc_' . $nom . '.php'); + $f = find_in_path($inc); + if ($f && is_readable($f)) { + if (!$GLOBALS['included_files'][$f]++) include($f); + } else { + $f = _DIR_INCLUDE . $inc; + if (is_readable($f)) { + if (!$GLOBALS['included_files'][$f]++) include($f); + } else { + // provisoire avant renommage php/php3 + $f = _DIR_INCLUDE . ('inc_' . $nom . '.php3'); + if (is_readable($f)) { + if (!$GLOBALS['included_files'][$f]++) include($f); + } else $inc = ""; + } + } + $f = str_replace('-','_',$nom); // pour config-fonc etc. A renommer + if (function_exists($f)) + return $f; + elseif (function_exists($f .= "_dist")) + return $f; + else { + spip_log("fonction $nom indisponible" . + ($inc ? "" : "(aucun fichier inc_$f disponible)")); + exit; + } +} + + + +// 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 + +function pipeline($cause, $val) { + global $spip_pipeline, $spip_matrice; + if (!is_array($spip_pipeline[$cause])) return $val; + + foreach ($spip_pipeline[$cause] as $plug) { + + // charger un fichier le cas echeant + if (!function_exists($plug)) { + if ($f = $spip_matrice[$plug]) { + include($f); + $ok = function_exists($plug); + } + if (!$ok) { + spip_log("Erreur - $plug n'est pas definie ($f)"); + return $val; + } + } + + // appliquer le filtre + $val = $plug($val); + } + + return $val; +} + + +// +// Enregistrement des evenements +// +function spip_log($message, $logname='spip') { + static $compteur; + if ($compteur++ > 100) return; + + $pid = '(pid '.@getmypid().')'; + if (!$ip = $GLOBALS['REMOTE_ADDR']) $ip = '-'; + + $message = date("M d H:i:s")." $ip $pid " + .preg_replace("/\n*$/", "\n", $message); + + $logfile = _DIR_SESSIONS . $logname . '.log'; + if (@file_exists($logfile) && (@filesize($logfile) > 10*1024)) { + $rotate = true; + $message .= "[-- rotate --]\n"; + } + $f = @fopen($logfile, "ab"); + if ($f) { + fputs($f, $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 +function spip_query($query) { + + // Remarque : si on est appele par l'install, + // la connexion initiale a ete faite avant + if (!$GLOBALS['db_ok']) { + // Essaie de se connecter + if (_FILE_CONNECT) + include_local(_FILE_CONNECT); + } + + // Erreur de connexion + if (!$GLOBALS['db_ok']) + return; + + // Vieux format de fichier connexion + // Note: la version 0.1 est compatible avec la 0.2 (mais elle gere + // moins bien les erreurs timeout sur SQL), on ne force donc pas l'upgrade + if ($GLOBALS['spip_connect_version'] < 0.1) { + if (!_DIR_RESTREINT) {$GLOBALS['db_ok'] = false; return;} + redirige_par_entete("upgrade.php3?reinstall=oui"); + exit; + } + + // Faire la requete + return spip_query_db($query); +} + + +class Link { + var $file; + var $vars; + var $arrays; + + // + // Contructeur : a appeler soit avec l'URL du lien a creer, + // soit sans parametres, auquel cas l'URL est l'URL courante + // + // parametre $root = demander un lien a partir de la racine du serveur / + function Link($url = '', $root = false) { + global $_POST; + static $link = ''; + + $this->vars = array(); + $this->arrays = array(); + + // Normal case + if ($link) { + if ($url) { + $v = split('[\?\&]', $url); + list(, $this->file) = each($v); + while (list(, $var) = each($v)) { + list($name, $value) = split('=', $var, 2); + $name = urldecode($name); + $value = urldecode($value); + if (preg_match(',^(.*)\[\]$,', $name, $regs)) { + $this->arrays[$regs[1]][] = $value; + } + else { + $this->vars[$name] = $value; + } + } + } + else { + $this->file = $link->file; + $this->vars = $link->vars; + $this->arrays = $link->arrays; + } + return; + } + + // Si aucun URL n'est specifie, creer le lien "propre" + // ou l'on supprime de l'URL courant les bidules inutiles + if (!$url) { + // GET variables are read from the original URL + // (_GET may contain additional variables + // introduced by rewrite-rules) + $url = $GLOBALS['REQUEST_URI']; + // Warning !!!! + // since non encoded arguments may be present + // (especially those coming from Rewrite Rule) + // find the begining of the query string + // to compute the script-name + if ($v = strpos($url,'?')) + $v = strrpos(substr($url, 0, $v), '/'); + else $v = strrpos($url, '/'); + if (!$root) $url = substr($url, $v + 1); + if (!$url) $url = "./"; + if (count($_POST)) { + $vars = array(); + foreach ($_POST as $var => $val) + if (preg_match('/^id_/', $var)) + $vars[$var] = $val; + } + } + $v = split('[\?\&]', $url); + list(, $this->file) = each($v); + if (!$vars) { + while (list(,$var) = each($v)) { + list($name, $value) = split('=', $var, 2); + $name = urldecode($name); + $value = urldecode($value); + if (preg_match(',^(.*)\[\]$,', $name, $regs)) + $vars[$regs[1]][] = $value; + else + $vars[$name] = $value; + } + } + + if (is_array($vars)) { + foreach ($vars as $name => $value) { + // items supprimes + if (!preg_match('/^('. + (!_DIR_RESTREINT ? + '|lang|set_options|set_couleur|set_disp|set_ecran': + 'var_mode|show_docs') + . ')$/i', $name)) { + if (is_array($value)) + $this->arrays[$name] = $value; + else + $this->vars[$name] = $value; + } + } + } + } + + // + // Effacer une variable + // + function delVar($name) { + if(isset($this->vars[$name])) unset($this->vars[$name]); + if($this->arrays[$name]) unset($this->arrays[$name]); + } + + // + // Ajouter une variable + // (si aucune valeur n'est specifiee, prend la valeur globale actuelle) + // + function addVar($name, $value = '__global__') { + if ($value == '__global__') $value = $GLOBALS[$name]; + if (is_array($value)) + $this->arrays[$name] = $value; + else + $this->vars[$name] = $value; + } + + // + // Recuperer l'URL correspondant au lien + // + function getUrl($anchor = '') { + $url = $this->file; + if (!$url) $url = './'; + $query = ''; + foreach($this->vars as $name => $value) { + $query .= '&'.$name; + if (strlen($value)) + $query .= '='.urlencode($value); + } + + foreach ($this->arrays as $name => $table) + foreach ($table as $value) + $query .= '&'.$name.'[]='.urlencode($value); + + if ($query) $query = '?'. substr($query, 1); + if ($anchor) $anchor = '#'.$anchor; + return "$url$query$anchor"; + } + + // + // Recuperer le debut de formulaire correspondant au lien + // (tag ouvrant + entrees cachees representant les variables) + // + + function getForm($method = 'get', $query = '', $enctype = '') { + $form = "<form method='$method' action='".$this->file.$query."'"; + if ($enctype) $form .= " enctype='$enctype'"; + $form .= " style='border: 0px; margin: 0px;'>\n"; + foreach ($this->vars as $name => $value) { + $value = preg_replace(',&(#[0-9]+;),', '&\1', + htmlspecialchars($value)); + $form .= "<input type=\"hidden\" name=\"$name\" " + . "value=\"$value\" />\n"; + } + foreach ($this->arrays as $name => $table) + foreach ($table as $value) { + $value = preg_replace(',&(#[0-9]+;),', '&\1', + htmlspecialchars($value)); + $form .= "<input type=\"hidden\" name=\"".$name."[]\" " + . "value=\"".$value."\" />\n"; + } + + return $form; + } +} + + +// +// Gerer les valeurs meta +// +// Fonction lire_meta abandonnee, remplacee par son contenu. Ne plus utiliser +function lire_meta($nom) { + global $meta; + return $meta[$nom]; +} + + +// +// Traduction des textes de SPIP +// +function _T($texte, $args = '') { + include_ecrire('inc_lang.php3'); + $text = traduire_chaine($texte, $args); + if (!empty($GLOBALS['xhtml'])) { + include_ecrire("inc_charsets.php3"); + $text = html2unicode($text); + } + + return $text ? $text : + // pour les chaines non traduites + (($n = strpos($texte,':')) === false ? $texte : + substr($texte, $n+1)); +} + +// chaines en cours de traduction +function _L($text) { + if ($GLOBALS['test_i18n']) + return "<span style='color:red;'>$text</span>"; + else + return $text; +} + + +// Nommage bizarre des tables d'objets +function table_objet($type) { + if ($type == 'site' OR $type == 'syndic') + return 'syndic'; + else if ($type == 'forum') + return 'forum'; + else + return $type.'s'; +} +function id_table_objet($type) { + if ($type == 'site' OR $type == 'syndic') + return 'id_syndic'; + else if ($type == 'forum') + return 'id_forum'; + else + return 'id_'.$type; +} + + +// +// spip_timer : on l'appelle deux fois et on a la difference, affichable +// +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()) +function spip_touch($fichier, $duree=0, $touch=true) { + if (!($exists = @file_exists($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; +} + +// +// cron() : execution des taches de fond +// quand il est appele par spip_background.php3, il est gourmand ; +// quand il est appele par inc-public il n'est pas gourmand +// +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_ecrire('inc_cron.php3'); + spip_cron(); + } + } +} + + + +// +// Entetes les plus courants (voir inc_headers.php pour les autres) +// + +function http_gmoddate($lastmodified) { + return gmdate("D, d M Y H:i:s", $lastmodified); +} + +function http_last_modified($lastmodified, $expire = 0) { + $gmoddate = http_gmoddate($lastmodified); + if ($GLOBALS['HTTP_IF_MODIFIED_SINCE'] + AND !preg_match(',IIS/,', $_SERVER['SERVER_SOFTWARE'])) # MSoft IIS is dumb + { + $if_modified_since = preg_replace('/;.*/', '', + $GLOBALS['HTTP_IF_MODIFIED_SINCE']); + $if_modified_since = trim(str_replace('GMT', '', $if_modified_since)); + if ($if_modified_since == $gmoddate) { + include_ecrire('inc_headers.php'); + http_status(304); + $headers_only = true; + } + } + @Header ("Last-Modified: ".$gmoddate." GMT"); + if ($expire) + @Header ("Expires: ".http_gmoddate($expire)." GMT"); + return $headers_only; +} + +// envoyer le navigateur sur une nouvelle adresse + +function redirige_par_entete($url) { + spip_log("redirige $url"); + include_ecrire('inc_headers.php'); + spip_header("Location: $url"); + exit; +} + +// transformation XML des "&" en "&" +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)]'; ? > +function texte_script($texte) { + return str_replace('\'', '\\\'', str_replace('\\', '\\\\', $texte)); +} + +// +// find_in_path() : chercher un fichier nomme x selon le chemin rep1:rep2:rep3 +// + +function find_in_path ($filename, $path='AUTO') { + // Chemin standard depuis l'espace public + + if ($path == 'AUTO') { + $path = _SPIP_PATH; + if ($GLOBALS['dossier_squelettes']) + $path = $GLOBALS['dossier_squelettes'].'/:'.$path; + } + + // Depuis l'espace prive, remonter d'un cran + $racine = (_DIR_RESTREINT ? '' : '../'); + + // Parcourir le chemin + foreach (split(':', $path) as $dir) { + if (substr($dir, 0,1)<>'/') $dir = "$racine$dir"; + if (substr($dir, -1,1)<>'/') $dir .= "/"; + $f = "$dir$filename"; +# spip_log("find_in_path: essai $f"); + if (@is_readable($f)) { + return $f; + } + } +} + +// charger les definitions des plugins +function charger_plugins($plugins) { + foreach ($plugins as $plug) { + include(_DIR_RACINE.'plugins/'.$plug.'/version.php'); + } +#var_dump($plugins);var_dump($spip_pipeline);var_dump($spip_matrice);exit; +} + + +?> diff --git a/ecrire/inc_version.php3 b/ecrire/inc_version.php3 index a98fe4d587af152cee38f220fc11951c64e3b8a3..9ebe4d223f1ff542f6ba69048c06198cd25cec83 100644 --- a/ecrire/inc_version.php3 +++ b/ecrire/inc_version.php3 @@ -14,10 +14,6 @@ if (defined("_ECRIRE_INC_VERSION")) return; define("_ECRIRE_INC_VERSION", "1"); -function define_once ($constant, $valeur) { - if (!defined($constant)) define($constant, $valeur); -} - // 6 constantes incontournables et prioritaires define('_EXTENSION_PHP', '.php3'); # a etendre @@ -31,66 +27,6 @@ define('_FILE_CONNECT', (_FILE_CONNECT_INS . _EXTENSION_PHP) : false)); -// -// Gestion des inclusions et infos repertoires -// - -$included_files = array(); - -function include_local($file) { - if (@$GLOBALS['included_files'][$file]++) return; - - if (is_readable($file)) - include($file); - else spip_log($file . " illisible"); -} - -function include_ecrire($file) { -# Hack pour etre compatible avec les mes_options qui appellent cette fonction - define_once('_DIR_INCLUDE', _DIR_RESTREINT); - $file = _DIR_INCLUDE . $file; - if ($GLOBALS['included_files'][$file]++) return; - if (is_readable($file)) - include($file); - else spip_log($file . " illisible"); -} - -// charge un fichier perso ou, a defaut, standard -// et retourne si elle existe le nom de la fonction homonyme, ou de prefixe _dist - -function include_fonction($nom) { -# Hack pour etre compatible avec les mes_options qui appellent cette fonction - define_once('_DIR_INCLUDE', _DIR_RESTREINT); - - $nom = preg_replace("/\.php3?$/",'', basename($nom)); - $inc = ('inc_' . $nom . '.php'); - $f = find_in_path($inc); - if ($f && is_readable($f)) { - if (!$GLOBALS['included_files'][$f]++) include($f); - } else { - $f = _DIR_INCLUDE . $inc; - if (is_readable($f)) { - if (!$GLOBALS['included_files'][$f]++) include($f); - } else { - // provisoire avant renommage php/php3 - $f = _DIR_INCLUDE . ('inc_' . $nom . '.php3'); - if (is_readable($f)) { - if (!$GLOBALS['included_files'][$f]++) include($f); - } else $inc = ""; - } - } - $f = str_replace('-','_',$nom); // pour config-fonc etc. A renommer - if (function_exists($f)) - return $f; - elseif (function_exists($f .= "_dist")) - return $f; - else { - spip_log("fonction $nom indisponible" . - ($inc ? "" : "(aucun fichier inc_$f disponible)")); - exit; - } -} - // *********** traiter les variables ************ // Recuperer les superglobales $_GET si non definies @@ -106,141 +42,27 @@ foreach (array('_GET', '_POST', '_COOKIE', '_SERVER') as $_table) { $GLOBALS[$http_table_vars] = & $GLOBALS[$_table]; } +include(_DIR_RESTREINT . 'inc_magicquotes.php'); // Annuler les magic quotes s'il y en a @set_magic_quotes_runtime(0); if (@get_magic_quotes_gpc() AND strstr(serialize($_GET).serialize($_POST).serialize($_COOKIE), '\\')) { - include(_DIR_RESTREINT.'inc_magicquotes.php'); -} - -// Dirty hack contre le register_globals a 'Off' (PHP 4.1.x) -// A remplacer (un jour!) 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 -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', - - # ecrire/inc_texte - 'debut_intertitre', 'fin_intertitre', 'ligne_horizontale', - 'ouvre_ref', 'ferme_ref', 'ouvre_note', 'ferme_note', - 'les_notes', 'compt_note', 'nombre_surligne', - 'url_glossaire_externe', 'puce', 'puce_rtl' - ); - - // 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 (@ini_get('register_globals')) { - foreach ($refuse_gpc as $var) { - if (isset($GLOBALS[$var])) { - foreach (array('_GET', '_POST', '_COOKIE') as $_table) { - if ( - // demande par le client - isset ($GLOBALS[$_table][$var]) - // et pas modifie par les fichiers d'appel - AND $GLOBALS[$_table][$var] == $GLOBALS[$var] - ) // On ne sait pas si c'est un hack - { - # REMOTE_USER ou fond, c'est grave ; - # pour le reste (cookie 'lang', par exemple), simplement - # interdire la mise en cache de la page produite - switch ($var) { - case 'REMOTE_USER': - die ("$var interdite"); - break; - case 'fond': - if (!defined('_SPIP_PAGE')) - die ("$var interdite"); - break; - default: - define ('spip_interdire_cache', true); - } - } - } - } - } - foreach ($refuse_c as $var) { - if (isset($GLOBALS[$var])) { - foreach (array('_COOKIE') as $_table) { - if ( - // demande par le client - isset ($GLOBALS[$_table][$var]) - // et pas modifie par les fichiers d'appel - AND $GLOBALS[$_table][$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]) - 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; - } - } - } + magic_unquote($_GET); + magic_unquote($_POST); + magic_unquote($_COOKIE); +# if (@ini_get('register_globals')) // pas fiable + magic_unquote($GLOBALS); + +# et a la fin supprimer la variable anti-recursion devenue inutile (et meme nuisible, notamment si on teste $_POST) + unset($_GET['spip_recursions']); + unset($_POST['spip_recursions']); + unset($_COOKIE['spip_recursions']); + unset($_GLOBALS['spip_recursions']); } -spip_register_globals(); - - -// 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 - -function pipeline($cause, $val) { - global $spip_pipeline, $spip_matrice; - if (!is_array($spip_pipeline[$cause])) return $val; - - foreach ($spip_pipeline[$cause] as $plug) { - - // charger un fichier le cas echeant - if (!function_exists($plug)) { - if ($f = $spip_matrice[$plug]) { - include($f); - $ok = function_exists($plug); - } - if (!$ok) { - spip_log("Erreur - $plug n'est pas definie ($f)"); - return $val; - } - } - - // appliquer le filtre - $val = $plug($val); - } - - return $val; -} +spip_register_globals(); // // *** Parametrage par defaut de SPIP *** @@ -368,7 +190,6 @@ $spip_matrice = array (); $plugins = array(); // voir le contenu du repertoire /plugins/ - // Masquer les warning error_reporting(E_ALL ^ E_NOTICE); @@ -391,6 +212,10 @@ $extension_squelette = 'html'; // Droits d'acces maximum par defaut @umask(0); +function define_once ($constant, $valeur) { + if (!defined($constant)) define($constant, $valeur); +} + // // Inclure le fichier ecrire/mes_options.php3 (ou equivalent) @@ -399,21 +224,10 @@ if (@file_exists(_FILE_OPTIONS)) { include(_FILE_OPTIONS); } -// charger les definitions des plugins -function charger_plugins($plugins) { - foreach ($plugins as $plug) { - include(_DIR_RACINE.'plugins/'.$plug.'/version.php'); - } -#var_dump($plugins);var_dump($spip_pipeline);var_dump($spip_matrice);exit; -} -if ($plugins) - charger_plugins($plugins); - // // Definition des repertoires standards // - // la taille maxi des logos (0 : pas de limite) define_once('_LOGO_MAX_SIZE', 0); # poids en ko define_once('_LOGO_MAX_WIDTH', 0); # largeur en pixels @@ -473,11 +287,11 @@ define_once('_AUTH_USER_FILE', '.htpasswd'); define_once('_DOCTYPE_ECRIRE', "<!DOCTYPE HTML PUBLIC '-//W3C//DTD HTML 4.01 Transitional//EN' 'http://www.w3.org/TR/html4/loose.dtd'>\n"); +define_once('_SPIP_PATH', './:squelettes/:dist/:formulaires/'); -// tidy en ligne de commande (si on ne l'a pas en module php, -// ou si le module php ne marche pas) -// '/bin/tidy' ou '/usr/local/bin/tidy' ou tout simplement 'tidy' -#define_once('_TIDY_COMMAND', 'tidy'); +// charge les fonctions indispensables, + +include('inc_utils.php'); // Version courante de SPIP // Stockee sous forme de nombre decimal afin de faciliter les comparaisons @@ -503,7 +317,6 @@ $hash_recherche = ''; $hash_recherche_strict = ''; - // // Capacites php (en fonction de la version) // @@ -518,7 +331,6 @@ $flag_revisions = function_exists("gzcompress"); $flag_get_cfg_var = (@get_cfg_var('error_reporting') != ""); $flag_upload = (!$flag_get_cfg_var || (get_cfg_var('upload_max_filesize') > 0)); - // // Sommes-nous dans l'empire du Mal ? // @@ -532,44 +344,6 @@ if (strpos($_SERVER['SERVER_SOFTWARE'], '(Win') !== false) if (!headers_sent()) @header("Composed-By: SPIP $spip_version_affichee @ www.spip.net"); - -// -// Enregistrement des evenements -// -function spip_log($message, $logname='spip') { - static $compteur; - if ($compteur++ > 100) return; - - $pid = '(pid '.@getmypid().')'; - if (!$ip = $GLOBALS['REMOTE_ADDR']) $ip = '-'; - - $message = date("M d H:i:s")." $ip $pid " - .preg_replace("/\n*$/", "\n", $message); - - $logfile = _DIR_SESSIONS . $logname . '.log'; - if (@file_exists($logfile) && (@filesize($logfile) > 10*1024)) { - $rotate = true; - $message .= "[-- rotate --]\n"; - } - $f = @fopen($logfile, "ab"); - if ($f) { - fputs($f, $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); -} - - // // Infos sur le fichier courant // @@ -581,38 +355,6 @@ if (!$REQUEST_URI) { $REQUEST_URI .= '?'.$QUERY_STRING; } - -// API d'appel a la base de donnees -function spip_query($query) { - - // Remarque : si on est appele par l'install, - // la connexion initiale a ete faite avant - if (!$GLOBALS['db_ok']) { - // Essaie de se connecter - if (_FILE_CONNECT) - include_local(_FILE_CONNECT); - } - - // Erreur de connexion - if (!$GLOBALS['db_ok']) - return; - - // Vieux format de fichier connexion - // Note: la version 0.1 est compatible avec la 0.2 (mais elle gere - // moins bien les erreurs timeout sur SQL), on ne force donc pas l'upgrade - if ($GLOBALS['spip_connect_version'] < 0.1) { - if (!_DIR_RESTREINT) {$GLOBALS['db_ok'] = false; return;} - redirige_par_entete("upgrade.php3?reinstall=oui"); - exit; - } - - // Faire la requete - return spip_query_db($query); -} - - - - // // Reglage de l'output buffering : si possible, generer une sortie // compressee pour economiser de la bande passante @@ -643,191 +385,22 @@ if ($flag_ob AND strlen(ob_get_contents())==0 AND !headers_sent()) { } -class Link { - var $file; - var $vars; - var $arrays; - - // - // Contructeur : a appeler soit avec l'URL du lien a creer, - // soit sans parametres, auquel cas l'URL est l'URL courante - // - // parametre $root = demander un lien a partir de la racine du serveur / - function Link($url = '', $root = false) { - global $_POST; - static $link = ''; - - $this->vars = array(); - $this->arrays = array(); - - // Normal case - if ($link) { - if ($url) { - $v = split('[\?\&]', $url); - list(, $this->file) = each($v); - while (list(, $var) = each($v)) { - list($name, $value) = split('=', $var, 2); - $name = urldecode($name); - $value = urldecode($value); - if (preg_match(',^(.*)\[\]$,', $name, $regs)) { - $this->arrays[$regs[1]][] = $value; - } - else { - $this->vars[$name] = $value; - } - } - } - else { - $this->file = $link->file; - $this->vars = $link->vars; - $this->arrays = $link->arrays; - } - return; - } - - // Si aucun URL n'est specifie, creer le lien "propre" - // ou l'on supprime de l'URL courant les bidules inutiles - if (!$url) { - // GET variables are read from the original URL - // (_GET may contain additional variables - // introduced by rewrite-rules) - $url = $GLOBALS['REQUEST_URI']; - // Warning !!!! - // since non encoded arguments may be present - // (especially those coming from Rewrite Rule) - // find the begining of the query string - // to compute the script-name - if ($v = strpos($url,'?')) - $v = strrpos(substr($url, 0, $v), '/'); - else $v = strrpos($url, '/'); - if (!$root) $url = substr($url, $v + 1); - if (!$url) $url = "./"; - if (count($_POST)) { - $vars = array(); - foreach ($_POST as $var => $val) - if (preg_match('/^id_/', $var)) - $vars[$var] = $val; - } - } - $v = split('[\?\&]', $url); - list(, $this->file) = each($v); - if (!$vars) { - while (list(,$var) = each($v)) { - list($name, $value) = split('=', $var, 2); - $name = urldecode($name); - $value = urldecode($value); - if (preg_match(',^(.*)\[\]$,', $name, $regs)) - $vars[$regs[1]][] = $value; - else - $vars[$name] = $value; - } - } - - if (is_array($vars)) { - foreach ($vars as $name => $value) { - // items supprimes - if (!preg_match('/^('. - (!_DIR_RESTREINT ? - '|lang|set_options|set_couleur|set_disp|set_ecran': - 'var_mode|show_docs') - . ')$/i', $name)) { - if (is_array($value)) - $this->arrays[$name] = $value; - else - $this->vars[$name] = $value; - } - } - } - } - - // - // Effacer une variable - // - function delVar($name) { - if(isset($this->vars[$name])) unset($this->vars[$name]); - if($this->arrays[$name]) unset($this->arrays[$name]); - } - - // - // Ajouter une variable - // (si aucune valeur n'est specifiee, prend la valeur globale actuelle) - // - function addVar($name, $value = '__global__') { - if ($value == '__global__') $value = $GLOBALS[$name]; - if (is_array($value)) - $this->arrays[$name] = $value; - else - $this->vars[$name] = $value; - } - - // - // Recuperer l'URL correspondant au lien - // - function getUrl($anchor = '') { - $url = $this->file; - if (!$url) $url = './'; - $query = ''; - foreach($this->vars as $name => $value) { - $query .= '&'.$name; - if (strlen($value)) - $query .= '='.urlencode($value); - } - - foreach ($this->arrays as $name => $table) - foreach ($table as $value) - $query .= '&'.$name.'[]='.urlencode($value); - - if ($query) $query = '?'. substr($query, 1); - if ($anchor) $anchor = '#'.$anchor; - return "$url$query$anchor"; - } - - // - // Recuperer le debut de formulaire correspondant au lien - // (tag ouvrant + entrees cachees representant les variables) - // - - function getForm($method = 'get', $query = '', $enctype = '') { - $form = "<form method='$method' action='".$this->file.$query."'"; - if ($enctype) $form .= " enctype='$enctype'"; - $form .= " style='border: 0px; margin: 0px;'>\n"; - foreach ($this->vars as $name => $value) { - $value = preg_replace(',&(#[0-9]+;),', '&\1', - htmlspecialchars($value)); - $form .= "<input type=\"hidden\" name=\"$name\" " - . "value=\"$value\" />\n"; - } - foreach ($this->arrays as $name => $table) - foreach ($table as $value) { - $value = preg_replace(',&(#[0-9]+;),', '&\1', - htmlspecialchars($value)); - $form .= "<input type=\"hidden\" name=\"".$name."[]\" " - . "value=\"".$value."\" />\n"; - } - - return $form; - } -} - - // Lien vers la page demandee et lien nettoye ne contenant que des id_objet $clean_link = new Link(); -// -// Module de lecture/ecriture/suppression de fichiers utilisant flock() -// -include_ecrire('inc_flock.php3'); +if ($plugins) + charger_plugins($plugins); +// tidy en ligne de commande (si on ne l'a pas en module php, +// ou si le module php ne marche pas) +// '/bin/tidy' ou '/usr/local/bin/tidy' ou tout simplement 'tidy' +#define_once('_TIDY_COMMAND', 'tidy'); // -// Gerer les valeurs meta +// Module de lecture/ecriture/suppression de fichiers utilisant flock() // -// Fonction lire_meta abandonnee, remplacee par son contenu. Ne plus utiliser -function lire_meta($nom) { - global $meta; - return $meta[$nom]; -} +include_ecrire('inc_flock.php3'); // Lire les meta cachees @@ -839,192 +412,12 @@ if (!is_array($meta) AND _FILE_CONNECT) { ecrire_metas(); } -// -// Traduction des textes de SPIP -// -function _T($texte, $args = '') { - include_ecrire('inc_lang.php3'); - $text = traduire_chaine($texte, $args); - if (!empty($GLOBALS['xhtml'])) { - include_ecrire("inc_charsets.php3"); - $text = html2unicode($text); - } - - return $text ? $text : - // pour les chaines non traduites - (($n = strpos($texte,':')) === false ? $texte : - substr($texte, $n+1)); -} - -// chaines en cours de traduction -function _L($text) { - if ($GLOBALS['test_i18n']) - return "<span style='color:red;'>$text</span>"; - else - return $text; -} - // Langue principale du site $langue_site = $GLOBALS['meta']['langue_site']; if (!$langue_site) include_ecrire('inc_lang.php3'); $spip_lang = $langue_site; -// Nommage bizarre des tables d'objets -function table_objet($type) { - if ($type == 'site' OR $type == 'syndic') - return 'syndic'; - else if ($type == 'forum') - return 'forum'; - else - return $type.'s'; -} -function id_table_objet($type) { - if ($type == 'site' OR $type == 'syndic') - return 'id_syndic'; - else if ($type == 'forum') - return 'id_forum'; - else - return 'id_'.$type; -} - - -// -// spip_timer : on l'appelle deux fois et on a la difference, affichable -// -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()) -function spip_touch($fichier, $duree=0, $touch=true) { - if (!($exists = @file_exists($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; -} - -// -// cron() : execution des taches de fond -// quand il est appele par spip_background.php3, il est gourmand ; -// quand il est appele par inc-public il n'est pas gourmand -// -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_ecrire('inc_cron.php3'); - spip_cron(); - } - } -} - - - -// -// Entetes les plus courants (voir inc_headers.php pour les autres) -// - -function http_gmoddate($lastmodified) { - return gmdate("D, d M Y H:i:s", $lastmodified); -} - -function http_last_modified($lastmodified, $expire = 0) { - $gmoddate = http_gmoddate($lastmodified); - if ($GLOBALS['HTTP_IF_MODIFIED_SINCE'] - AND !preg_match(',IIS/,', $_SERVER['SERVER_SOFTWARE'])) # MSoft IIS is dumb - { - $if_modified_since = preg_replace('/;.*/', '', - $GLOBALS['HTTP_IF_MODIFIED_SINCE']); - $if_modified_since = trim(str_replace('GMT', '', $if_modified_since)); - if ($if_modified_since == $gmoddate) { - include_ecrire('inc_headers.php'); - http_status(304); - $headers_only = true; - } - } - @Header ("Last-Modified: ".$gmoddate." GMT"); - if ($expire) - @Header ("Expires: ".http_gmoddate($expire)." GMT"); - return $headers_only; -} - -// envoyer le navigateur sur une nouvelle adresse - -function redirige_par_entete($url) { - spip_log("redirige $url"); - include_ecrire('inc_headers.php'); - spip_header("Location: $url"); - exit; -} - -// transformation XML des "&" en "&" -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)]'; ? > -function texte_script($texte) { - return str_replace('\'', '\\\'', str_replace('\\', '\\\\', $texte)); -} - -// -// find_in_path() : chercher un fichier nomme x selon le chemin rep1:rep2:rep3 -// -define_once('_SPIP_PATH', './:squelettes/:dist/:formulaires/'); -function find_in_path ($filename, $path='AUTO') { - // Chemin standard depuis l'espace public - - if ($path == 'AUTO') { - $path = _SPIP_PATH; - if ($GLOBALS['dossier_squelettes']) - $path = $GLOBALS['dossier_squelettes'].'/:'.$path; - } - - // Depuis l'espace prive, remonter d'un cran - $racine = (_DIR_RESTREINT ? '' : '../'); - - // Parcourir le chemin - foreach (split(':', $path) as $dir) { - if (substr($dir, 0,1)<>'/') $dir = "$racine$dir"; - if (substr($dir, -1,1)<>'/') $dir .= "/"; - $f = "$dir$filename"; -# spip_log("find_in_path: essai $f"); - if (@is_readable($f)) { - return $f; - } - } -} - - // // Que faire si Spip n'est pas installe... sauf si justement on l'installe! //