212 lines
12 KiB
PHP
212 lines
12 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:5px;font-size:11.5px;color:var(--mw-t4);cursor:pointer">
|
|
<input type="checkbox" x-model="autoScroll" style="accent-color:var(--mw-v2)">
|
|
Auto-Scroll
|
|
</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.6;white-space:pre-wrap;word-break:break-all">
|
|
@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="color:{{ $color }}">{{ $line }}</span>
|
|
<br>
|
|
@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>
|