mailwolt/resources/views/livewire/ui/system/update-page.blade.php

214 lines
13 KiB
PHP

<x-slot:breadcrumbParent>System</x-slot:breadcrumbParent>
<x-slot:breadcrumb>Updates</x-slot:breadcrumb>
<div>
{{-- ═══ Page Header ═══ --}}
<div class="mbx-page-header">
<div class="mbx-page-title">
<svg width="16" height="16" viewBox="0 0 16 16" fill="none">
<path d="M8 2a6 6 0 1 1-4.24 1.76" stroke="currentColor" stroke-width="1.3" stroke-linecap="round"/>
<path d="M2 2v3.5H5.5" stroke="currentColor" stroke-width="1.3" stroke-linecap="round" stroke-linejoin="round"/>
</svg>
Updates
</div>
<div class="mbx-page-actions">
<button wire:click="checkForUpdates"
wire:loading.attr="disabled"
wire:target="checkForUpdates"
class="mbx-btn-mute">
<svg width="12" height="12" viewBox="0 0 14 14" fill="none"><path d="M7 1.5A5.5 5.5 0 1 1 1.5 7" stroke="currentColor" stroke-width="1.3" stroke-linecap="round"/><path d="M1.5 2v3.5H5" stroke="currentColor" stroke-width="1.3" stroke-linecap="round" stroke-linejoin="round"/></svg>
<span wire:loading.remove wire:target="checkForUpdates">Auf Updates prüfen</span>
<span wire:loading wire:target="checkForUpdates">Prüfe </span>
</button>
@if($hasUpdate)
<button wire:click="runUpdate"
wire:loading.attr="disabled"
wire:target="runUpdate"
class="mbx-btn-primary"
@if($state === 'running') disabled @endif>
<svg width="12" height="12" viewBox="0 0 14 14" fill="none"><path d="M7 1L7 9" stroke="currentColor" stroke-width="1.3" stroke-linecap="round"/><path d="M4 6l3 3 3-3" stroke="currentColor" stroke-width="1.3" stroke-linecap="round" stroke-linejoin="round"/><path d="M2 12h10" stroke="currentColor" stroke-width="1.3" stroke-linecap="round"/></svg>
<span wire:loading.remove wire:target="runUpdate">{{ $displayLatest ?? 'Update' }} installieren</span>
<span wire:loading wire:target="runUpdate">Starte </span>
</button>
@endif
</div>
</div>
{{-- Polling when running --}}
@if($state === 'running' || $running)
<div wire:poll.2s="pollStatus"></div>
@endif
<div class="mbx-sections">
{{-- ═══ Section 1: Version Info ═══ --}}
<div class="mbx-section">
<div class="mbx-domain-head">
<div class="mbx-domain-info">
<span class="mbx-badge-mute">Versionsinformation</span>
</div>
<div style="display:flex;align-items:center;gap:8px">
@if($state === 'running')
<span style="font-size:11px;padding:3px 10px;border-radius:5px;background:rgba(14,165,233,.1);border:1px solid rgba(14,165,233,.3);color:#7dd3fc">
Update läuft
</span>
@elseif($rc !== null && $rc !== 0)
<span class="mbx-badge-warn">Fehlgeschlagen (rc={{ $rc }})</span>
@elseif($hasUpdate)
<span class="mbx-badge-warn">Update verfügbar</span>
@else
<span class="mbx-badge-ok">Aktuell</span>
@endif
</div>
</div>
<div style="padding:16px 18px">
<div class="mw-modal-grid2">
<div>
<div class="mw-modal-label">Installierte Version</div>
<div style="font-family:monospace;font-size:14px;color:var(--mw-t1);margin-top:4px">
{{ $displayCurrent ?? '—' }}
</div>
</div>
<div>
<div class="mw-modal-label">Verfügbare Version</div>
<div style="font-family:monospace;font-size:14px;color:{{ $hasUpdate ? '#fbbf24' : 'var(--mw-t1)' }};margin-top:4px">
{{ $displayLatest ?? '—' }}
</div>
</div>
</div>
@if(!$hasUpdate && $displayCurrent)
<div style="margin-top:14px;display:flex;align-items:center;gap:7px;font-size:12px;color:rgba(34,197,94,.8)">
<svg width="12" height="12" viewBox="0 0 12 12" fill="none"><path d="M2 6.5l2.5 2.5 5.5-5.5" stroke="currentColor" stroke-width="1.4" stroke-linecap="round" stroke-linejoin="round"/></svg>
Du bist auf dem neuesten Stand.
</div>
@endif
</div>
</div>
{{-- ═══ Section 2: Status ═══ --}}
<div class="mbx-section">
<div class="mbx-domain-head">
<div class="mbx-domain-info">
<span class="mbx-badge-mute">Update-Status</span>
</div>
</div>
<div style="padding:16px 18px;display:flex;flex-direction:column;gap:14px">
@if($state === 'running')
{{-- Running state --}}
<div style="display:flex;align-items:center;gap:12px">
<div style="width:36px;height:36px;border-radius:50%;background:rgba(14,165,233,.1);border:1px solid rgba(14,165,233,.3);display:flex;align-items:center;justify-content:center;flex-shrink:0;animation:spin 1.5s linear infinite">
<svg width="16" height="16" viewBox="0 0 16 16" fill="none"><path d="M8 2a6 6 0 1 1-4.24 1.76" stroke="#7dd3fc" stroke-width="1.4" stroke-linecap="round"/><path d="M2 2v3.5H5.5" stroke="#7dd3fc" stroke-width="1.4" stroke-linecap="round" stroke-linejoin="round"/></svg>
</div>
<div>
<div style="font-size:13px;font-weight:600;color:var(--mw-t1)">Update wird installiert </div>
<div style="font-size:11.5px;color:var(--mw-t4);margin-top:2px">Bitte nicht unterbrechen. Die Seite aktualisiert sich automatisch.</div>
</div>
</div>
@elseif($rc !== null && $rc !== 0)
{{-- Error state --}}
<div style="display:flex;align-items:flex-start;gap:10px;padding:12px 14px;background:rgba(239,68,68,.07);border:1px solid rgba(239,68,68,.25);border-radius:8px">
<svg width="14" height="14" viewBox="0 0 14 14" fill="none" style="flex-shrink:0;margin-top:1px;color:#f87171"><path d="M7 1L13 12H1L7 1Z" stroke="currentColor" stroke-width="1.3" stroke-linejoin="round"/><path d="M7 5.5v3" stroke="currentColor" stroke-width="1.3" stroke-linecap="round"/><circle cx="7" cy="10" r=".7" fill="currentColor"/></svg>
<div>
<div style="font-size:12.5px;font-weight:600;color:#f87171">Update fehlgeschlagen (rc={{ $rc }})</div>
<div style="font-size:11.5px;color:var(--mw-t4);margin-top:3px">Bitte das Log unten prüfen. Der Mailserver bleibt weiter in Betrieb.</div>
</div>
</div>
@elseif($rc === 0)
{{-- Success state --}}
<div style="display:flex;align-items:center;gap:10px;padding:12px 14px;background:rgba(34,197,94,.06);border:1px solid rgba(34,197,94,.2);border-radius:8px">
<svg width="14" height="14" viewBox="0 0 12 12" fill="none" style="color:#22c55e;flex-shrink:0"><path d="M2 6.5l2.5 2.5 5.5-5.5" stroke="currentColor" stroke-width="1.4" stroke-linecap="round" stroke-linejoin="round"/></svg>
<span style="font-size:12.5px;color:rgba(34,197,94,.9)">Update erfolgreich abgeschlossen.</span>
</div>
@else
{{-- Idle state --}}
<div style="font-size:12.5px;color:var(--mw-t4)">Kein Update aktiv. Klicke "Auf Updates prüfen" um die neueste Version zu ermitteln.</div>
@endif
{{-- Progress Bar --}}
@if($state === 'running' || $progressPct > 0)
<div>
<div style="display:flex;justify-content:space-between;margin-bottom:5px">
<span style="font-size:11px;color:var(--mw-t4)">Fortschritt</span>
<span style="font-size:11px;color:var(--mw-t4);font-family:monospace">{{ $progressPct }}%</span>
</div>
<div style="height:6px;background:var(--mw-bg4);border-radius:4px;overflow:hidden;border:1px solid var(--mw-b2)">
<div style="height:100%;width:{{ $progressPct }}%;transition:width .6s ease;background:{{ $rc !== null && $rc !== 0 ? 'rgba(239,68,68,.7)' : ($progressPct >= 100 ? 'rgba(34,197,94,.8)' : 'rgba(14,165,233,.8)') }};border-radius:4px"></div>
</div>
</div>
@endif
</div>
</div>
{{-- ═══ Section 3: Log Viewer ═══ --}}
<div class="mbx-section"
x-data="{
autoScroll: true,
init() {
this.$watch('$wire.logLines', () => {
if (this.autoScroll && this.$refs.logBox) {
this.$nextTick(() => {
this.$refs.logBox.scrollTop = this.$refs.logBox.scrollHeight;
});
}
});
}
}">
<div class="mbx-domain-head">
<div class="mbx-domain-info">
<span class="mbx-badge-mute">Update-Log</span>
</div>
<div style="display:flex;align-items:center;gap:8px">
<label style="display:flex;align-items:center;gap:7px;cursor:pointer;user-select:none">
<span x-on:click="autoScroll=!autoScroll"
style="position:relative;display:inline-block;width:28px;height:16px;flex-shrink:0">
<span :style="autoScroll ? 'background:var(--mw-v);border-color:var(--mw-v)' : 'background:var(--mw-bg4);border-color:var(--mw-b2)'"
style="position:absolute;inset:0;border-radius:8px;border:1px solid var(--mw-b2);transition:background .2s,border-color .2s"></span>
<span :style="autoScroll ? 'left:14px' : 'left:2px'"
style="position:absolute;top:2px;width:10px;height:10px;border-radius:50%;background:white;transition:left .2s;box-shadow:0 1px 3px rgba(0,0,0,.4)"></span>
</span>
<span style="font-size:11.5px;color:var(--mw-t4)">Auto-Scroll</span>
</label>
<button wire:click="clearLog"
wire:loading.attr="disabled"
wire:target="clearLog"
class="mbx-btn-mute"
style="font-size:11px;padding:3px 10px">
<svg width="11" height="11" viewBox="0 0 14 14" fill="none"><path d="M2 4h10M5 4V2.5h4V4M11.5 4l-.5 8H3L2.5 4" stroke="currentColor" stroke-width="1.2" stroke-linecap="round" stroke-linejoin="round"/></svg>
Log leeren
</button>
</div>
</div>
<div style="padding:12px 14px">
<div x-ref="logBox"
style="background:var(--mw-bg3);border:1px solid var(--mw-b2);border-radius:7px;padding:12px 14px;max-height:400px;overflow-y:auto;font-family:monospace;font-size:11.5px;color:var(--mw-t3);line-height:1.55">
@if(count($logLines) === 0)
<span style="color:var(--mw-t5);font-style:italic">Keine Log-Einträge vorhanden.</span>
@else
@foreach($logLines as $line)@php
$color = 'inherit';
if (str_contains($line, '[!]') || str_contains($line, 'error') || str_contains($line, 'Error') || str_contains($line, 'fehlgeschlagen')) {
$color = '#f87171';
} elseif (str_contains($line, '[✓]') || str_contains($line, 'beendet') || str_contains($line, 'abgeschlossen')) {
$color = 'rgba(34,197,94,.85)';
} elseif (str_contains($line, '[i]')) {
$color = 'var(--mw-t3)';
} elseif (str_contains($line, '=====')) {
$color = 'rgba(14,165,233,.8)';
}
@endphp<span style="display:block;color:{{ $color }}">{{ $line }}</span>@endforeach
@endif
</div>
<div style="margin-top:6px;font-size:11px;color:var(--mw-t5)">
{{ count($logLines) }} Zeilen · /var/log/mailwolt-update.log
</div>
</div>
</div>
</div>
</div>