Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit 4c03331

Browse files
committedFeb 23, 2024
Rethink the pulse:check command
1 parent bfc4bb3 commit 4c03331

File tree

8 files changed

+306
-64
lines changed

8 files changed

+306
-64
lines changed
 

‎src/Commands/CheckCommand.php

+16-17
Original file line numberDiff line numberDiff line change
@@ -3,11 +3,12 @@
33
namespace Laravel\Pulse\Commands;
44

55
use Carbon\CarbonImmutable;
6-
use Carbon\CarbonInterval;
76
use Illuminate\Console\Command;
87
use Illuminate\Contracts\Cache\LockProvider;
98
use Illuminate\Events\Dispatcher;
9+
use Illuminate\Support\Env;
1010
use Illuminate\Support\Sleep;
11+
use Illuminate\Support\Str;
1112
use Laravel\Pulse\Events\IsolatedBeat;
1213
use Laravel\Pulse\Events\SharedBeat;
1314
use Laravel\Pulse\Pulse;
@@ -25,7 +26,7 @@ class CheckCommand extends Command
2526
*
2627
* @var string
2728
*/
28-
public $signature = 'pulse:check';
29+
public $signature = 'pulse:check {--once : Take a single snapshot}';
2930

3031
/**
3132
* The command's description.
@@ -42,38 +43,36 @@ public function handle(
4243
CacheStoreResolver $cache,
4344
Dispatcher $event,
4445
): int {
45-
$lastRestart = $cache->store()->get('laravel:pulse:restart');
46+
$isVapor = (bool) Env::get('VAPOR_SSM_PATH');
4647

47-
$interval = CarbonInterval::seconds(5);
48+
$instance = $isVapor ? 'vapor' : Str::random();
4849

49-
$lastSnapshotAt = CarbonImmutable::now()->floorSeconds((int) $interval->totalSeconds);
50+
$lastRestart = $cache->store()->get('laravel:pulse:restart');
5051

5152
$lock = ($store = $cache->store()->getStore()) instanceof LockProvider
52-
? $store->lock('laravel:pulse:check', (int) $interval->totalSeconds)
53+
? $store->lock('laravel:pulse:check', 1)
5354
: null;
5455

5556
while (true) {
56-
$now = CarbonImmutable::now();
57-
58-
if ($now->subSeconds((int) $interval->totalSeconds)->lessThan($lastSnapshotAt)) {
59-
Sleep::for(500)->milliseconds();
60-
61-
continue;
62-
}
63-
6457
if ($lastRestart !== $cache->store()->get('laravel:pulse:restart')) {
6558
return self::SUCCESS;
6659
}
6760

68-
$lastSnapshotAt = $now->floorSeconds((int) $interval->totalSeconds);
61+
$now = CarbonImmutable::now();
6962

7063
if ($lock?->get()) {
71-
$event->dispatch(new IsolatedBeat($lastSnapshotAt, $interval));
64+
$event->dispatch(new IsolatedBeat($now));
7265
}
7366

74-
$event->dispatch(new SharedBeat($lastSnapshotAt, $interval));
67+
$event->dispatch(new SharedBeat($now, $instance));
7568

7669
$pulse->ingest();
70+
71+
if ($isVapor || $this->option('once')) {
72+
return self::SUCCESS;
73+
}
74+
75+
Sleep::until($now->addSecond());
7776
}
7877
}
7978
}

‎src/Events/IsolatedBeat.php

-2
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,6 @@
33
namespace Laravel\Pulse\Events;
44

55
use Carbon\CarbonImmutable;
6-
use Carbon\CarbonInterval;
76

87
class IsolatedBeat
98
{
@@ -12,7 +11,6 @@ class IsolatedBeat
1211
*/
1312
public function __construct(
1413
public CarbonImmutable $time,
15-
public CarbonInterval $interval,
1614
) {
1715
//
1816
}

‎src/Events/SharedBeat.php

+1-2
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,6 @@
33
namespace Laravel\Pulse\Events;
44

55
use Carbon\CarbonImmutable;
6-
use Carbon\CarbonInterval;
76

