Routing helps Craft smartly handle requests to your site. When a request comes in to Craft, it checks to determine where to route the request.
The checks detailed below explain how this process works. This information can helpful to you when troubleshooting template loading, plugin action URLs, dynamic routes, and unexpected 404 errors.
Here is how Craft handles each request:
-
Should Craft handle this request in the first place?
It’s important to keep in mind that Craft doesn’t actually get involved for every request that touches your server – only requests that go to your
index.php
file.The
.htaccess
file that comes with Craft will redirect all would-be 404 requests over toindex.php
behind the scenes. Because of this Craft responds to URLs that don’t point to a valid directory or file in your web root. But if you point your browser directly at a file that does exist (such as an image URL), your web server will serve that file directly without Craft intervening. -
Is it an action request?
Action requests either have a URL that begins with
actions/
(or whatever your actionTrigger config setting is set to), or anaction
parameter in the POST request or the query string.Craft routes action requests to a controller action that perform actions. Craft has system Controller actions for core actions, but plugins may also have Controllers that define their own custom actions.
The request doesn’t necessarily end after a controller call. The controller may allow it to keep going.
-
Is it an entry/category request?
If the URI matches an entry’s or category’s URI, Craft loads the section’s template or category group’s template. Craft makes the matched element available to the template via a pre-populated
entry
orcategory
variable.(This step is not limited to entries and categories – plugins are also capable of adding their own types of elements, which can opt to have their own dedicated URLs.)
-
Does the URI match any Dynamic Routes?
If the URI matches any dynamic routes, the template specified by that route will get loaded. If the route contains any tokens, they will be available as variables to that template.
-
Does the URI match a template?
Finally, Craft will check if the URI is a valid template path. If it is, Craft will return the matched template. Note: if any of the URI segments begin with an underscore (
_
), Craft will return a 404. Craft hides from direct access any template path segments that begin with an underscore. -
404
If none of the above checks are successful, Craft will return a 404. You can customize your site’s 404 page by placing a
404.html
template at the root of yourcraft/templates/
directory.
In some cases you want a URL to load a template, but you don’t want the URI to match the template path.
A good example of this is a yearly archive page, where you want the year to be one of the segments in the URL (e.g. news/archive/2017
). It would be silly to create a new template for every year. Instead, you should set up a new Dynamic Route.
To create a new Route, go to Settings → Routes and click the New Route button. A modal window will appear where you can define the route settings.
The modal has the following settings:
- What should the URI look like?
- Which template should get loaded?
The first setting can contain “tokens”, which represent a range of possible matches, rather than a specific string. (The “year” token, for example, represents four consecutive digits.) When you click on a token, Craft inserts it into the URI setting wherever the cursor is.
If you want to match URIs that look like news/archive/2017
, you type news/archive/
into the URI field, and then click on the “year” token.
Note: Your URI should not begin with a slash (/).
After defining your URI pattern and entering a template path, click the Save button. The modal will close, revealing your new route on the page.
When you point your browser to http://example.com/news/archive/2017
, it will match your new route, and Craft will load the specified template.
The value of the "year" token will also be available to the template as a variable called year
.
The following tokens are available to the URI setting:
-
- – Any string of characters, except for a forward slash (/)
- day – Day of a month (1-31 or 01-31)
- month – Numeric representation of a month (1-12 or 01-12)
- number – Any positive integer
- page – Any positive integer
- slug – Any string of characters, except for a forward slash (/)
- tag – Any string of characters, except for a forward slash (/)
- year – Four consecutive digits
If you need to set up routes to controller actions, or you need to create a template route that matches a URI pattern that’s not covered by the available route tokens in Settings → Routes, you can set up your routes in craft/config/routes.php
as well.
When a request comes in, Craft checks this file first, and then the routes defined in Settings → Routes.
Routes in craft/config/routes.php
generally follow this syntax:
// routes "news/subscribe" to a "lists/subscribe/news" action:
'news/subscribe' => 'lists/subscribe/news',
// routes "news/archive/2020" to a "news/_archive" template:
'news/archive/<year:\d{4}>' => ['template' => 'news/_archive'],
If your Craft installation has multiple sites, then you can target locale specific routes via:
'news/archive/<year:\d{4}>' => 'news/_archive',
'de' => array(
'nachrichten/archiv/<year:\d{4}>' => 'de/nachrichten/_archiv',
),
You can find a full specification for defining URL rules and routes at the Yii documentation. The only Craft-specific addition is support for template routes (['template' => 'some/template']
).
If you have a template route that contains subpatterns, Craft passes a matches
array to the matched template containing them. For example, with this route:
'news/(\d{4})/(\d{2})' => 'news/_archive',
If you access http://example.com/news/2020/10
, your news/_archive.html
template will get loaded with a matches
variable set to this:
[
0 => 'news/2020/10',
1 => '2020',
2 => '10'
]
If you specify any named subpatterns, then those matches will also get their own variables. For example, with this route:
'news/<year:\d{4}>/<month:\d{2}>' => 'news/_archive',
…if you access http://example.com/news/2020/10
, your news/_archive.html
template will get loaded with year
and month
variables set to '2017'
and '04'
.