Apache redirect vs. rewrite
February 24, 2014
Apache rewrite rules can be very confusing at times. They're very powerful, but not easily understood or mastered. I don't claim to be an expert by any means, but I recently came to some clarity on this issue:
What is the difference between Apache redirect and rewrite?
Redirect
A redirect is a server response that tells the client to make a new, modified request. The dialog goes something like this:
Client: “GET /page.php.”
Server: “Sorry, /page.php has moved. Ask for /otherpage.php instead”.
Client: “Fine. GET /otherpage.php then.”
Server: “OK. Here’s the content of /otherpage.php.”
Because a second request is made, now for a different resource, the URL changes in the browser's address bar. The user is aware that his original request has been redirected to something different than what he typed.
The redirect rule in your .htaccess file would look like this:
Redirect 301 /page.php /otherpage.php
You can send whatever response code is appropriate. If you don't give one, a 302 response will be sent, which is the default for a redirect. See the list of HTTP status codes for more information.
Rewrite
A rewrite maps the requested path to some different path on the server, based on whatever rewrite rules are defined. The server does this without notifying the client, and then serves up the content from the new path. It goes something like this:
Client: “GET /page.php.”
Server: “Let me consult the rewrite rules. Ok, it looks like all requests for /page.php should be served from /otherpage.php. Here’s the content of /otherpage.php.”
The original request was fulfilled by substituting a different path for the one that was requested. The client is unaware that the content being served came from /otherpage.php
because the URL remains unchanged in the address bar.
The rewrite rule in your .htaccess
file would look like this:
RewriteEngine On
RewriteRule /page.php /otherpage.php
Rewriting is often used to map pretty URLs (that make sense to the user) to uglier ones on the server that actually retrieve the content. Using this technique you can take a request for something like /shoes/running/nike
and map it to a script that serves the correct content based on query string variables, such as /products.php?product=shoes&type=running&brand=nike
. You probably don't want the user to be redirected to that ugly URL, so you just rewrite it to whatever it needs to be on the server side and serve up the contents. Nobody needs to know what's going on behind the scenes.
However, if you DO want the user to be redirected to the new path, you can force a redirect by using the [R]
flag after your rule:
RewriteRule /page.php /otherpage.php [RL]
Usually you want to stop rewriting at this point, so you should also include the [L]
flag to make this the last rule that gets evaluated. Otherwise, any rewrite rules that follow this one will also be applied before the redirect happens, which could result in the user being redirected to the wrong place.
Examples of redirects
This rule will redirect all requests to another site:
# Redirect all requests to another site
RedirectMatch 301 .* http://example.com
This one redirects from one directory to another directory:
# Redirect requests from one directory to another directory
RedirectMatch ^/directory/(.*) /other-directory/$1
Here $1
is a back-reference to the portion of the pattern in parentheses. So anything beyond the directory will remain the same.