87
class SharedBeat
98
{
@@ -12,7 +11,7 @@ class SharedBeat
1211
*/
1312
public function __construct(
1413
public CarbonImmutable $time,
15-
public CarbonInterval $interval,
14+
public string $instance,
1615
) {
1716
//
1817
}

‎src/Recorders/Concerns/Throttling.php

+42
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,42 @@
1+
<?php
2+
3+
namespace Laravel\Pulse\Recorders\Concerns;
4+
5+
use Carbon\CarbonImmutable;
6+
use DateInterval;
7+
use Illuminate\Support\Facades\App;
8+
use Illuminate\Support\InteractsWithTime;
9+
use Laravel\Pulse\Events\IsolatedBeat;
10+
use Laravel\Pulse\Events\SharedBeat;
11+
use Laravel\Pulse\Support\CacheStoreResolver;
12+
13+
trait Throttling
14+
{
15+
use InteractsWithTime;
16+
17+
/**
18+
* Determine if the recorder is ready to record another snapshot.
19+
*/
20+
protected function throttle(DateInterval|int $interval, SharedBeat|IsolatedBeat $event, callable $callback, ?string $key = null): void
21+
{
22+
$key ??= static::class;
23+
24+
if ($event instanceof SharedBeat) {
25+
$key = $event->instance.":{$key}";
26+
}
27+
28+
$cache = App::make(CacheStoreResolver::class);
29+
30+
$key = 'laravel:pulse:throttle:'.$key;
31+
32+
$lastRunAt = $cache->store()->get($key);
33+
34+
if ($lastRunAt !== null && CarbonImmutable::createFromTimestamp($lastRunAt)->addSeconds($this->secondsUntil($interval))->isFuture()) {
35+
return;
36+
}
37+
38+
$callback($event);
39+
40+
$cache->store()->put($key, $event->time->getTimestamp(), $interval);
41+
}
42+
}

‎src/Recorders/Servers.php

+42-42
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,8 @@
1313
*/
1414
class Servers
1515
{
16+
use Concerns\Throttling;
17+
1618
/**
1719
* The events to listen for.
1820
*
@@ -35,51 +37,49 @@ public function __construct(
3537
*/
3638
public function record(SharedBeat $event): void
3739
{
38-
if ($event->time->second % 15 !== 0) {
39-
return;
40-
}
41-
42-
$server = $this->config->get('pulse.recorders.'.self::class.'.server_name');
43-
$slug = Str::slug($server);
40+
$this->throttle(15, $event, function ($event) {
41+
$server = $this->config->get('pulse.recorders.'.self::class.'.server_name');
42+
$slug = Str::slug($server);
4443

45-
$memoryTotal = match (PHP_OS_FAMILY) {
46-
'Darwin' => intval(`sysctl hw.memsize | grep -Eo '[0-9]+'` / 1024 / 1024),
47-
'Linux' => intval(`cat /proc/meminfo | grep MemTotal | grep -E -o '[0-9]+'` / 1024),
48-
'Windows' => intval(((int) trim(`wmic ComputerSystem get TotalPhysicalMemory | more +1`)) / 1024 / 1024),
49-
'BSD' => intval(`sysctl hw.physmem | grep -Eo '[0-9]+'` / 1024 / 1024),
50-
default => throw new RuntimeException('The pulse:check command does not currently support '.PHP_OS_FAMILY),
51-
};
44+
$memoryTotal = match (PHP_OS_FAMILY) {
45+
'Darwin' => intval(`sysctl hw.memsize | grep -Eo '[0-9]+'` / 1024 / 1024),
46+
'Linux' => intval(`cat /proc/meminfo | grep MemTotal | grep -E -o '[0-9]+'` / 1024),
47+
'Windows' => intval(((int) trim(`wmic ComputerSystem get TotalPhysicalMemory | more +1`)) / 1024 / 1024),
48+
'BSD' => intval(`sysctl hw.physmem | grep -Eo '[0-9]+'` / 1024 / 1024),
49+
default => throw new RuntimeException('The pulse:check command does not currently support '.PHP_OS_FAMILY),
50+
};
5251

53-
$memoryUsed = match (PHP_OS_FAMILY) {
54-
'Darwin' => $memoryTotal - intval(intval(`vm_stat | grep 'Pages free' | grep -Eo '[0-9]+'`) * intval(`pagesize`) / 1024 / 1024), // MB
55-
'Linux' => $memoryTotal - intval(`cat /proc/meminfo | grep MemAvailable | grep -E -o '[0-9]+'` / 1024), // MB
56-
'Windows' => $memoryTotal - intval(((int) trim(`wmic OS get FreePhysicalMemory | more +1`)) / 1024), // MB
57-
'BSD' => intval(intval(`( sysctl vm.stats.vm.v_cache_count | grep -Eo '[0-9]+' ; sysctl vm.stats.vm.v_inactive_count | grep -Eo '[0-9]+' ; sysctl vm.stats.vm.v_active_count | grep -Eo '[0-9]+' ) | awk '{s+=$1} END {print s}'`) * intval(`pagesize`) / 1024 / 1024), // MB
58-
default => throw new RuntimeException('The pulse:check command does not currently support '.PHP_OS_FAMILY),
59-
};
52+
$memoryUsed = match (PHP_OS_FAMILY) {
53+
'Darwin' => $memoryTotal - intval(intval(`vm_stat | grep 'Pages free' | grep -Eo '[0-9]+'`) * intval(`pagesize`) / 1024 / 1024), // MB
54+
'Linux' => $memoryTotal - intval(`cat /proc/meminfo | grep MemAvailable | grep -E -o '[0-9]+'` / 1024), // MB
55+
'Windows' => $memoryTotal - intval(((int) trim(`wmic OS get FreePhysicalMemory | more +1`)) / 1024), // MB
56+
'BSD' => intval(intval(`( sysctl vm.stats.vm.v_cache_count | grep -Eo '[0-9]+' ; sysctl vm.stats.vm.v_inactive_count | grep -Eo '[0-9]+' ; sysctl vm.stats.vm.v_active_count | grep -Eo '[0-9]+' ) | awk '{s+=$1} END {print s}'`) * intval(`pagesize`) / 1024 / 1024), // MB
57+
default => throw new RuntimeException('The pulse:check command does not currently support '.PHP_OS_FAMILY),
58+
};
6059

61-
$cpu = match (PHP_OS_FAMILY) {
62-
'Darwin' => (int) `top -l 1 | grep -E "^CPU" | tail -1 | awk '{ print $3 + $5 }'`,
63-
'Linux' => (int) `top -bn1 | grep -E '^(%Cpu|CPU)' | awk '{ print $2 + $4 }'`,
64-
'Windows' => (int) trim(`wmic cpu get loadpercentage | more +1`),
65-
'BSD' => (int) `top -b -d 2| grep 'CPU: ' | tail -1 | awk '{print$10}' | grep -Eo '[0-9]+\.[0-9]+' | awk '{ print 100 - $1 }'`,
66-
default => throw new RuntimeException('The pulse:check command does not currently support '.PHP_OS_FAMILY),
67-
};
60+
$cpu = match (PHP_OS_FAMILY) {
61+
'Darwin' => (int) `top -l 1 | grep -E "^CPU" | tail -1 | awk '{ print $3 + $5 }'`,
62+
'Linux' => (int) `top -bn1 | grep -E '^(%Cpu|CPU)' | awk '{ print $2 + $4 }'`,
63+
'Windows' => (int) trim(`wmic cpu get loadpercentage | more +1`),
64+
'BSD' => (int) `top -b -d 2| grep 'CPU: ' | tail -1 | awk '{print$10}' | grep -Eo '[0-9]+\.[0-9]+' | awk '{ print 100 - $1 }'`,
65+
default => throw new RuntimeException('The pulse:check command does not currently support '.PHP_OS_FAMILY),
66+
};
6867

69-
$this->pulse->record('cpu', $slug, $cpu, $event->time)->avg()->onlyBuckets();
70-
$this->pulse->record('memory', $slug, $memoryUsed, $event->time)->avg()->onlyBuckets();
71-
$this->pulse->set('system', $slug, json_encode([
72-
'name' => $server,
73-
'cpu' => $cpu,
74-
'memory_used' => $memoryUsed,
75-
'memory_total' => $memoryTotal,
76-
'storage' => collect($this->config->get('pulse.recorders.'.self::class.'.directories')) // @phpstan-ignore argument.templateType argument.templateType
77-
->map(fn (string $directory) => [
78-
'directory' => $directory,
79-
'total' => $total = intval(round(disk_total_space($directory) / 1024 / 1024)), // MB
80-
'used' => intval(round($total - (disk_free_space($directory) / 1024 / 1024))), // MB
81-
])
82-
->all(),
83-
], flags: JSON_THROW_ON_ERROR), $event->time);
68+
$this->pulse->record('cpu', $slug, $cpu, $event->time)->avg()->onlyBuckets();
69+
$this->pulse->record('memory', $slug, $memoryUsed, $event->time)->avg()->onlyBuckets();
70+
$this->pulse->set('system', $slug, json_encode([
71+
'name' => $server,
72+
'cpu' => $cpu,
73+
'memory_used' => $memoryUsed,
74+
'memory_total' => $memoryTotal,
75+
'storage' => collect($this->config->get('pulse.recorders.'.self::class.'.directories')) // @phpstan-ignore argument.templateType argument.templateType
76+
->map(fn (string $directory) => [
77+
'directory' => $directory,
78+
'total' => $total = intval(round(disk_total_space($directory) / 1024 / 1024)), // MB
79+
'used' => intval(round($total - (disk_free_space($directory) / 1024 / 1024))), // MB
80+
])
81+
->all(),
82+
], flags: JSON_THROW_ON_ERROR), $event->time);
83+
});
8484
}
8585
}

0 commit comments

Comments
 (0)