vito/app/Models/ServerLog.php
2025-03-12 13:31:10 +01:00

193 lines
5.0 KiB
PHP
Executable File

<?php
namespace App\Models;
use Exception;
use Illuminate\Database\Eloquent\Builder;
use Illuminate\Database\Eloquent\Factories\HasFactory;
use Illuminate\Database\Eloquent\Relations\BelongsTo;
use Illuminate\Support\Facades\File;
use Illuminate\Support\Facades\Log;
use Illuminate\Support\Facades\Storage;
use Illuminate\Support\Str;
use Symfony\Component\HttpFoundation\StreamedResponse;
use Throwable;
/**
* @property int $server_id
* @property ?int $site_id
* @property string $type
* @property string $name
* @property string $disk
* @property bool $is_remote
* @property Server $server
* @property ?Site $site
*/
class ServerLog extends AbstractModel
{
/** @use HasFactory<\Database\Factories\ServerLogFactory> */
use HasFactory;
protected $fillable = [
'server_id',
'site_id',
'type',
'name',
'disk',
'is_remote',
];
protected $casts = [
'server_id' => 'integer',
'site_id' => 'integer',
'is_remote' => 'boolean',
];
public static function boot(): void
{
parent::boot();
self::deleting(function (ServerLog $log): void {
if ($log->is_remote) {
try {
if (Storage::disk($log->disk)->exists($log->name)) {
Storage::disk($log->disk)->delete($log->name);
}
} catch (Exception $e) {
Log::error($e->getMessage(), ['exception' => $e]);
}
}
});
}
public function getRouteKey(): string
{
return 'log';
}
/**
* @return BelongsTo<Server, $this>
*/
public function server(): BelongsTo
{
return $this->belongsTo(Server::class);
}
/**
* @return BelongsTo<Site, $this>
*/
public function site(): BelongsTo
{
return $this->belongsTo(Site::class);
}
/**
* @throws Throwable
*/
public function download(): StreamedResponse
{
if ($this->is_remote) {
$tmpName = $this->server->id.'-'.strtotime('now').'-'.$this->type.'.log';
$tmpPath = Storage::disk('local')->path($tmpName);
$this->server->ssh()->download($tmpPath, $this->name);
dispatch(function () use ($tmpPath): void {
if (File::exists($tmpPath)) {
File::delete($tmpPath);
}
})
->delay(now()->addMinutes(5))
->onQueue('default');
return Storage::disk('local')->download($tmpName, str($this->name)->afterLast('/'));
}
return Storage::disk($this->disk)->download($this->name);
}
/**
* @param Builder<ServerLog> $query
* @return Builder<ServerLog>
*/
public static function getRemote(Builder $query, bool $active = true, ?Site $site = null): Builder
{
$query->where('is_remote', $active);
if ($site instanceof \App\Models\Site) {
$query->where('name', 'like', $site->path.'%');
}
return $query;
}
public function write(string $buf): void
{
if (Str::contains($buf, 'VITO_SSH_ERROR')) {
$buf = str_replace('VITO_SSH_ERROR', '', $buf);
}
if (Storage::disk($this->disk)->exists($this->name)) {
Storage::disk($this->disk)->append($this->name, $buf);
} else {
Storage::disk($this->disk)->put($this->name, $buf);
}
}
public function getContent(?int $lines = null): ?string
{
if ($this->is_remote) {
return $this->server->os()->tail($this->name, $lines ?? 150);
}
if (Storage::disk($this->disk)->exists($this->name)) {
if ($lines !== null && $lines !== 0) {
return tail(Storage::disk($this->disk)->path($this->name), $lines);
}
$content = Storage::disk($this->disk)->get($this->name);
return $content ?? 'Empty log file!';
}
return "Log file doesn't exist!";
}
public static function log(Server $server, string $type, string $content, ?Site $site = null): ServerLog
{
$log = new self([
'server_id' => $server->id,
'site_id' => $site?->id,
'name' => $server->id.'-'.strtotime('now').'-'.$type.'.log',
'type' => $type,
'disk' => config('core.logs_disk'),
]);
$log->save();
$log->write($content);
return $log;
}
public static function newLog(Server $server, string $type): ServerLog
{
return new self([
'server_id' => $server->id,
'name' => $server->id.'-'.strtotime('now').'-'.$type.'.log',
'type' => $type,
'disk' => config('core.logs_disk'),
]);
}
public function forSite(Site|int $site): ServerLog
{
if ($site instanceof Site) {
$site = $site->id;
return $this;
}
$this->site_id = $site;
return $this;
}
}