Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
pterodactyl
GitHub Repository: pterodactyl/panel
Path: blob/1.0-develop/app/Http/Controllers/Api/Remote/ActivityProcessingController.php
10284 views
1
<?php
2
3
namespace Pterodactyl\Http\Controllers\Api\Remote;
4
5
use Carbon\Carbon;
6
use Illuminate\Support\Str;
7
use Pterodactyl\Models\User;
8
use Webmozart\Assert\Assert;
9
use Pterodactyl\Models\Server;
10
use Illuminate\Support\Facades\Log;
11
use Pterodactyl\Models\ActivityLog;
12
use Pterodactyl\Models\ActivityLogSubject;
13
use Pterodactyl\Http\Controllers\Controller;
14
use Pterodactyl\Http\Requests\Api\Remote\ActivityEventRequest;
15
16
class ActivityProcessingController extends Controller
17
{
18
public function __invoke(ActivityEventRequest $request)
19
{
20
$tz = Carbon::now()->getTimezone();
21
22
/** @var \Pterodactyl\Models\Node $node */
23
$node = $request->attributes->get('node');
24
25
$servers = $node->servers()->whereIn('uuid', $request->servers())->get()->keyBy('uuid');
26
$users = User::query()->whereIn('uuid', $request->users())->get()->keyBy('uuid');
27
28
$logs = [];
29
foreach ($request->input('data') as $datum) {
30
/** @var Server|null $server */
31
$server = $servers->get($datum['server']);
32
if (is_null($server) || !Str::startsWith($datum['event'], 'server:')) {
33
continue;
34
}
35
36
try {
37
$when = Carbon::createFromFormat(
38
\DateTimeInterface::RFC3339,
39
preg_replace('/(\.\d+)Z$/', 'Z', $datum['timestamp']),
40
'UTC'
41
);
42
} catch (\Exception $exception) {
43
Log::warning($exception, ['timestamp' => $datum['timestamp']]);
44
45
// If we cannot parse the value for some reason don't blow up this request, just go ahead
46
// and log the event with the current time, and set the metadata value to have the original
47
// timestamp that was provided.
48
$when = Carbon::now();
49
$datum['metadata'] = array_merge($datum['metadata'] ?? [], ['original_timestamp' => $datum['timestamp']]);
50
}
51
52
$log = [
53
'ip' => empty($datum['ip']) ? '127.0.0.1' : $datum['ip'],
54
'event' => $datum['event'],
55
'properties' => json_encode($datum['metadata'] ?? []),
56
// We have to change the time to the current timezone due to the way Laravel is handling
57
// the date casting internally. If we just leave it in UTC it ends up getting double-cast
58
// and the time is way off.
59
'timestamp' => $when->setTimezone($tz),
60
];
61
62
if ($user = $users->get($datum['user'])) {
63
$log['actor_id'] = $user->id;
64
$log['actor_type'] = $user->getMorphClass();
65
}
66
67
if (!isset($logs[$datum['server']])) {
68
$logs[$datum['server']] = [];
69
}
70
71
$logs[$datum['server']][] = $log;
72
}
73
74
foreach ($logs as $key => $data) {
75
Assert::isInstanceOf($server = $servers->get($key), Server::class);
76
77
$batch = [];
78
foreach ($data as $datum) {
79
$id = ActivityLog::insertGetId($datum);
80
$batch[] = [
81
'activity_log_id' => $id,
82
'subject_id' => $server->id,
83
'subject_type' => $server->getMorphClass(),
84
];
85
}
86
87
ActivityLogSubject::insert($batch);
88
}
89
}
90
}
91
92