123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410 |
- <?php
- /***************************************************************************\
- * SPIP, Systeme de publication pour l'internet *
- * *
- * Copyright (c) 2001-2014 *
- * 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; // securiser
- # donner un exemple d'url pour le formulaire de choix
- define('URLS_PROPRES_EXEMPLE', 'Titre-de-l-article -Rubrique-');
- # specifier le form de config utilise pour ces urls
- define('URLS_PROPRES_CONFIG', 'propres');
- // TODO: une interface permettant de verifier qu'on veut effectivment modifier
- // une adresse existante
- defined('CONFIRMER_MODIFIER_URL') || define('CONFIRMER_MODIFIER_URL', false);
- /*
- - Comment utiliser ce jeu d'URLs ?
- Recopiez le fichier "htaccess.txt" du repertoire de base du site SPIP sous
- le sous le nom ".htaccess" (attention a ne pas ecraser d'autres reglages
- que vous pourriez avoir mis dans ce fichier) ; si votre site est en
- "sous-repertoire", vous devrez aussi editer la ligne "RewriteBase" ce fichier.
- Les URLs definies seront alors redirigees vers les fichiers de SPIP.
- Dans les pages de configuration, choisissez 'propres' comme type d'url
- SPIP calculera alors ses liens sous la forme
- "Mon-titre-d-article".
- La variante 'propres2' ajoutera '.html' aux adresses generees :
- "Mon-titre-d-article.html"
- Variante 'qs' (experimentale) : ce systeme fonctionne en "Query-String",
- c'est-a-dire sans utilisation de .htaccess ; les adresses sont de la forme
- "/?Mon-titre-d-article"
- */
- if (!defined('_terminaison_urls_propres')) define ('_terminaison_urls_propres', '');
- if (!defined('_debut_urls_propres')) define ('_debut_urls_propres', '');
- $config_urls_propres = isset($GLOBALS['meta']['urls_propres'])?unserialize($GLOBALS['meta']['urls_propres']):array();
- // pour choisir le caractere de separation titre-id en cas de doublon
- // (ne pas utiliser '/')
- if (!defined('_url_propres_sep_id')) define('_url_propres_sep_id',isset($config_urls_propres['url_propres_sep_id'])?$config_urls_propres['url_propres_sep_id']:'-');
- // option pour tout passer en minuscules
- if (!defined('_url_minuscules')) define('_url_minuscules',isset($config_urls_propres['url_minuscules'])?$config_urls_propres['url_minuscules']:0);
- if (!defined('_URLS_PROPRES_MAX')) define('_URLS_PROPRES_MAX', isset($config_urls_propres['URLS_PROPRES_MAX'])?$config_urls_propres['URLS_PROPRES_MAX']:80);
- if (!defined('_URLS_PROPRES_MIN')) define('_URLS_PROPRES_MIN', isset($config_urls_propres['URLS_PROPRES_MIN'])?$config_urls_propres['URLS_PROPRES_MIN']:3);
- if (!defined('_url_sep_id')) define('_url_sep_id',_url_propres_sep_id);
- // Ces chaines servaient de marqueurs a l'epoque ou les URL propres devaient
- // indiquer la table ou les chercher (articles, auteurs etc),
- // et elles etaient retirees par les preg_match dans la fonction ci-dessous.
- // Elles peuvent a present etre definies a "" pour avoir des URL plus jolies.
- // Les preg_match restent necessaires pour gerer les anciens signets.
- if (!defined('_MARQUEUR_URL')) define('_MARQUEUR_URL', serialize(array('rubrique1' => '-', 'rubrique2' => '-', 'breve1' => '+', 'breve2' => '+', 'site1' => '@', 'site2' => '@', 'auteur1' => '_', 'auteur2' => '_', 'mot1' => '+-', 'mot2' => '-+')));
- // Retire les marqueurs de type dans une URL propre ancienne maniere
- // http://code.spip.net/@retirer_marqueurs_url_propre
- function retirer_marqueurs_url_propre($url_propre) {
- if (preg_match(',^[+][-](.*?)[-][+]$,', $url_propre, $regs)) {
- return $regs[1];
- }
- else if (preg_match(',^([-+_@])(.*?)\1?$,', $url_propre, $regs)) {
- return $regs[2];
- }
- // les articles n'ont pas de marqueur
- return $url_propre;
- }
- // Pipeline pour creation d'une adresse : il recoit l'url propose par le
- // precedent, un tableau indiquant le titre de l'objet, son type, son id,
- // et doit donner en retour une chaine d'url, sans se soucier de la
- // duplication eventuelle, qui sera geree apres
- // http://code.spip.net/@creer_chaine_url
- function urls_propres_creer_chaine_url($x) {
- // NB: ici url_old ne sert pas, mais un plugin qui ajouterait une date
- // pourrait l'utiliser pour juste ajouter la
- $url_old = $x['data'];
- $objet = $x['objet'];
- include_spip('inc/filtres');
- include_spip('action/editer_url');
- if (!$url = url_nettoyer($objet['titre'],_URLS_PROPRES_MAX,_URLS_PROPRES_MIN,'-',_url_minuscules?'strtolower':''))
- $url = $objet['type'].$objet['id_objet'];
- $x['data'] = $url;
- return $x;
- }
- // Trouver l'URL associee a la n-ieme cle primaire d'une table SQL
- // http://code.spip.net/@declarer_url_propre
- function declarer_url_propre($type, $id_objet) {
- $trouver_table = charger_fonction('trouver_table', 'base');
- $desc = $trouver_table(table_objet($type));
- $table = $desc['table'];
- $champ_titre = $desc['titre'] ? $desc['titre'] : 'titre';
- $col_id = @$desc['key']["PRIMARY KEY"];
- if (!$col_id) return false; // Quand $type ne reference pas une table
- $id_objet = intval($id_objet);
- // Recuperer une URL propre correspondant a l'objet.
- // mais urls a 1 segment uniquement (pas d'urls /)
- // de preference avec id_parent=0, puis perma, puis par date desc
- $row = sql_fetsel("U.url, U.date, U.id_parent, U.perma, $champ_titre",
- "$table AS O LEFT JOIN spip_urls AS U ON (U.type='$type' AND U.id_objet=O.$col_id)",
- "O.$col_id=$id_objet AND (U.segments IS NULL OR U.segments=1)", '', 'U.id_parent=0 DESC, U.perma DESC, U.date DESC', 1);
- // en SQLite le left join retourne du vide si il y a une url mais qui ne correspond pas pour la condition sur le segment
- // on verifie donc que l'objet existe bien avant de sortir ou de creer une url pour cet objet
- if (!$row)
- $row = sql_fetsel("'' as url, '' as date, 0 as id_parent, 0 as perma, $champ_titre",
- "$table AS O",
- "O.$col_id=$id_objet");
- if (!$row) return ""; # Quand $id_objet n'est pas un numero connu
- $url_propre = $row['url'];
- // si url_propre connue mais avec id_parent non nul, essayer de reinserer tel quel avec id_parent=0
- if ($url_propre AND $row['id_parent']){
- include_spip('action/editer_url');
- $set = array('url' => $url_propre, 'type' => $type, 'id_objet' => $id_objet, 'perma' => $row['perma']);
- // si on arrive pas a reinserer tel quel, on annule url_propre pour forcer un recalcul d'url
- if (!url_insert($set,false,_url_propres_sep_id))
- $url_propre = "";
- else
- $url_propre = $row['url'] = $set['url'];
- }
- // Se contenter de cette URL si elle existe ;
- // sauf si on invoque par "voir en ligne" avec droit de modifier l'url
- // l'autorisation est verifiee apres avoir calcule la nouvelle url propre
- // car si elle ne change pas, cela ne sert a rien de verifier les autorisations
- // qui requetent en base
- $modifier_url = (defined('_VAR_URLS') AND _VAR_URLS AND !$row['perma']);
- if ($url_propre AND !$modifier_url)
- return $url_propre;
- // Sinon, creer une URL
- $url = pipeline('propres_creer_chaine_url',
- array(
- 'data' => $url_propre, // le vieux url_propre
- 'objet' => array_merge($row,
- array('type' => $type, 'id_objet' => $id_objet)
- )
- )
- );
- // Eviter de tamponner les URLs a l'ancienne (cas d'un article
- // intitule "auteur2")
- include_spip('inc/urls');
- $objets = urls_liste_objets();
- if (preg_match(',^('.$objets.')[0-9]+$,', $url, $r)
- AND $r[1] != $type)
- $url = $url._url_propres_sep_id.$id_objet;
- // Pas de changement d'url
- if ($url == $url_propre)
- return $url_propre;
- // verifier l'autorisation, maintenant qu'on est sur qu'on va agir
- if ($modifier_url) {
- include_spip('inc/autoriser');
- $modifier_url = autoriser('modifierurl', $type, $id_objet);
- }
- // Verifier si l'utilisateur veut effectivement changer l'URL
- if ($modifier_url
- AND CONFIRMER_MODIFIER_URL
- AND $url_propre
- AND $url != preg_replace('/'.preg_quote(_url_propres_sep_id,'/').'.*/', '', $url_propre))
- $confirmer = true;
- else
- $confirmer = false;
- if ($confirmer AND !_request('ok')) {
- die ("vous changez d'url ? $url_propre -> $url");
- }
- $set = array('url' => $url, 'type' => $type, 'id_objet' => $id_objet);
- include_spip('action/editer_url');
- if (!url_insert($set,$confirmer,_url_propres_sep_id))
- return $url_propre; //serveur out ? retourner au mieux
- return $set['url'];
- }
- // http://code.spip.net/@_generer_url_propre
- function _generer_url_propre($type, $id, $args='', $ancre='') {
- if ($generer_url_externe = charger_fonction("generer_url_$type",'urls',true)) {
- $url = $generer_url_externe($id, $args, $ancre);
- if (NULL != $url) return $url;
- }
- // Mode compatibilite pour conserver la distinction -Rubrique-
- if (_MARQUEUR_URL) {
- $marqueur = unserialize(_MARQUEUR_URL);
- $marqueur1 = isset($marqueur[$type.'1']) ? $marqueur[$type.'1'] : '' ; // debut '+-'
- $marqueur2 = isset($marqueur[$type.'2']) ? $marqueur[$type.'2'] : '' ; // fin '-+'
- } else
- $marqueur1 = $marqueur2 = '';
- // fin
- // Mode propre
- $propre = declarer_url_propre($type, $id);
- if ($propre === false) return ''; // objet inconnu. raccourci ?
- if ($propre) {
- $url = _debut_urls_propres
- . $marqueur1
- . $propre
- . $marqueur2
- . _terminaison_urls_propres;
- if (!defined('_SET_HTML_BASE') OR !_SET_HTML_BASE)
- // Repositionne l'URL par rapport a la racine du site (#GLOBALS)
- $url = str_repeat('../', $GLOBALS['profondeur_url']).$url;
- else
- $url = _DIR_RACINE . $url;
- } else {
- // objet connu mais sans possibilite d'URL lisible, revenir au defaut
- include_spip('base/connect_sql');
- $id_type = id_table_objet($type);
- $url = _DIR_RACINE . get_spip_script('./')."?"._SPIP_PAGE."=$type&$id_type=$id";
- }
- // Ajouter les args
- if ($args)
- $url .= ((strpos($url, '?')===false) ? '?' : '&') . $args;
- // Ajouter l'ancre
- if ($ancre)
- $url .= "#$ancre";
- return $url;
- }
- // retrouve le fond et les parametres d'une URL propre
- // ou produit une URL propre si on donne un parametre
- // @return array([contexte],[type],[url_redirect],[fond]) : url decodee
- // http://code.spip.net/@urls_propres_dist
- function urls_propres_dist($i, $entite, $args='', $ancre='') {
- if (is_numeric($i))
- return _generer_url_propre($entite, $i, $args, $ancre);
- $url = $i;
- $id_objet = $type = 0;
- $url_redirect = null;
- // recuperer les &debut_xx;
- if (is_array($args))
- $contexte = $args;
- else
- parse_str($args,$contexte);
- // Migration depuis anciennes URLs ?
- // traiter les injections domain.tld/spip.php/n/importe/quoi/rubrique23
- if ($GLOBALS['profondeur_url']<=0
- AND $_SERVER['REQUEST_METHOD'] != 'POST') {
- include_spip('inc/urls');
- $r = nettoyer_url_page($i, $contexte);
- if ($r) {
- list($contexte, $type,,, $suite) = $r;
- $_id = id_table_objet($type);
- $id_objet = $contexte[$_id];
- $url_propre = generer_url_entite($id_objet, $type);
- if (strlen($url_propre)
- AND !strstr($url,$url_propre)) {
- list(,$hash) = explode('#', $url_propre);
- $args = array();
- foreach(array_filter(explode('&', $suite)) as $fragment) {
- if ($fragment != "$_id=$id_objet")
- $args[] = $fragment;
- }
- $url_redirect = generer_url_entite($id_objet, $type, join('&',array_filter($args)), $hash);
- return array($contexte, $type, $url_redirect, $type);
- }
- }
- }
- /* Fin compatibilite anciennes urls */
- // Chercher les valeurs d'environnement qui indiquent l'url-propre
- if (isset($_SERVER['REDIRECT_url_propre']))
- $url_propre = $_SERVER['REDIRECT_url_propre'];
- elseif (isset($_ENV['url_propre']))
- $url_propre = $_ENV['url_propre'];
- else {
- // ne prendre que le segment d'url qui correspond, en fonction de la profondeur calculee
- $url = ltrim($url,'/');
- $url = explode('/',$url);
- while (count($url)>$GLOBALS['profondeur_url']+1)
- array_shift($url);
- $url = implode('/',$url);
- $url_propre = preg_replace(',[?].*,', '', $url);
- }
- // Mode Query-String ?
- $is_qs = false;
- if (!$url_propre
- AND preg_match(',[?]([^=/?&]+)(&.*)?$,', $url, $r)) {
- $url_propre = $r[1];
- $is_qs = true;
- }
- if (!$url_propre
- OR $url_propre==_DIR_RESTREINT_ABS
- OR $url_propre==_SPIP_SCRIPT) return; // qu'est-ce qu'il veut ???
-
- // gerer le cas de retour depuis des urls arbos
- // mais si url arbo ne trouve pas, on veut une 404 par securite
- if ($GLOBALS['profondeur_url']>0 AND !defined('_FORCE_URLS_PROPRES')){
- $urls_anciennes = charger_fonction('arbo','urls');
- return $urls_anciennes($url_propre, $entite, $contexte);
- }
-
- include_spip('base/abstract_sql'); // chercher dans la table des URLS
- // Compatibilite avec propres2
- $url_propre = preg_replace(',\.html$,i', '', $url_propre);
- // Revenir en utf-8 si encodage type %D8%A7 (farsi)
- $url_propre = rawurldecode($url_propre);
- // Compatibilite avec les anciens marqueurs d'URL propres
- // Tester l'entree telle quelle (avec 'url_libre' des sites ont pu avoir des entrees avec marqueurs dans la table spip_urls)
- if (!$row = sql_fetsel('id_objet, type, date, url', 'spip_urls', 'url='.sql_quote($url_propre, '', 'TEXT'))) {
- // Sinon enlever les marqueurs eventuels
- $url_propre2 = retirer_marqueurs_url_propre($url_propre);
- $row = sql_fetsel('id_objet, type, date, url', 'spip_urls', 'url='.sql_quote($url_propre2, '', 'TEXT'));
- }
- if ($row) {
- $type = $row['type'];
- $col_id = id_table_objet($type);
- $contexte[$col_id] = $row['id_objet'];
- $entite = $row['type'];
- // Si l'url est vieux, donner le nouveau
- if ($recent = sql_fetsel('url, date', 'spip_urls',
- 'type='.sql_quote($row['type'], '', 'TEXT').' AND id_objet='.sql_quote($row['id_objet'])
- .' AND date>'.sql_quote($row['date'], '', 'TEXT')
- .' AND url<>'.sql_quote($row['url'], '', 'TEXT'), '', 'date DESC', 1)) {
- // Mode compatibilite pour conserver la distinction -Rubrique-
- if (_MARQUEUR_URL) {
- $marqueur = unserialize(_MARQUEUR_URL);
- $marqueur1 = $marqueur[$type.'1']; // debut '+-'
- $marqueur2 = $marqueur[$type.'2']; // fin '-+'
- } else
- $marqueur1 = $marqueur2 = '';
- $url_redirect = $marqueur1 . $recent['url'] . $marqueur2;
- }
- }
- if ($entite=='' OR $entite=='type_urls' /* compat .htaccess 2.0 */) {
- if ($type) {
- $entite = objet_type($type);
- } else {
- // Si ca ressemble a une URL d'objet, ce n'est pas la home
- // et on provoque un 404
- if (preg_match(',^[^\.]+(\.html)?$,', $url)) {
- $entite = '404';
- $contexte['erreur'] = '';
- // l'url n'existe pas...
- // on ne sait plus dire de quel type d'objet il s'agit
- // sauf si on a le marqueur. et la c'est un peu sale...
- if (_MARQUEUR_URL) {
- $fmarqueur = @array_flip(unserialize(_MARQUEUR_URL));
- preg_match(',^([+][-]|[-+@_]),', $url_propre, $regs);
- $objet = $regs ? substr($fmarqueur[$regs[1]],0,n-1) : 'article';
- $contexte['erreur'] = _T(
- ($objet=='rubrique' OR $objet=='breve')
- ? 'public:aucune_'.$objet
- : 'public:aucun_'.$objet
- );
- }
- }
- }
- }
- return array($contexte, $entite, $url_redirect, $is_qs?$entite:null);
- }
- ?>
|