Laravel uses the latest features of PHP 5.3 to make routing simple and expressive. It's important that building everything from APIs to complex web applications is as easy as possible. Routes are typically defined in application/routes.php.
Unlike many other frameworks with Laravel it's possible to embed application logic in two ways. While controllers are the most common way to implement application logic it's also possible to embed your logic directly into routes. This is especially nice for small sites that contain only a few pages as you don't have to create a bunch of controllers just to expose half a dozen methods or put a handful of unrelated methods into the same controller and then have to manually designate routes that point to them.
In the following example the first parameter is the route that you're "registering" with the router. The second parameter is the function containing the logic for that route. Routes are defined without a front-slash. The only exception to this is the default route which is represented with only a front-slash.
Note: Routes are evaluated in the order that they are registered, so register any "catch-all" routes at the bottom of your routes.php file.
Route::get('/', function()
{
return "Hello World!";
});
Route::any('/', function()
{
return "Hello World!";
});
Route::post('user', function()
{
//
});
Route::put('user/(:num)', function($id)
{
//
});
Route::delete('user/(:num)', function($id)
{
//
});
Registering a single URI for multiple HTTP verbs:
Router::register(array('GET', 'POST'), $uri, $callback);
Route::get('user/(:num)', function($id)
{
//
});
Route::get('post/(:any)', function($title)
{
//
});
Route::get('page/(:any?)', function($page = 'index')
{
//
});
If a request enters your application but does not match any existing route, the 404 event will be raised. You can find the default event handler in your application/routes.php file.
Event::listen('404', function()
{
return Response::error('404');
});
You are free to change this to fit the needs of your application!
Futher Reading:
Route filters may be run before or after a route is executed. If a "before" filter returns a value, that value is considered the response to the request and the route is not executed, which is conveniont when implementing authentication filters, etc. Filters are typically defined in application/routes.php.
Route::filter('filter', function()
{
return Redirect::to('home');
});
Route::get('blocked', array('before' => 'filter', function()
{
return View::make('blocked');
}));
Route::get('download', array('after' => 'log', function()
{
//
}));
Route::get('create', array('before' => 'auth|csrf', function()
{
//
}));
Route::get('panel', array('before' => 'role:admin', function()
{
//
}));
Sometimes you may want to attach a filter to all requests that begin with a given URI. For example, you may want to attach the "auth" filter to all requests with URIs that begin with "admin". Here's how to do it:
Route::filter('pattern: admin/*', 'auth');
Optionally you can register filters directly when attaching filters to a given URI by supplying an array with the name of the filter and a callback.
Route::filter('pattern: admin/*', array('name' => 'auth', function()
{
//
}));
Laravel has two "global" filters that run before and after every request to your application. You can find them both in the application/routes.php file. These filters make great places to start common bundles or add global assets.
Note: The after filter receives the Response object for the current request.
Route groups allow you to attach a set of attributes to a group of routes, allowing you to keep your code neat and tidy.
Route::group(array('before' => 'auth'), function()
{
Route::get('panel', function()
{
//
});
Route::get('dashboard', function()
{
//
});
});
Constantly generating URLs or redirects using a route's URI can cause problems when routes are later changed. Assigning the route a name gives you a convenient way to refer to the route throughout your application. When a route change occurs the generated links will point to the new route with no further configuration needed.
Route::get('/', array('as' => 'home', function()
{
return "Hello World";
}));
$url = URL::to_route('home');
return Redirect::to_route('home');
Once you have named a route, you may easily check if the route handling the current request has a given name.
if (Request::route()->is('home'))
{
// The "home" route is handling the request!
}
When defining routes, you may use the "https" attribute to indicate that the HTTPS protocol should be used when generating a URL or Redirect to that route.
Route::get('login', array('https' => true, function()
{
return View::make('login');
}));
Route::secure('GET', 'login', function()
{
return View::make('login');
});
Bundles are Laravel's modular package system. Bundles can easily be configured to handle requests to your application. We'll be going over bundles in more detail in another document. For now, read through this section and just be aware that not only can routes be used to expose functionality in bundles, but they can also be registered from within bundles.
Let's open the application/bundles.php file and add something:
return array(
'admin' => array('handles' => 'admin'),
);
Notice the new handles option in our bundle configuration array? This tells Laravel to load the Admin bundle on any requests where the URI begins with "admin".
Now you're ready to register some routes for your bundle, so create a routes.php file within the root directory of your bundle and add the following:
Route::get('(:bundle)', function()
{
return 'Welcome to the Admin bundle!';
});
Let's explore this example. Notice the (:bundle) place-holder? That will be replaced with the value of the handles clause that you used to register your bundle. This keeps your code D.R.Y. and allows those who use your bundle to change it's root URI without breaking your routes! Nice, right?
Of course, you can use the (:bundle) place-holder for all of your routes, not just your root route.
Route::get('(:bundle)/panel', function()
{
return "I handle requests to admin/panel!";
});
Controllers provide another way to manage your application logic. If you're unfamiliar with controllers you may want to read about controllers and return to this section.
It is important to be aware that all routes in Laravel must be explicitly defined, including routes to controllers. This means that controller methods that have not been exposed through route registration cannot be accessed. It's possible to automatically expose all methods within a controller using controller route registration. Controller route registrations are typically defined in application/routes.php.
Most likely, you just want to register all of the controllers in your application's "controllers" directory. You can do it in one simple statement. Here's how:
Route::controller(Controller::detect());
The Controller::detect method simply returns an array of all of the controllers defined for the application.
If you wish to automatically detect the controllers in a bundle, just pass the bundle name to the method. If no bundle is specified, the application folder's controller directory will be searched.
Route::controller(Controller::detect('admin'));
Route::controller('home');
Route::controller(array('dashboard.panel', 'admin'));
Once a controller is registered, you may access its methods using a simple URI convention:
http://localhost/controller/method/arguments
This convention is similar to that employed by CodeIgniter and other popular frameworks, where the first segment is the controller name, the second is the method, and the remaining segments are passed to the method as arguments. If no method segment is present, the "index" method will be used.
This routing convention may not be desirable for every situation, so you may also explicitly route URIs to controller actions using a simple, intuitive syntax.
Route::get('welcome', 'home@index');
Route::get('welcome', array('after' => 'log', 'uses' => 'home@index'));
Route::get('welcome', array('as' => 'home.welcome', 'uses' => 'home@index'));
You may test your routes using Laravel's "Artisan" CLI. Simple specify the request method and URI you want to use. The route response will be var_dump'd back to the CLI.
php artisan route:call get api/user/1