From 7d94ec0d18b46893e23680cf8438109a34cc2a10 Mon Sep 17 00:00:00 2001
From: Matthias Nott <mnott@mnsoft.org>
Date: Sun, 22 Feb 2026 16:55:03 +0100
Subject: [PATCH] feat: promote/sync/rebuild UI, operations page, bidirectional sync, lifecycle ops

---
 static/index.html |   89 +++++++++++++++++++++++++++++++++++++++++---
 1 files changed, 83 insertions(+), 6 deletions(-)

diff --git a/static/index.html b/static/index.html
index 7d7d251..d51b7bf 100644
--- a/static/index.html
+++ b/static/index.html
@@ -5,7 +5,7 @@
   <meta name="viewport" content="width=device-width, initial-scale=1.0">
   <title>OPS Dashboard</title>
   <script src="https://cdn.tailwindcss.com"></script>
-  <link rel="stylesheet" href="/static/css/style.css">
+  <link rel="stylesheet" href="/static/css/style.css?v=10">
   <style>
     body { background: #0f172a; color: #e2e8f0; margin: 0; }
     #app { display: flex; min-height: 100vh; }
@@ -79,13 +79,13 @@
         <svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M21 15v4a2 2 0 01-2 2H5a2 2 0 01-2-2v-4"/><polyline points="7 10 12 15 17 10"/><line x1="12" y1="15" x2="12" y2="3"/></svg>
         Backups
       </a>
+      <a class="sidebar-link" data-page="operations" onclick="showPage('operations')">
+        <svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><path d="M12 2L2 7l10 5 10-5-10-5z"/><path d="M2 17l10 5 10-5"/><path d="M2 12l10 5 10-5"/></svg>
+        Operations
+      </a>
       <a class="sidebar-link" data-page="system" onclick="showPage('system')">
         <svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><circle cx="12" cy="12" r="3"/><path d="M19.4 15a1.65 1.65 0 00.33 1.82l.06.06a2 2 0 010 2.83 2 2 0 01-2.83 0l-.06-.06a1.65 1.65 0 00-1.82-.33 1.65 1.65 0 00-1 1.51V21a2 2 0 01-4 0v-.09A1.65 1.65 0 009 19.4a1.65 1.65 0 00-1.82.33l-.06.06a2 2 0 01-2.83-2.83l.06-.06A1.65 1.65 0 004.68 15a1.65 1.65 0 00-1.51-1H3a2 2 0 010-4h.09A1.65 1.65 0 004.6 9a1.65 1.65 0 00-.33-1.82l-.06-.06a2 2 0 012.83-2.83l.06.06A1.65 1.65 0 009 4.68a1.65 1.65 0 001-1.51V3a2 2 0 014 0v.09a1.65 1.65 0 001 1.51 1.65 1.65 0 001.82-.33l.06-.06a2 2 0 012.83 2.83l-.06.06A1.65 1.65 0 0019.4 9a1.65 1.65 0 001.51 1H21a2 2 0 010 4h-.09a1.65 1.65 0 00-1.51 1z"/></svg>
         System
-      </a>
-      <a class="sidebar-link" data-page="restore" onclick="showPage('restore')">
-        <svg width="18" height="18" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><polyline points="1 4 1 10 7 10"/><path d="M3.51 15a9 9 0 102.13-9.36L1 10"/></svg>
-        Restore
       </a>
     </nav>
     <div class="sidebar-footer" id="sidebar-footer">
@@ -139,6 +139,83 @@
   </div>
 </div>
 
-<script src="/static/js/app.js?v=4"></script>
+<!-- Restore Modal -->
+<div id="restore-modal" class="modal-overlay" style="display:none;" onclick="if(event.target===this)closeRestoreModal()">
+  <div class="modal-box" style="max-width:640px;">
+    <div class="modal-header">
+      <span style="font-weight:600;color:#f3f4f6;">Restore Backup</span>
+      <button onclick="closeRestoreModal()" style="background:none;border:none;color:#9ca3af;font-size:1.25rem;cursor:pointer;">&times;</button>
+    </div>
+    <div class="modal-body">
+      <!-- Info rows -->
+      <div class="restore-info-row">
+        <span class="restore-info-label">Target</span>
+        <span class="restore-info-value" id="restore-modal-project"></span>
+      </div>
+      <div class="restore-info-row" id="restore-source-row">
+        <span class="restore-info-label">Source</span>
+        <span class="restore-info-value" id="restore-modal-source"></span>
+      </div>
+      <div id="restore-source-selector" style="display:none;margin-bottom:0.625rem;">
+        <div style="display:flex;align-items:baseline;gap:0.75rem;font-size:0.875rem;">
+          <span class="restore-info-label">Source</span>
+          <label style="display:flex;align-items:center;gap:0.375rem;color:#d1d5db;cursor:pointer;">
+            <input type="radio" name="restore-source" value="local" checked style="accent-color:#3b82f6;"> Local
+          </label>
+          <label style="display:flex;align-items:center;gap:0.375rem;color:#d1d5db;cursor:pointer;">
+            <input type="radio" name="restore-source" value="offsite" style="accent-color:#3b82f6;"> Offsite
+          </label>
+        </div>
+      </div>
+      <div class="restore-info-row" style="margin-bottom:1rem;">
+        <span class="restore-info-label">Backup</span>
+        <span class="restore-info-value mono" id="restore-modal-name" style="font-size:0.8125rem;word-break:break-all;"></span>
+      </div>
+
+<!-- Restore mode selector -->      <div style="margin-bottom:1rem;">        <div style="font-size:0.8125rem;font-weight:500;color:#9ca3af;margin-bottom:0.5rem;">Restore Mode</div>        <div style="display:flex;gap:1rem;">          <label style="display:flex;align-items:center;gap:0.375rem;font-size:0.875rem;color:#d1d5db;cursor:pointer;">            <input type="radio" name="restore-mode" value="full" checked style="accent-color:#3b82f6;"> Full          </label>          <label style="display:flex;align-items:center;gap:0.375rem;font-size:0.875rem;color:#d1d5db;cursor:pointer;">            <input type="radio" name="restore-mode" value="db" style="accent-color:#3b82f6;"> Database only          </label>          <label style="display:flex;align-items:center;gap:0.375rem;font-size:0.875rem;color:#d1d5db;cursor:pointer;">            <input type="radio" name="restore-mode" value="wp" style="accent-color:#3b82f6;"> WP-Content only          </label>        </div>      </div>      <!-- Dry run checkbox -->      <label style="display:flex;align-items:center;gap:0.5rem;font-size:0.875rem;color:#d1d5db;cursor:pointer;margin-bottom:1rem;">        <input type="checkbox" id="restore-dry-run" checked style="width:1rem;height:1rem;accent-color:#3b82f6;">        Dry run (preview only — no changes made)      </label>      <!-- Warning -->
+      <div style="background:rgba(220,38,38,0.1);border:1px solid rgba(220,38,38,0.3);border-radius:0.5rem;padding:0.75rem 1rem;font-size:0.8125rem;color:#fca5a5;margin-bottom:1rem;">
+        Warning: a real restore will stop services, replace data, and restart containers.
+        Always run a dry run first.
+      </div>
+
+      <!-- SSE output (shown after start) -->
+      <div id="restore-modal-output" style="display:none;">
+        <div style="font-size:0.8125rem;font-weight:500;color:#9ca3af;margin-bottom:0.375rem;">Output</div>
+        <div id="restore-modal-terminal" class="terminal" style="max-height:300px;"></div>
+      </div>
+    </div>
+    <div class="modal-footer">
+      <button class="btn btn-ghost btn-sm" onclick="closeRestoreModal()">Cancel</button>
+      <button id="restore-start-btn" class="btn btn-danger btn-sm" onclick="startRestore()">Start Restore</button>
+    </div>
+  </div>
+</div>
+
+<!-- Operations Modal -->
+<div id="ops-modal" class="modal-overlay" style="display:none;" onclick="if(event.target===this)closeOpsModal()">
+  <div class="modal-box" style="max-width:700px;">
+    <div class="modal-header">
+      <span id="ops-modal-title" style="font-weight:600;color:#f3f4f6;">Operation</span>
+      <button onclick="closeOpsModal()" style="background:none;border:none;color:#9ca3af;font-size:1.25rem;cursor:pointer;">&times;</button>
+    </div>
+    <div class="modal-body">
+      <div id="ops-modal-info" style="margin-bottom:1rem;"></div>
+      <label id="ops-dry-run-row" style="display:flex;align-items:center;gap:0.5rem;font-size:0.875rem;color:#d1d5db;cursor:pointer;margin-bottom:1rem;">
+        <input type="checkbox" id="ops-dry-run" checked style="width:1rem;height:1rem;accent-color:#3b82f6;">
+        Dry run (preview only)
+      </label>
+      <div id="ops-modal-output" style="display:none;">
+        <div style="font-size:0.8125rem;font-weight:500;color:#9ca3af;margin-bottom:0.375rem;">Output</div>
+        <div id="ops-modal-terminal" class="terminal" style="max-height:350px;"></div>
+      </div>
+    </div>
+    <div class="modal-footer">
+      <button class="btn btn-ghost btn-sm" onclick="closeOpsModal()">Cancel</button>
+      <button id="ops-start-btn" class="btn btn-primary btn-sm" onclick="startOperation()">Start</button>
+    </div>
+  </div>
+</div>
+
+<script src="/static/js/app.js?v=12"></script>
 </body>
 </html>

--
Gitblit v1.3.1