Skip to content

Middleware

Middleware is simply code that can run either before or after a request is processed. It can be used to modify the request or response, or to perform some other action, such as rate-limiting, logging, or authentication.

Writing Middleware

Middleware are created as invokable classes that implement the __invoke method. The __invoke method receives the request, response, and the next middleware in the pipeline as arguments. The middleware can then modify the request or response, and optionally call the next middleware in the pipeline.

Here is an example of a simple middleware that checks if a user is authenticated:

class LoggerMiddleware
{
    public function __invoke($request, $response, $next)
    {
        // Check if the user is authenticated
        if (!Auth::check()) {
            return response()
                ->withStatus(401);
        }

        // Call the next middleware in the pipeline
        return $next($request, $response);
    }
}

As seen the middleware chain can be broken by returning a response object from the middleware. This is useful for implementing authentication, rate limiting, or other checks that should prevent the request from being processed.

Your actions can happen either before the next middleware is called, or after. For example, you could log the request before the next middleware is called, or log the response after the next middleware is called.

class LoggerMiddleware
{
    public function __invoke($request, $response, $next)
    {
        // Log the request before the request is processed
        Log::info('Request received: ' . $request->url());

        // Call the next middleware in the pipeline
        $response = $next($request, $response);

        // Log the response after the request is processed
        Log::info('Response sent: ' . $response->status());

        return $response;
    }
}

Registering Middleware

Middleware can be registered in the config/middleware.php file. Middleware is grouped together under an alias. The config class must contain a key with the group name that contains an array of middleware classes that should be run on that request. The order of the middleware in the array is the order in which they will be run.

return [
    // The 'web' group
    'web' => [
        App\Middleware\LoggerMiddleware::class,
        App\Middleware\AuthMiddleware::class,
    ],
];

A default group named web is provided that is run on every request. You can create additional groups by adding a new key to the config file.

return [
    // A 'web' group
    'web' => [
        App\Middleware\LoggerMiddleware::class,
        App\Middleware\AuthMiddleware::class,
    ],

    // An 'api' group
    'api' => [
        App\Middleware\RateLimitMiddleware::class,
    ],
];

Specifying Middleware

By default, the web group is run on every request. You can specify which group of middleware should be run on a route by using the middleware method on the route's control file, if one exists. It takes HTTP request verb as the only argument.

The middleware method should return the name of the middleware group that should be run on the request, or an array of middleware classes that should be run on the request.

// routes/products.control.php
<?php

return new class()
{
    public function get()
    {
        //
    }

    public function post()
    {
        //
    }

    public function middleware(string $method)
    {
        return match ($method) {
            'post' => [
                ...config('middleware.web'),
                App\Middleware\LoggerMiddleware::class,
                App\Middleware\AuthMiddleware::class,
            ],
            default => 'web',
        };
    }
}

Skeleton File

The following is a skeleton middleware file that can be copied and pasted into your project:

<?php

namespace App\Middleware;

class SkeletonMiddleware
{
    public function __invoke($request, $response, $next)
    {
        //

        return $next($request, $response);
    }
}