Apache/URL Rewriting

Principe

modifier

Sur Internet on distingue globalement quatre types de redirection automatique[1] des navigateurs vers d'autres adresses que celles sur lesquelles ils arrivent :

  1. HTML (<meta http-equiv="refresh" content="1; URL=http://destination.fr">).
  2. PHP (header('Location: http://destination.fr');).
  3. Javascript[2].
  4. HTTP, proposant elle-même plusieurs techniques décrites ci-dessous.

Les règles de réécriture appliquées sur tout le serveur Apache se situent dans /etc/apache2/sites-available/default[3] ou /usr/local/apache/conf/httpd.conf.

Les fichiers .htaccess peuvent également en contenir[4] pour personnaliser chaque répertoire.

D'une manière générale, Apache permet de rediriger vers une autre page grâce à :

  1. Dans les systèmes Unix, la ligne Options +FollowSymLinks permet au navigateur de suivre les liens symboliques, en naviguant dans le système de fichier.
  2. Alias[5].
  3. AliasMatch (pareil avec du regex).
  4. RewriteRule réécrit l'URL selon des règles pouvant être précisées en regex[6]. Il nécessite d'être installé sous peine d'erreurs 500, via la commande Unix a2enmod rewrite.
  5. Redirect renvoie simplement ailleurs.
  6. RedirectMatch (pareil avec du regex).


 

Définir une redirection définitive (Redirect permanent) ne s'annule pas seulement en changeant le code qui l'a définit, mais dure jusqu'à expiration du cache serveur.

Pour réaliser des tests, on préférera donc définir une redirection 302 à une 301.

Pour forcer la suppression du cache des redirections, lancer htcacheclean[7]. Exemples :

  • Sur Debian 6 :
/usr/local/apache2/bin/htcacheclean -p/var/cache/ -l500M
  • Sur Ubuntu :
/usr/bin/htcacheclean -p/var/cache/ -l500M
  • Sur Gentoo :
/usr/local/apache/bin/htcacheclean -p/var/cache/edb/dep/usr/local/portage-ovh/www-apache -l500M

Puis relancer Apache.

 

Quand on redirige en 301 ou 302, on peut conserver les paramètres GET mais pas les POST. Pour ce faire, il faut rediriger en 307[8].

AllowOverride

modifier

Attention, si vous n'avez pas créé de VirtualHost, un VirtualHost par défaut est utilisé par apache2 qui ignore les fichiers .htaccess

Pour y remédier :

 sudo vi /etc/apache2/sites-enabled/000-default

et remplacez AllowOverride none par AllowOverride All (normalement 2 fois), puis relancez apache :

 sudo /etc/init.d/apache2 reload

Exemples

modifier

Dans la syntaxe suivante, le point seul représente le répertoire courant, et le slash seul la racine de l'URL (le domaine).

Pour changer le domaine vers localhost, tout en gardant la même URL :

SetEnv PHP_VER 5
Redirect / http://localhost/

Plus subtile, on peut changer l'ordre des paramètres initiaux dans l'URL de destination[9] :

<IfModule mod_rewrite.c>
    RewriteEngine on
    RewriteBase /
    RewriteRule ^/xtools/ec/*$ /xtools/pcount/index.php
</IfModule>

Plusieurs flags peuvent être placés en fin de ligne pour préciser la règle[10] :

  • [R] : (redirect) type de redirection (ex : R=301).
  • [L] : (last) stoppe les règles de réécritures après la courante.
  • [QSA] : (qsappend) conserve les paramètres GET en cas de redirection.
  • [NC] : (no case) insensibilité à la casse.

...

Pour rediriger les requêtes sous certaines condition il existe RewriteCond. Par exemple pour que toutes les pages non trouvées renvoient vers l'accueil au lieu d'afficher une erreur :

RewriteCond %{REQUEST_FILENAME} !-f
RewriteCond %{REQUEST_FILENAME} !-d
RewriteRule ^(.*)$ index.php

 

Dans la RewriteRule, $1 correspond au premier groupe de capture de la règle, et %1 à celui de la dernière condition.

Les paramètres du RewriteCond peuvent être[11][12] :

Variable Définition Exemple de valeur (sur https://fr.wikibooks.org/wiki/Apache?debug=true)
REQUEST_SCHEME Protocole de l'URL https
HTTP_HOST Adresse du site fr.wikibooks.org
SERVER_PORT Port logiciel 443
REQUEST_URI Uniform Resource Identifier /wiki/Apache
QUERY_STRING Paramètres HTTP debug=true
THE_REQUEST REQUEST_URI + QUERY_STRING + REQUEST_SCHEME GET /wiki/Apache?debug=true HTTP/1.1
REQUEST_FILENAME Chemin ciblé par l'URL dans le système de fichier /var/www/wiki/index.php

etc.

Retirer les doubles slashs dans les URL

modifier

La règle suivante permet de remplacer "//" ou "///" par "/"[13]

RewriteCond %{THE_REQUEST} //+(.+)\sHTTP
RewriteRule ^ /%1 [L,R]

Autre solution, qui évite la redirection si les paramètres de l'URL en contiennent une autre (avec "http(s)://") :

RewriteCond %{QUERY_STRING} !/(https?:\/\/)/ [NC]
RewriteCond %{REQUEST_URI} ^(.+)//+(.+)$
RewriteRule ^(.*)$ %1/%2 [QSA]

Références

modifier