Requests and Routing
Mantle provides a MVC framework on-top of WordPress. You can add a route
fluently and send a response straight back without needing to work with
WordPress's add_rewrite_rule()
at all.
Route::get( '/example-route', function() {
return 'Welcome!';
} );
Route::get( '/hello/{who}', function( $name ) {
return "Welcome {$name}!";
} );
By default, web routes are defined in the routes/web.php
file and REST API routes are defined in the routes/rest-api.php
file. Routes are controlled by
application's Route_Service_Provider
located in the application.
Registering Routes
Routes are registered for the application in the routes/
folder of the
application. Underneath all of it, routes are a wrapper on-top of Symfony
routing with a fluent-interface
on top.
Closure Routes
At its most basic level, routes can be a simple anonymous function.
Route::get( '/endpoint', function() {
return 'Hello!';
} );
Controller Routes
You can use a controller to handle routes as well. In the future, resource and automatic controller routing will be added.
Route::get( '/controller-endpoint', Controller_Class::class . '@method_to_invoke' );
Route::get( '/controller-endpoint', [ Controller_Class::class, 'method_to_invoke' ] );
Generating a Controller
A controller can be generated through the CLI.
bin/mantle make:controller <name>
Invokable Controllers
A single-method controller is supported by defining a controller with an __invoke
method.
use Mantle\Http\Controller;
class Invokable_Controller extends Controller {
/**
* Method to run.
*/
public function __invoke() {
return [ ... ];
}
}
Invokable controllers can be registered by passing the controller class name to the router.
Route::get( '/example-route', Invokable_Controller::class );
Invokable controllers are also generate-able through the CLI.
wp mantle make:controller --invokable
Named Routes
Naming a route provides an easy-to-reference way of generating URLs for a route.
Route::get( '/post/{slug}', function() {
//
} )->name( 'route-name' );
Routes can also pass the name to the router through as an array.
Route::get( '/posts/{slug}', [
'name' => 'named-route',
'callback' => function() { ... },
] );
Generating URLs to Named Routes
Once a route has a name assigned to it, you may use the route's name when
generating URLs or redirects via the helper route
function.
$url = route( 'route-name' );
Route Middleware
Middleware can be used to filter incoming requests and the response sent to the browser. Think of it like a WordPress filter on top of the request and the end response.
Example Middleware
/**
* Example_Middleware class file.
*
* @package Mantle
*/
namespace App\Middleware;
use Closure;
use Mantle\Http\Request;
/**
* Example Middleware
*/
class Example_Middleware {
/**
* Handle the request.
*
* @param Request $request Request object.
* @param Closure $next Callback to proceed.
* @return \Mantle\Http\Response
*/
public function handle( Request $request, Closure $next ) {
// Modify the request or bail early.
$request->setMethod( 'POST' );
/**
* @var Mantle\Http\Response
*/
$response = $next( $request );
// Modify the response.
$response->headers->set( 'Special-Header', 'Value' );
return $response;
}
}
Authentication Middleware
Included with Mantle, a route can check a user's capability before allowing them to view a page.
use Mantle\Facade\Route;
Route::get('/route-to-protect', function() {
// The current user can 'manage_options'.
} )->middleware( 'can:manage_options', Example_Middleware::class );
Removing Middleware
Middleware can be removed from a route by using the without_middleware
method.
You can pass a single middleware, an array of middleware to remove, or remove
all middleware.
use Mantle\Facade\Route;
Route::get( '/route', function() {
// ...
} )->without_middleware( 'middleware_name' );
Once common use case is to remove the wrap template middleware (which will wrap
your response in a WordPress header/footer). You can use the
without_wrap_template()
method.
use Mantle\Facade\Route;
Route::get( '/route', function() {
// ...
} )->without_wrap_template();
Route Prefix
Routes can be prefixed to make it easier to group routes together.
Route::prefix( 'prefix/to/use' )->group( function() {
// Register a route with a prefix here!
} );
Available Router Methods
The router has all HTTP request methods available:
Route::get( $uri, $callback );
Route::post( $uri, $callback );
Route::put( $uri, $callback );
Route::patch( $uri, $callback );
Route::delete( $uri, $callback );
Route::options( $uri, $callback );
Requests Pass-Through to WordPress Routing
By default, requests will pass down to WordPress if there is no match in Mantle.
That can be changed inside of Route_Service_Provider
. If the request doesn't
have a match, the request will 404 and terminate before going through
WordPress' require rules. REST API requests will always pass through to
WordPress and bypass Mantle routing.
/**
* Route_Service_Provider class file.
*
* @package Mantle
*/
namespace App\Providers;
use Mantle\Facade\Request;
use Mantle\Framework\Providers\Route_Service_Provider as Service_Provider;
/**
* Route Service Provider
*/
class Route_Service_Provider extends Service_Provider {
/**
* Bootstrap any application services.
*/
public function boot() {
parent::boot();
$this->allow_pass_through_requests();
}
}
Model Routing
Models can have their permalinks and routing handled automatically. The underlying WordPress object will use the Mantle-generated URL for the model, too. The application will generate an archive route for post models and singular routes for post and term models.
Models that are automatically registered will have their routing automatically registered. Read more about model registration here.
Model Routing Controller
Similar to a resource controller, the router will invoke a single controller for the 'resource' (the model).
Method | Callback |
---|---|
Archive /product/{slug}/ | Product_Controller::index() |
Singular /product/{slug}/ | Product_Controller::show() |
Registering a Model for Routing
Route models can be registered like any other route and also supports prefixes, middleware, etc.
Route::model( Product::class, Product_Controller::class );
In the above example the Product
model will have /product/{slug}
and
/product
routes registered.
use Mantle\Database\Model\Post;
use Mantle\Database\Model\Concerns\Custom_Post_Permalink;
class Product extends Post {
use Custom_Post_Permalink;
public function get_route(): ?string {
return '/route/{slug}';
}
}
Route Model Binding
Routes support model binding that will automatically resolve a model based on a route parameter and a type-hint on the route method. This supports implicit and explicit binding from a service provider.
Implicit Binding
Mantle will automatically resolve models that are type-hinted for the route's method. For example:
Route::get( 'users/{user}', function ( App\User $user ) {
return $user->title;
} );
Since the $user
variable is type-hinted as App\User
model and the variable
name matches the {user}
segment, Mantle will automatically inject the model
instance that has an ID matching the corresponding value from the request URI.
If a matching model instance is not found in the database, a 404 HTTP response
will automatically be generated.
Customizing The Default Key Name
If you would like model binding to use a default database column other than id
when retrieving a given model class, you may override the get_route_key_name
method on the model:
/**
* Get the route key for the model.
*
* @return string
*/
public function get_route_key_name(): string {
return 'slug';
}
Explicit Binding
To register an explicit binding, use the router's model method to specify the
class for a given parameter. You should define your explicit model bindings in
the boot method of the Route_Service_Provider
class:
public function boot() {
parent::boot();
Route::bind_model( 'user', App\User::class );
}
Next, define a route that contains a {user}
parameter:
Route::get( 'profile/{user}', function ( App\User $user ) {
//
} );
Since we have bound all {user}
parameters to the App\User
model, a User
instance will be injected into the route. So, for example, a request to
profile/1
will inject the User
instance from the database which has an ID of
1
.
If a matching model instance is not found in the database, a 404 HTTP response will be automatically generated.
Customizing The Resolution Logic
If you wish to use your own resolution logic, you may use the Route::bind method. The Closure you pass to the bind method will receive the value of the URI segment and should return the instance of the class that should be injected into the route:
/**
* Bootstrap any application services.
*
* @return void
*/
public function boot() {
parent::boot();
Route::bind( 'user', function ( $value ) {
return App\User::where( 'name', $value )->firstOrFail();
} );
}
Alternatively, you may override the resolve_route_binding
method on your
model. This method will receive the value of the URI segment and should return
the instance of the class that should be injected into the route:
/**
* Retrieve the model for a bound value.
*
* @param mixed $value
* @param string|null $field
* @return static|null
*/
public function resolve_route_binding( $value, $field = null ) {
return $this->where( 'name', $value )->firstOrFail();
}
Responses
Responses for routed requests can come in all shapes and sizes. Underneath all
of it, the response will always come out to be a
Symfony\Component\HttpFoundation\Response
object. The response()
helper
exists to help with returning responses.
Strings & Arrays
use Mantle\Facade\Route;
Route::get( '/', function () {
return 'Hello World';
} );
Route::get( '/', function () {
return [ 1, 2, 3 ];
} );
Views
WordPress template parts can be returned for a route.
Route::get( '/', function () {
return response()->view( 'template-parts/block', [ 'variable' => '123' ] );
} );
View Location
By default WordPress will only load a template part from the active theme and parent theme if applicable. Mantle supports loading views from a dynamic set of locations. Mantle support automatically register the current theme and parent theme as view locations.
Default View Locations
- Active Theme
- Parent of Active Theme
{root of mantle site}/views
For more information about views, read the 'Templating' documentation.
Redirect to Endpoint and Route
Redirects can be generated using the response()
helper.
use Mantle\Facade\Route;
Route::get( '/logout', function() {
return response()->redirect_to( '/home' );
} );
Route::get( '/old-page', function() {
return response()->redirect_to( '/home', 301 );
} );
Route::get( '/oh-no', function() {
return response()->redirect_to_route( 'route_name' );
} );
REST API Routing
Mantle supports registering to the WordPress REST API directly. REST API Routes are registered underneath with native core functions and does not use the Symfony-based routing that web requests pass through.
Registering Routes
Registering a REST API route requires a different function call if you do not wish to use a closure.
use Mantle\Facade\Route;
Route::rest_api( 'namespace/v1', '/route-to-use', function() {
return [ 1, 2, 3 ];
} );
Routes can also be registered using the same HTTP verbs web routes use with some minor differences.
use Mantle\Facade\Route;
use WP_REST_Request;
Route::rest_api(
'namespace/v1',
function() {
Route::get(
'/example-group-get',
function() {
return [ 1, 2, 3 ];
}
);
Route::get(
'/example-with-param/(?P<slug>[a-z\-]+)',
function( WP_REST_Request $request) {
return $request['slug'];
}
);
}
);
REST API routes can also be registered using the same arguments you would pass
to register_rest_route()
.
use Mantle\Facade\Route;
Route::rest_api(
'namespace/v1',
function() {
Route::get(
'/example-endpoint',
function() {
// This callback can be omitted, too.
},
[
'permission_callback' => function() {
// ...
}
]
);
}
);
Using Controllers
REST API routes can also use controllers to handle the request. You can use a specific method on a controller or make the controller invokeable to handle the request.
use Mantle\Facade\Route;
use WP_REST_Request;
Route::rest_api(
'namespace/v1',
function() {
Route::get( '/example-endpoint', Example_Controller::class );
Route::get( '/another-endpoint', [ Another_Example_Controller::class, 'method' ] );
}
);
class Example_Controller {
public function __invoke( WP_REST_Request $request ) {
return [ 1, 2, 3 ];
}
}
class Another_Example_Controller {
public function method( WP_REST_Request $request ) {
return [ 1, 2, 3 ];
}
}
Using Middleware
REST API routes also support the same Route Middleware that
web requests use. The only difference is that web requests are passed a Mantle
Request object while REST API requests are passed a WP_REST_Request
one.
use Mantle\Facade\Route;
Route::middleware( Example_Middleware::class )->group(
function() {
Route::rest_api( 'namespace/v1', '/example-route', function() { /* ... */ } );
}
)
Events
Mantle-powered routes for both web and REST API will fire the Route_Matched
event when a route has been matched. It will always include the current request
object. For web routes it will include the Route
object as the route property.
For REST API requests it will include an array of information about the current
matched route.
use Mantle\Http\Routing\Events\Route_Matched;
Event::listen(
Route_Matched::class,
function( Route_Matched $event ) {
var_dump( $event->route );
var_dump( $event->request->ip() );
}
);
New Relic
Using the Route_Matched
event Mantle will automatically fill in transaction
information for New Relic if the extension is loaded. All requests will
have the transaction name properly formatted instead of relying on New Relic to
fill in the blanks.