From 710fa56840d299a74bd05f30d27b79dddfc221e6 Mon Sep 17 00:00:00 2001 From: Tim MacDonald Date: Mon, 23 Oct 2023 12:16:30 +1100 Subject: [PATCH 1/4] Improve slow outgoing request performance --- .../2023_06_07_000001_create_pulse_tables.php | 5 ++++- src/Queries/SlowOutgoingRequests.php | 17 +++++++++++++---- src/Recorders/OutgoingRequests.php | 1 + .../Livewire/SlowOutgoingRequestsTest.php | 8 ++++---- 4 files changed, 22 insertions(+), 9 deletions(-) diff --git a/database/migrations/2023_06_07_000001_create_pulse_tables.php b/database/migrations/2023_06_07_000001_create_pulse_tables.php index 862a7870..aca06198 100644 --- a/database/migrations/2023_06_07_000001_create_pulse_tables.php +++ b/database/migrations/2023_06_07_000001_create_pulse_tables.php @@ -106,8 +106,11 @@ public function up(): void $table->text('uri'); $table->char('uri_hash', 16)->charset('binary')->virtualAs('UNHEX(MD5(`uri`))'); $table->unsignedInteger('duration'); + $table->boolean('slow'); - $table->index(['uri_hash', 'date', 'duration']); + $table->index(['date']); // trim + $table->index(['uri_hash']); // slow_outgoing_requests + $table->index(['slow', 'date', 'uri_hash', 'duration']); // slow_outgoing_requests }); } diff --git a/src/Queries/SlowOutgoingRequests.php b/src/Queries/SlowOutgoingRequests.php index e3a89750..e2bd8a41 100644 --- a/src/Queries/SlowOutgoingRequests.php +++ b/src/Queries/SlowOutgoingRequests.php @@ -6,6 +6,7 @@ use Carbon\CarbonInterval as Interval; use Illuminate\Config\Repository; use Illuminate\Database\DatabaseManager; +use Illuminate\Database\Query\Builder; use Illuminate\Support\Collection; use Laravel\Pulse\Recorders\OutgoingRequests; @@ -35,13 +36,21 @@ public function __invoke(Interval $interval): Collection { $now = new CarbonImmutable; - return $this->connection()->table('pulse_outgoing_requests') - ->selectRaw('MAX(`uri`) AS `uri`, COUNT(*) AS `count`, MAX(`duration`) AS `slowest`') + return $this->connection()->query()->select([ + 'count', + 'slowest', + 'uri' => fn (Builder $query) => $query->select('uri') + ->from('pulse_outgoing_requests', as: 'child') + ->whereRaw('`child`.`uri_hash` = `parent`.`uri_hash`') + ->limit(1), + ])->fromSub(fn (Builder $query) => $query->selectRaw('`uri_hash`, MAX(`duration`) as `slowest`, COUNT(*) as `count`') + ->from('pulse_outgoing_requests') + ->where('slow', true) ->where('date', '>', $now->subSeconds((int) $interval->totalSeconds)->toDateTimeString()) - ->where('duration', '>=', $this->config->get('pulse.recorders.'.OutgoingRequests::class.'.threshold')) ->groupBy('uri_hash') ->orderByDesc('slowest') - ->limit(101) + ->orderByDesc('count') + ->limit(101), as: 'parent') ->get(); } } diff --git a/src/Recorders/OutgoingRequests.php b/src/Recorders/OutgoingRequests.php index a1f2f68c..cd8b3bfe 100644 --- a/src/Recorders/OutgoingRequests.php +++ b/src/Recorders/OutgoingRequests.php @@ -65,6 +65,7 @@ public function record(RequestInterface $request, CarbonImmutable $startedAt): ? 'date' => $startedAt->toDateTimeString(), 'duration' => $startedAt->diffInMilliseconds($endedAt), 'user_id' => $this->pulse->authenticatedUserIdResolver(), + 'slow' => $duration >= $this->config->get('pulse.recorders.'.self::class.'.threshold'), ]); } diff --git a/tests/Feature/Livewire/SlowOutgoingRequestsTest.php b/tests/Feature/Livewire/SlowOutgoingRequestsTest.php index 2b75eb97..f2b01a57 100644 --- a/tests/Feature/Livewire/SlowOutgoingRequestsTest.php +++ b/tests/Feature/Livewire/SlowOutgoingRequestsTest.php @@ -16,10 +16,10 @@ it('renders slow outgoing requests', function () { Pulse::ignore(fn () => DB::table('pulse_outgoing_requests')->insert([ - ['date' => '2000-01-02 03:04:05', 'uri' => 'GET http://example.com', 'duration' => 1234], - ['date' => '2000-01-02 03:04:05', 'uri' => 'GET http://example.com', 'duration' => 2468], - ['date' => '2000-01-02 03:04:05', 'uri' => 'GET http://example.org', 'duration' => 123], - ['date' => '2000-01-02 03:04:05', 'uri' => 'GET http://example.org', 'duration' => 1234], + ['date' => '2000-01-02 03:04:05', 'uri' => 'GET http://example.com', 'duration' => 1234, 'slow' => true], + ['date' => '2000-01-02 03:04:05', 'uri' => 'GET http://example.com', 'duration' => 2468, 'slow' => true], + ['date' => '2000-01-02 03:04:05', 'uri' => 'GET http://example.org', 'duration' => 123, 'slow' => false], + ['date' => '2000-01-02 03:04:05', 'uri' => 'GET http://example.org', 'duration' => 1234, 'slow' => true], ])); Carbon::setTestNow('2000-01-02 03:04:15'); From 8cc8accbe057c506aecf48a93d538917b24fa999 Mon Sep 17 00:00:00 2001 From: Tim MacDonald Date: Mon, 23 Oct 2023 12:16:41 +1100 Subject: [PATCH 2/4] Fix limit --- src/Queries/SlowRoutes.php | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Queries/SlowRoutes.php b/src/Queries/SlowRoutes.php index 64bd070d..5af025fe 100644 --- a/src/Queries/SlowRoutes.php +++ b/src/Queries/SlowRoutes.php @@ -57,7 +57,7 @@ public function __invoke(Interval $interval): Collection ->groupBy('route_hash') ->orderByDesc('slowest') ->orderByDesc('count') - ->limit(100), as: 'parent') + ->limit(101), as: 'parent') ->get() ->map(fn (stdClass $row) => (object) [ 'route' => (string) $row->route, From a4e99c4b56c645771ff4b06c7240aa489f7bf21f Mon Sep 17 00:00:00 2001 From: timacdonald Date: Mon, 23 Oct 2023 01:17:13 +0000 Subject: [PATCH 3/4] Fix code styling --- src/Queries/SlowOutgoingRequests.php | 1 - 1 file changed, 1 deletion(-) diff --git a/src/Queries/SlowOutgoingRequests.php b/src/Queries/SlowOutgoingRequests.php index e2bd8a41..1e4c3d42 100644 --- a/src/Queries/SlowOutgoingRequests.php +++ b/src/Queries/SlowOutgoingRequests.php @@ -8,7 +8,6 @@ use Illuminate\Database\DatabaseManager; use Illuminate\Database\Query\Builder; use Illuminate\Support\Collection; -use Laravel\Pulse\Recorders\OutgoingRequests; /** * @internal From a7920275ce94ad8ec0f8a7c86bfba1ac17ca9a77 Mon Sep 17 00:00:00 2001 From: Tim MacDonald Date: Mon, 23 Oct 2023 12:19:30 +1100 Subject: [PATCH 4/4] Fix config --- src/Recorders/OutgoingRequests.php | 4 ++-- src/Recorders/Requests.php | 2 +- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/src/Recorders/OutgoingRequests.php b/src/Recorders/OutgoingRequests.php index cd8b3bfe..204fe32a 100644 --- a/src/Recorders/OutgoingRequests.php +++ b/src/Recorders/OutgoingRequests.php @@ -63,9 +63,9 @@ public function record(RequestInterface $request, CarbonImmutable $startedAt): ? return new Entry($this->table, [ 'uri' => $this->normalizeUri($request), 'date' => $startedAt->toDateTimeString(), - 'duration' => $startedAt->diffInMilliseconds($endedAt), + 'duration' => $duration = $startedAt->diffInMilliseconds($endedAt), 'user_id' => $this->pulse->authenticatedUserIdResolver(), - 'slow' => $duration >= $this->config->get('pulse.recorders.'.self::class.'.threshold'), + 'slow' => $duration >= $this->config->get('pulse.recorders.'.static::class.'.threshold'), ]); } diff --git a/src/Recorders/Requests.php b/src/Recorders/Requests.php index aba1df4f..70d810ce 100644 --- a/src/Recorders/Requests.php +++ b/src/Recorders/Requests.php @@ -66,7 +66,7 @@ public function record(Carbon $startedAt, Request $request, Response $response): 'route' => $request->method().' '.$path, 'duration' => $duration = $startedAt->diffInMilliseconds(), 'user_id' => $this->pulse->authenticatedUserIdResolver(), - 'slow' => $duration >= $this->config->get('pulse.recorders.'.self::class.'.threshold'), + 'slow' => $duration >= $this->config->get('pulse.recorders.'.static::class.'.threshold'), ]); }