Facades
Laravel facades serve as “static proxies” to underlying classes in the service container, providing the benefit of a terse, expressive syntax while maintaining more testability and flexibility than traditional static methods. It’s perfectly fine if you don’t totally understand how facades work - just go with the flow and continue learning about Laravel.
All of Laravel’s facades are defined in the namespace. So, we can easily access a facade like so:
Throughout the Laravel documentation, many of the examples will use facades to demonstrate various features of the framework.
Helper Functions
To complement facades, Laravel offers a variety of global “helper functions” that make it even easier to interact with common Laravel features. Some of the common helper functions you may interact with are view
, response
, url
, config
, and more. Each helper function offered by Laravel is documented with their corresponding feature; however, a complete list is available within the dedicated helper documentation.
For example, instead of using the Illuminate\Support\Facades\Response
facade to generate a JSON response, we may simply use the response
function. Because helper functions are globally available, you do not need to import any classes in order to use them:
Facades have many benefits. They provide a terse, memorable syntax that allows you to use Laravel’s features without remembering long class names that must be injected or configured manually. Furthermore, because of their unique usage of PHP’s dynamic methods, they are easy to test.
One of the primary benefits of dependency injection is the ability to swap implementations of the injected class. This is useful during testing since you can inject a mock or stub and assert that various methods were called on the stub.
Typically, it would not be possible to mock or stub a truly static class method. However, since facades use dynamic methods to proxy method calls to objects resolved from the service container, we actually can test facades just as we would test an injected class instance. For example, given the following route:
use Illuminate\Support\Facades\Cache; Route::get('/cache', function () { return Cache::get('key');});
Using Laravel’s facade testing methods, we can write the following test to verify that the Cache::get
method was called with the argument we expected:
use Illuminate\Support\Facades\Cache; /** * A basic functional test example. */public function test_basic_example(): void{ Cache::shouldReceive('get') ->with('key') ->andReturn('value'); $response = $this->get('/cache'); $response->assertSee('value');}
Facades Vs. Helper Functions
In addition to facades, Laravel includes a variety of “helper” functions which can perform common tasks like generating views, firing events, dispatching jobs, or sending HTTP responses. Many of these helper functions perform the same function as a corresponding facade. For example, this facade call and helper call are equivalent:
There is absolutely no practical difference between facades and helper functions. When using helper functions, you may still test them exactly as you would the corresponding facade. For example, given the following route:
Route::get('/cache', function () { return cache('key');});
The cache
helper is going to call the get
method on the class underlying the Cache
facade. So, even though we are using the helper function, we can write the following test to verify that the method was called with the argument we expected:
use Illuminate\Support\Facades\Cache; /** * A basic functional test example. */public function test_basic_example(): void{ Cache::shouldReceive('get') ->with('key') ->andReturn('value'); $response = $this->get('/cache'); $response->assertSee('value');}
The Facade
base class makes use of the __callStatic()
magic-method to defer calls from your facade to an object resolved from the container. In the example below, a call is made to the Laravel cache system. By glancing at this code, one might assume that the static get
method is being called on the Cache
class:
<?php namespace App\Http\Controllers; use App\Http\Controllers\Controller;use Illuminate\Support\Facades\Cache;use Illuminate\View\View; class UserController extends Controller{ /** * Show the profile for the given user. */ public function showProfile(string $id): View { $user = Cache::get('user:'.$id); return view('profile', ['user' => $user]); }}
Notice that near the top of the file we are “importing” the Cache
facade. This facade serves as a proxy for accessing the underlying implementation of the Illuminate\Contracts\Cache\Factory
interface. Any calls we make using the facade will be passed to the underlying instance of Laravel’s cache service.
If we look at that Illuminate\Support\Facades\Cache
class, you’ll see that there is no static method :
Instead, the Cache
facade extends the base Facade
class and defines the method getFacadeAccessor()
. This method’s job is to return the name of a service container binding. When a user references any static method on the Cache
facade, Laravel resolves the cache
binding from the and runs the requested method (in this case, get
) against that object.
Using real-time facades, you may treat any class in your application as if it was a facade. To illustrate how this can be used, let’s first examine some code that does not use real-time facades. For example, let’s assume our Podcast
model has a publish
method. However, in order to publish the podcast, we need to inject a Publisher
instance:
<?php namespace App\Models; use App\Contracts\Publisher;use Illuminate\Database\Eloquent\Model; class Podcast extends Model{ /** * Publish the podcast. */ public function publish(Publisher $publisher): void { $this->update(['publishing' => now()]); $publisher->publish($this); }}
Injecting a publisher implementation into the method allows us to easily test the method in isolation since we can mock the injected publisher. However, it requires us to always pass a publisher instance each time we call the publish
method. Using real-time facades, we can maintain the same testability while not being required to explicitly pass a Publisher
instance. To generate a real-time facade, prefix the namespace of the imported class with Facades
:
<?php namespace App\Models; use Facades\App\Contracts\Publisher;use Illuminate\Database\Eloquent\Model; class Podcast extends Model{ /** * Publish the podcast. */ public function publish(): void { $this->update(['publishing' => now()]); Publisher::publish($this); }}
Below you will find every facade and its underlying class. This is a useful tool for quickly digging into the API documentation for a given facade root. The service container binding key is also included where applicable.