Laravel add additional attributes to request from middleware and access it from elsewhere

Sometimes, you might want to attach additional attributes to the original request that has made to the Laravel application.

For instance, you want to put your own check in the middleware to see if the request has some headers or not, and pass that 'check' by attaching it as an additional attribute to the request.

This way you can also prevent duplicating code, where you would repeat the same code (or query) in the middleware and in the controller.

For instance take following. If I want to get a header value like x-customer-email from the api consumer means, I would write the following code.

1. CheckCustomerHeaderMiddleware.php

public function handle($request, Closure $next) { $customerEmailHeader = 'x-customer-email'; //name of the header we are expecting

if (!$request->headers->has($customerEmailHeader)) { //checking if the header exists in the request

return response()->json(['error' => "Please set '{$customerEmailHeader}' header to consume this api"], 400); }

$customerEmail = trim($request->header($customerEmailHeader, '')); //get the email from the request

$account = Account::where(['email' => trim($customerEmail)])->first(); //find the account

if (!$account->exists()) { //check if the account exists

return response()->json(['error' => "Acount not exist"]); //return error message if they don't exist }

return $next($request); }

2. AccountController.php

public function getAccountDetails(Request $request) { $customerEmailHeader = 'x-customer-email'; //name of the header we are expecting

$customerEmail = trim($request->header($customerEmailHeader, '')); //get the email from the request

$account = Account::where(['email' => trim($customerEmail)])->first(); //find the account

return response()->json($account); }

As you can see, i repeated my code to get the email header value and even performed the fetch query twice to get the account.

This is simply the waste of resources, and might cause issues in the future if we changed the logic in one place and forgot to change it in the other, correct?

Magic of merge method

You can re-factor the above code by calling merge method on the request and passing the $account from the middleware like so.

1. CheckCustomerHeaderMiddleware.php

public function handle($request, Closure $next) { $customerEmailHeader = 'x-customer-email'; //name of the header we are expecting

if (!$request->headers->has($customerEmailHeader)) { //checking if the header exists in the request

return response()->json(['error' => "Please set '{$customerEmailHeader}' header to consume this api"], 400); }

$customerEmail = trim($request->header($customerEmailHeader, '')); //get the email from the request

$account = Account::where(['email' => trim($customerEmail)])->first(); //find the account

if (!$account->exists()) { //check if the account exists

return response()->json(['error' => "Acount not exist"]); //return error message if they don't exist }

//attach the account attribute onto the request $request->merge(['account' => $account]);

return $next($request); }

2. AccountController.php

public function getAccountDetails(Request $request) { $account = $request->account; // just get the account from request

return response()->json($account); }

The benefit of above is now we only need to execute the single query and pass it onto the request which can be re-utilized elsewhere (in our case a controller).