feat: обновление с Git из админки (/admin/system)

Co-authored-by: Cursor <cursoragent@cursor.com>
This commit is contained in:
shop
2026-05-17 14:26:11 +03:00
parent d4dd1fb587
commit 69dfd2a93a
14 changed files with 482 additions and 54 deletions
+7 -9
View File
@@ -2,17 +2,15 @@
<div class="admin-header">
<h1>Админ-панель</h1>
<nav class="admin-nav">
<a href="/admin" class="admin-nav__link admin-nav__link--active">Обзор</a>
<a href="/admin/orders" class="admin-nav__link">Заказы</a>
<a href="/admin/users" class="admin-nav__link">Пользователи</a>
<a href="/admin/products" class="admin-nav__link">Товары</a>
<a href="/admin/promo-codes" class="admin-nav__link">Промокоды</a>
<a href="/admin/reservations" class="admin-nav__link">Бронирования</a>
<a href="/" class="admin-nav__link">В магазин</a>
</nav>
<%- include('../partials/admin-nav', { adminNav: 'dashboard' }) %>
</div>
<section class="card" style="margin-bottom:1.5rem">
<h2 style="margin-top:0">Обновление с Git</h2>
<p class="muted">Подтянуть новую версию и перезапустить магазин без SSH.</p>
<a href="/admin/system" class="btn btn--primary">Перейти к обновлению →</a>
</section>
<div class="stats-grid">
<div class="stat-card">
<span class="stat-card__label">Пользователи</span>
+1 -9
View File
@@ -2,15 +2,7 @@
<div class="admin-header">
<h1>Заказы</h1>
<nav class="admin-nav">
<a href="/admin" class="admin-nav__link">Обзор</a>
<a href="/admin/orders" class="admin-nav__link admin-nav__link--active">Заказы</a>
<a href="/admin/users" class="admin-nav__link">Пользователи</a>
<a href="/admin/products" class="admin-nav__link">Товары</a>
<a href="/admin/promo-codes" class="admin-nav__link">Промокоды</a>
<a href="/admin/reservations" class="admin-nav__link">Бронирования</a>
<a href="/" class="admin-nav__link">В магазин</a>
</nav>
<%- include('../partials/admin-nav', { adminNav: 'orders' }) %>
</div>
<% const statusLabels = { pending: 'Ожидает', paid: 'Оплачен', shipped: 'Отправлен', cancelled: 'Отменён' }; %>
+1 -9
View File
@@ -2,15 +2,7 @@
<div class="admin-header">
<h1>Товары — цены и скидки</h1>
<nav class="admin-nav">
<a href="/admin" class="admin-nav__link">Обзор</a>
<a href="/admin/orders" class="admin-nav__link">Заказы</a>
<a href="/admin/users" class="admin-nav__link">Пользователи</a>
<a href="/admin/products" class="admin-nav__link admin-nav__link--active">Товары</a>
<a href="/admin/promo-codes" class="admin-nav__link">Промокоды</a>
<a href="/admin/reservations" class="admin-nav__link">Бронирования</a>
<a href="/" class="admin-nav__link">В магазин</a>
</nav>
<%- include('../partials/admin-nav', { adminNav: 'products' }) %>
</div>
<% if (stockUpdated) { %>
+1 -9
View File
@@ -2,15 +2,7 @@
<div class="admin-header">
<h1>Промокоды и скидки</h1>
<nav class="admin-nav">
<a href="/admin" class="admin-nav__link">Обзор</a>
<a href="/admin/orders" class="admin-nav__link">Заказы</a>
<a href="/admin/users" class="admin-nav__link">Пользователи</a>
<a href="/admin/products" class="admin-nav__link">Товары</a>
<a href="/admin/promo-codes" class="admin-nav__link admin-nav__link--active">Промокоды</a>
<a href="/admin/reservations" class="admin-nav__link">Бронирования</a>
<a href="/" class="admin-nav__link">В магазин</a>
</nav>
<%- include('../partials/admin-nav', { adminNav: 'promo' }) %>
</div>
<% if (created) { %><p class="alert alert--success">Промокод создан</p><% } %>
+1 -9
View File
@@ -2,15 +2,7 @@
<div class="admin-header">
<h1>Бронирования</h1>
<nav class="admin-nav">
<a href="/admin" class="admin-nav__link">Обзор</a>
<a href="/admin/orders" class="admin-nav__link">Заказы</a>
<a href="/admin/users" class="admin-nav__link">Пользователи</a>
<a href="/admin/products" class="admin-nav__link">Товары</a>
<a href="/admin/promo-codes" class="admin-nav__link">Промокоды</a>
<a href="/admin/reservations" class="admin-nav__link admin-nav__link--active">Бронирования</a>
<a href="/" class="admin-nav__link">В магазин</a>
</nav>
<%- include('../partials/admin-nav', { adminNav: 'reservations' }) %>
</div>
<% const resStatus = { active: 'Активна', fulfilled: 'Выполнена', cancelled: 'Отменена', expired: 'Истекла' }; %>
+109
View File
@@ -0,0 +1,109 @@
<%- include('../partials/layout-start') %>
<div class="admin-header">
<h1>Обновление с Git</h1>
<%- include('../partials/admin-nav', { adminNav: 'system' }) %>
</div>
<% if (updateOk) { %>
<p class="alert alert--success">Обновление выполнено успешно. Если сайт не открывается — подождите 10–20 сек и обновите страницу.</p>
<% } %>
<% if (updateFail) { %>
<p class="alert alert--error">Обновление завершилось с ошибкой (код <%= updateCode %>).</p>
<% } %>
<% if (confirmError) { %>
<p class="alert alert--error">Введите <strong>update</strong> для подтверждения.</p>
<% } %>
<% if (disabledError) { %>
<p class="alert alert--warn">Обновление из админки отключено на этом сервере.</p>
<% } %>
<section class="card admin-system">
<h2>Текущая версия</h2>
<% if (!git.available) { %>
<p class="alert alert--warn"><%= git.reason || 'Git недоступен' %></p>
<% } else { %>
<dl class="profile-dl admin-system__meta">
<dt>Версия приложения</dt>
<dd><strong>v<%= git.packageVersion || '?' %></strong></dd>
<dt>Каталог</dt>
<dd><code class="admin-system__path"><%= git.shopRoot %></code></dd>
<dt>Ветка</dt>
<dd><%= git.branch %></dd>
<dt>Коммит</dt>
<dd>
<code><%= git.commitShort %></code>
— <%= git.commitSubject %>
<span class="muted">(<%= git.commitDate %>)</span>
</dd>
<% if (git.dirty) { %>
<dt>Состояние</dt>
<dd><span class="badge badge--warn">Есть незакоммиченные изменения</span></dd>
<% } %>
<% if (git.behind != null) { %>
<dt>На origin/main</dt>
<dd>
<% if (git.behind > 0) { %>
<span class="badge badge--sale">Доступно обновлений: <%= git.behind %></span>
<% } else { %>
<span class="badge">Актуально</span>
<% } %>
</dd>
<% } %>
<% if (git.fetchError) { %>
<dt>origin</dt>
<dd class="muted">Не удалось проверить: <%= git.fetchError %></dd>
<% } %>
</dl>
<% } %>
<div class="admin-system__actions">
<form action="/admin/system/check" method="post" class="inline-form">
<button type="submit" class="btn btn--ghost">
<%- include('../partials/icon', { name: 'refresh', iconSize: 18 }) %>
Проверить на Git
</button>
</form>
</div>
<% if (git.updateEnabled) { %>
<hr class="admin-system__hr">
<h2>Обновить сейчас</h2>
<p class="muted admin-hint">
Выполняется <code>git pull</code>, <code>npm install</code> и перезапуск службы <code>shop</code>.
Страница может оборваться на несколько секунд — это нормально.
</p>
<form action="/admin/system/update" method="post" class="form admin-system__form" onsubmit="return confirm('Обновить код с Git и перезапустить магазин?');">
<label class="label">
Подтверждение: введите <strong>update</strong>
<input type="text" name="confirm" class="input" required autocomplete="off" placeholder="update">
</label>
<button type="submit" class="btn btn--primary btn--lg">
<%- include('../partials/icon', { name: 'download', iconSize: 20 }) %>
Обновить с Git
</button>
</form>
<% } else if (git.available) { %>
<p class="alert alert--warn">
Обновление из админки отключено (Windows, нет .git или <code>ADMIN_UPDATE_ENABLED=0</code>).
На сервере: <code>bash "$SHOP_ROOT/scripts/server-update.sh"</code>
</p>
<% } %>
</section>
<% if (updateLog) { %>
<section class="card admin-system__log">
<h2>Журнал обновления</h2>
<pre class="admin-system__pre"><%= updateLog %></pre>
</section>
<% } %>
<section class="card admin-system__help muted">
<h2>Настройка сервера</h2>
<p>В <code>.env</code>: <code>SHOP_ROOT=/opt/shop</code>, <code>ADMIN_UPDATE_ENABLED=1</code>.</p>
<p>Если служба работает от <code>www-data</code>, добавьте в sudoers (от root):</p>
<pre class="admin-system__pre">www-data ALL=(root) NOPASSWD: <%= git.shopRoot || '/opt/shop' %>/scripts/admin-web-update.sh</pre>
<p>И в .env: <code>ADMIN_UPDATE_USE_SUDO=1</code></p>
</section>
<%- include('../partials/layout-end') %>
+1 -9
View File
@@ -2,15 +2,7 @@
<div class="admin-header">
<h1>Пользователи</h1>
<nav class="admin-nav">
<a href="/admin" class="admin-nav__link">Обзор</a>
<a href="/admin/orders" class="admin-nav__link">Заказы</a>
<a href="/admin/users" class="admin-nav__link admin-nav__link--active">Пользователи</a>
<a href="/admin/products" class="admin-nav__link">Товары</a>
<a href="/admin/promo-codes" class="admin-nav__link">Промокоды</a>
<a href="/admin/reservations" class="admin-nav__link">Бронирования</a>
<a href="/" class="admin-nav__link">В магазин</a>
</nav>
<%- include('../partials/admin-nav', { adminNav: 'users' }) %>
</div>
<p class="muted admin-hint">
+16
View File
@@ -0,0 +1,16 @@
<%
const nav = typeof adminNav !== 'undefined' ? adminNav : '';
function navClass(id) {
return 'admin-nav__link' + (nav === id ? ' admin-nav__link--active' : '');
}
%>
<nav class="admin-nav">
<a href="/admin" class="<%= navClass('dashboard') %>">Обзор</a>
<a href="/admin/orders" class="<%= navClass('orders') %>">Заказы</a>
<a href="/admin/users" class="<%= navClass('users') %>">Пользователи</a>
<a href="/admin/products" class="<%= navClass('products') %>">Товары</a>
<a href="/admin/promo-codes" class="<%= navClass('promo') %>">Промокоды</a>
<a href="/admin/reservations" class="<%= navClass('reservations') %>">Бронирования</a>
<a href="/admin/system" class="<%= navClass('system') %>">Обновление</a>
<a href="/" class="admin-nav__link">В магазин</a>
</nav>
+4
View File
@@ -20,6 +20,10 @@
<svg class="<%= cls %>" width="<%= sz %>" height="<%= sz %>" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" aria-hidden="true"><circle cx="12" cy="12" r="10"/><polyline points="12 6 12 12 16 14"/></svg>
<% } else if (name === 'arrow-left') { %>
<svg class="<%= cls %>" width="<%= sz %>" height="<%= sz %>" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" aria-hidden="true"><path d="m12 19-7-7 7-7M19 12H5"/></svg>
<% } else if (name === 'refresh') { %>
<svg class="<%= cls %>" width="<%= sz %>" height="<%= sz %>" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" aria-hidden="true"><path d="M21 12a9 9 0 1 1-2.64-6.36"/><path d="M21 3v6h-6"/></svg>
<% } else if (name === 'download') { %>
<svg class="<%= cls %>" width="<%= sz %>" height="<%= sz %>" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" aria-hidden="true"><path d="M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4"/><polyline points="7 10 12 15 17 10"/><line x1="12" y1="15" x2="12" y2="3"/></svg>
<% } else if (name === 'package') { %>
<svg class="<%= cls %>" width="<%= sz %>" height="<%= sz %>" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" aria-hidden="true"><path d="M16.5 9.4 7.55 4.24M21 16V8a2 2 0 0 0-1-1.73l-7-4a2 2 0 0 0-2 0l-7 4A2 2 0 0 0 3 8v8a2 2 0 0 0 1 1.73l7 4a2 2 0 0 0 2 0l7-4A2 2 0 0 0 21 16z"/><polyline points="3.27 6.96 12 12.01 20.73 6.96"/><line x1="12" y1="22.08" x2="12" y2="12"/></svg>
<% } %>