September 4, 2024
When you're deploying with services like Laravel Vapor, paired with AWS services like CloudWatch, you have powerful tools at your disposal to gain deep insights into your application's events and behaviors.
You get an email from the client, "We've received a report that a user cannot reset their password. The password reset email link isn't working. They click the link, but nothing happens."
First stop: the logs. You sift through them, expecting to find a red flag or exception. But instead, it's business as usual—no errors, no anomalies.
Next, you dive into the database, zeroing in on the password_reset_tokens
table. Sure enough, you find fresh entries—evidence that password reset requests are being processed as expected.
Then, we look at SES. The emails are going out and the users are receiving them, so the email delivery isn’t the culprit.
Curious about the email content - what did that link look like? Well, that's where things get tricky. Since the emails are sent via SES, there's no record of the content on your end. You're flying blind.
Returning to the database, we see the reset record but now the password reset token is expired, having been generated days ago. So we cannot replicate the issue.
You're left scratching your head, "What could be the issue? Did the user wait too long? Was it their email client? Did they even click the link?
This is a common scenario when you're debugging issues in production. You have the logs, but they don't provide enough context to understand the issue.
"If only I could see what the user did...", how many times have you said that as a developer?
Laravel has many default events that fire during the application's lifecycle. These events are the perfect opportunity to gain insights into your application's behavior. By logging these events, you can see what's happening in your application in real-time.
Let's look at a few:
Illuminate\Auth\Events\Login
- fired when a user logs inIlluminate\Auth\Events\Logout
- fired when a user logs outIlluminate\Auth\Events\PasswordReset
- fired when a user resets their passwordIlluminate\Auth\Events\Registered
- fired when a user registersIlluminate\Auth\Events\Attempting
- fired when a user attempts to log inIlluminate\Foundation\Http\Events\RequestHandled
- fired when a request is handled by the applicationIlluminate\Database\Events\QueryExecuted
- fired when a database query is executedThese events provide lots of context about what's happening in your application. By logging them, you can gain insights into your application's performance and behavior.
Laravel makes it easy to log events using the built-in event system. You can create listeners for events and log them to your preferred logging service.
The easiest way to add these listeners is in your AppServiceProvider.php
. Here's an example of how you can log the Login
event:
1namespace App\Providers; 2 3use Illuminate\Auth\Events\Login; 4use Illuminate\Support\Facades\Event; 5use Illuminate\Support\Facades\Log; 6use Illuminate\Support\ServiceProvider; 7 8class AppServiceProvider extends ServiceProvider 9{10 /**11 * Bootstrap any application services.12 */13 public function boot(): void14 {15 Event::listen(Login::class, function (Login $event) {16 Log::info('Successful Authentication', [ 17 'user_id' => $event->user->id, 18 'email' => $event->user->email, 19 ]); 20 }); 21 }22}
In this example, we're listening for the Login
event and logging the user's ID and email address.
Which in return, the log will look like this:
1[2024-08-22 09:00:00] local.INFO: Successful Authentication {"user_id":1,"email":"test@example.com"}
This looks okay, but I like to prefix my logs with a group name to make it easier to filter and it's consistent across all authentication-related logs.
1Log::info('[AUTH] Successful Authentication', [ 2 'user_id' => $event->user->id,3 'email' => $event->user->email,4]);
In Laravel 11, Laravel core team member Tim MacDonald introduced Context to logging. This feature allows you to add context to your logs, making it easier to filter and search for specific logs.
In the example above, I added context to the log message with the second parameter of the Log::info
method. This context is an array of additional data that you want to include in the log message.
The real magic happens when you use the Log::withContext
method. This method allows you to add context to all log messages within a given closure.
1Log::shareContext([2 'email' => $request->user()?->email,3 'ip' => $request->ip(),4 'via' => $request->expectsJson() ? 'api' : 'web',5 'user_agent' => $request->userAgent(),6]);
In this example, we're sharing context with all log messages within the closure. This means that the email
, ip
, via
, and user_agent
fields will be added to all log messages within the closure.
Now we can remove the context in the middle of the code from the log message and let Laravel handle it for us. This allows us to have consistent context across all log messages.
1Log::info('[AUTH] Successful Authentication');
The log message will now look like this:
1[2024-08-22 09:00:00] local.INFO: [AUTH] Successful Authentication {"email":" foo@bar.com","ip":"xxx.xx.xxx.xx","via":"web","user_agent":"Mozilla/5.0"}
You can add the `sharedContext` to all log messages by creating a middleware that adds the context to the request:
1namespace App\Http\Middleware; 2 3use Closure; 4use Illuminate\Http\Request; 5use Illuminate\Support\Facades\Log; 6use Symfony\Component\HttpFoundation\Response; 7 8class CloudWatchContextMiddleware 9{10 /**11 * Handle an incoming request.12 *13 * @param \Closure(\Illuminate\Http\Request): (\Symfony\Component\HttpFoundation\Response) $next14 */15 public function handle(Request $request, Closure $next): Response 16 { 17 Log::shareContext([ 18 'email' => $request->user()?->email, 19 'ip' => $request->ip(), 20 'via' => $request->expectsJson() ? 'api' : 'web', 21 'user_agent' => $request->userAgent(), 22 ]); 23 24 return $next($request); 25 } 26}
In this example, we're adding the context to the request in the middleware. This context will be shared with all log messages within the request lifecycle.
You can add this middleware to the web
middleware group in your bootstrap/app.php
file:
1// use App\Http\Middleware\CloudWatchContextMiddleware; 2 3return Application::configure(basePath: dirname(__DIR__)) 4 ->withProviders() 5 ->withRouting( 6 web: __DIR__.'/../routes/web.php', 7 commands: __DIR__.'/../routes/console.php', 8 health: '/status', 9 )10 ->withMiddleware(function (Middleware $middleware) { 11 $middleware->web([ 12 CloudWatchContextMiddleware::class, 13 ]); 14 })15 ->withExceptions(function (Exceptions $exceptions) {16 Integration::handles($exceptions);17 })->create();
Ensure that you add it to the web
middleware group to share authentication context with all web requests. Otherwise, at a base middleware level, it will miss user identifying context like email or user ID.
Laravel Vapor makes it easy to log to AWS CloudWatch. Laravel Vapor uses CloudWatch by default to log your application's output. If you are outside of Vapor, you can use the laravel-cloudwatch-logs package to log to CloudWatch.
To log to CloudWatch, you may need to set up the necessary permissions in your AWS account and configure the package with your AWS credentials.
Once you're logging to CloudWatch, you can use the CloudWatch Log Insights feature to query and analyze your logs. Log Insights allows you to run queries on your logs to gain insights into your application's behavior.
For example, you can use Log Insights to query all successful authentication logs and see how many users are logging in each day. This can help you identify patterns and trends in your application's usage.
In this example, we're querying the logs for successful authentications and grouping them by the method of authentication (web or API) and the time of day.
1fields @timestamp, @message2| filter @message like "[AUTH] Successful Authentication"3| fields coalesce(context.via, "unknown") as via4| parse via "web" as web_request5| parse via "api" as api_request6| stats count(*) as Total_Authentications, count(web_request) as Web_Authentications, count(api_request) as API_Authentications by bin(15m)
Once you have your queries set up, you can create a dashboard in CloudWatch to visualize your logs. Dashboards allow you to create custom visualizations of your log data, such as line charts, bar charts, and pie charts.
From a saved query, you can use the "Add to dashboard" button to create a new widget on your dashboard. You can then customize the widget to display the data in the way you want. Just ensure the data is in the correct format for the given visualization.
That's it! You're now deeply monitoring your Laravel events with AWS CloudWatch. By logging your events and analyzing them with CloudWatch, you can gain insights into your application's performance and behavior.
Feel free to add more events to your logging system to gain even more insights into your application. The more events you log, the more context you'll have about what's happening in your application.
AWS CloudWatch is affordable with a generous free tier, so you can get started with logging your events without breaking the bank. Give it a try and see how it can help you monitor your Laravel application.
Happy logging!