- 前端添加更新和重启按钮,支持一键更新 Release 构建 - 修复条件判断优先级问题,确保错误/成功状态正确显示 - 后端使用原子文件替换模式,确保更新过程安全可靠 - 在可执行文件同目录创建临时文件,保证 rename 原子性 - 删除未使用的 copyFile 函数,保持代码整洁
380 lines
19 KiB
Vue
380 lines
19 KiB
Vue
<template>
|
|
<div class="relative">
|
|
<!-- Admin: Full version badge with dropdown -->
|
|
<template v-if="isAdmin">
|
|
<button
|
|
@click="toggleDropdown"
|
|
class="flex items-center gap-1.5 px-2 py-1 text-xs rounded-lg transition-colors"
|
|
:class="[
|
|
hasUpdate
|
|
? 'bg-amber-100 dark:bg-amber-900/30 text-amber-700 dark:text-amber-400 hover:bg-amber-200 dark:hover:bg-amber-900/50'
|
|
: 'bg-gray-100 dark:bg-dark-800 text-gray-600 dark:text-dark-400 hover:bg-gray-200 dark:hover:bg-dark-700'
|
|
]"
|
|
:title="hasUpdate ? 'New version available' : 'Up to date'"
|
|
>
|
|
<span class="font-medium">v{{ currentVersion }}</span>
|
|
<!-- Update indicator -->
|
|
<span v-if="hasUpdate" class="relative flex h-2 w-2">
|
|
<span class="animate-ping absolute inline-flex h-full w-full rounded-full bg-amber-400 opacity-75"></span>
|
|
<span class="relative inline-flex rounded-full h-2 w-2 bg-amber-500"></span>
|
|
</span>
|
|
</button>
|
|
|
|
<!-- Dropdown -->
|
|
<transition name="dropdown">
|
|
<div
|
|
v-if="dropdownOpen"
|
|
ref="dropdownRef"
|
|
class="absolute left-0 mt-2 w-64 bg-white dark:bg-dark-800 rounded-xl shadow-lg border border-gray-200 dark:border-dark-700 z-50 overflow-hidden"
|
|
>
|
|
<!-- Header with refresh button -->
|
|
<div class="flex items-center justify-between px-4 py-3 border-b border-gray-100 dark:border-dark-700">
|
|
<span class="text-sm font-medium text-gray-700 dark:text-dark-300">{{ t('version.currentVersion') }}</span>
|
|
<button
|
|
@click="refreshVersion(true)"
|
|
class="p-1.5 rounded-lg text-gray-400 hover:text-gray-600 dark:hover:text-dark-200 hover:bg-gray-100 dark:hover:bg-dark-700 transition-colors"
|
|
:disabled="loading"
|
|
:title="t('version.refresh')"
|
|
>
|
|
<svg class="w-4 h-4" :class="{ 'animate-spin': loading }" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2">
|
|
<path stroke-linecap="round" stroke-linejoin="round" d="M16.023 9.348h4.992v-.001M2.985 19.644v-4.992m0 0h4.992m-4.993 0l3.181 3.183a8.25 8.25 0 0013.803-3.7M4.031 9.865a8.25 8.25 0 0113.803-3.7l3.181 3.182m0-4.991v4.99" />
|
|
</svg>
|
|
</button>
|
|
</div>
|
|
|
|
<div class="p-4">
|
|
<!-- Loading state -->
|
|
<div v-if="loading" class="flex items-center justify-center py-6">
|
|
<svg class="animate-spin h-6 w-6 text-primary-500" fill="none" viewBox="0 0 24 24">
|
|
<circle class="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" stroke-width="4"></circle>
|
|
<path class="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"></path>
|
|
</svg>
|
|
</div>
|
|
|
|
<!-- Content -->
|
|
<template v-else>
|
|
<!-- Version display - centered and prominent -->
|
|
<div class="text-center mb-4">
|
|
<div class="inline-flex items-center gap-2">
|
|
<span class="text-2xl font-bold text-gray-900 dark:text-white">v{{ currentVersion }}</span>
|
|
<!-- Show check mark when up to date -->
|
|
<span v-if="!hasUpdate" class="flex items-center justify-center w-5 h-5 rounded-full bg-green-100 dark:bg-green-900/30">
|
|
<svg class="w-3 h-3 text-green-600 dark:text-green-400" fill="currentColor" viewBox="0 0 20 20">
|
|
<path fill-rule="evenodd" d="M16.707 5.293a1 1 0 010 1.414l-8 8a1 1 0 01-1.414 0l-4-4a1 1 0 011.414-1.414L8 12.586l7.293-7.293a1 1 0 011.414 0z" clip-rule="evenodd" />
|
|
</svg>
|
|
</span>
|
|
</div>
|
|
<p class="text-xs text-gray-500 dark:text-dark-400 mt-1">
|
|
{{ hasUpdate ? t('version.latestVersion') + ': v' + latestVersion : t('version.upToDate') }}
|
|
</p>
|
|
</div>
|
|
|
|
<!-- Priority 1: Update error (must check before hasUpdate) -->
|
|
<div v-if="updateError" class="space-y-2">
|
|
<div class="flex items-center gap-3 p-3 rounded-lg bg-red-50 dark:bg-red-900/20 border border-red-200 dark:border-red-800/50">
|
|
<div class="flex-shrink-0 w-8 h-8 rounded-full bg-red-100 dark:bg-red-900/50 flex items-center justify-center">
|
|
<svg class="w-4 h-4 text-red-600 dark:text-red-400" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2">
|
|
<path stroke-linecap="round" stroke-linejoin="round" d="M6 18L18 6M6 6l12 12" />
|
|
</svg>
|
|
</div>
|
|
<div class="flex-1 min-w-0">
|
|
<p class="text-sm font-medium text-red-700 dark:text-red-300">{{ t('version.updateFailed') }}</p>
|
|
<p class="text-xs text-red-600/70 dark:text-red-400/70 truncate">{{ updateError }}</p>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Retry button -->
|
|
<button
|
|
@click="handleUpdate"
|
|
:disabled="updating"
|
|
class="w-full flex items-center justify-center gap-2 px-4 py-2 rounded-lg text-sm font-medium text-white bg-red-500 hover:bg-red-600 disabled:opacity-50 disabled:cursor-not-allowed transition-colors"
|
|
>
|
|
{{ t('version.retry') }}
|
|
</button>
|
|
</div>
|
|
|
|
<!-- Priority 2: Update success - need restart -->
|
|
<div v-else-if="updateSuccess && needRestart" class="space-y-2">
|
|
<div class="flex items-center gap-3 p-3 rounded-lg bg-green-50 dark:bg-green-900/20 border border-green-200 dark:border-green-800/50">
|
|
<div class="flex-shrink-0 w-8 h-8 rounded-full bg-green-100 dark:bg-green-900/50 flex items-center justify-center">
|
|
<svg class="w-4 h-4 text-green-600 dark:text-green-400" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2">
|
|
<path stroke-linecap="round" stroke-linejoin="round" d="M5 13l4 4L19 7" />
|
|
</svg>
|
|
</div>
|
|
<div class="flex-1 min-w-0">
|
|
<p class="text-sm font-medium text-green-700 dark:text-green-300">{{ t('version.updateComplete') }}</p>
|
|
<p class="text-xs text-green-600/70 dark:text-green-400/70">{{ t('version.restartRequired') }}</p>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Restart button -->
|
|
<button
|
|
@click="handleRestart"
|
|
:disabled="restarting"
|
|
class="w-full flex items-center justify-center gap-2 px-4 py-2 rounded-lg text-sm font-medium text-white bg-green-500 hover:bg-green-600 disabled:opacity-50 disabled:cursor-not-allowed transition-colors"
|
|
>
|
|
<svg v-if="restarting" class="animate-spin h-4 w-4" fill="none" viewBox="0 0 24 24">
|
|
<circle class="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" stroke-width="4"></circle>
|
|
<path class="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"></path>
|
|
</svg>
|
|
<svg v-else class="w-4 h-4" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2">
|
|
<path stroke-linecap="round" stroke-linejoin="round" d="M4 4v5h.582m15.356 2A8.001 8.001 0 004.582 9m0 0H9m11 11v-5h-.581m0 0a8.003 8.003 0 01-15.357-2m15.357 2H15" />
|
|
</svg>
|
|
{{ restarting ? t('version.restarting') : t('version.restartNow') }}
|
|
</button>
|
|
</div>
|
|
|
|
<!-- Priority 3: Update available for source build - show git pull hint -->
|
|
<div v-else-if="hasUpdate && !isReleaseBuild" class="space-y-2">
|
|
<a
|
|
v-if="releaseInfo?.html_url && releaseInfo.html_url !== '#'"
|
|
:href="releaseInfo.html_url"
|
|
target="_blank"
|
|
rel="noopener noreferrer"
|
|
class="flex items-center gap-3 p-3 rounded-lg bg-amber-50 dark:bg-amber-900/20 border border-amber-200 dark:border-amber-800/50 hover:bg-amber-100 dark:hover:bg-amber-900/30 transition-colors group"
|
|
>
|
|
<div class="flex-shrink-0 w-8 h-8 rounded-full bg-amber-100 dark:bg-amber-900/50 flex items-center justify-center">
|
|
<svg class="w-4 h-4 text-amber-600 dark:text-amber-400" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2">
|
|
<path stroke-linecap="round" stroke-linejoin="round" d="M4 16v1a3 3 0 003 3h10a3 3 0 003-3v-1m-4-4l-4 4m0 0l-4-4m4 4V4" />
|
|
</svg>
|
|
</div>
|
|
<div class="flex-1 min-w-0">
|
|
<p class="text-sm font-medium text-amber-700 dark:text-amber-300">{{ t('version.updateAvailable') }}</p>
|
|
<p class="text-xs text-amber-600/70 dark:text-amber-400/70">v{{ latestVersion }}</p>
|
|
</div>
|
|
<svg class="w-4 h-4 text-amber-500 dark:text-amber-400 group-hover:translate-x-0.5 transition-transform" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2">
|
|
<path stroke-linecap="round" stroke-linejoin="round" d="M9 5l7 7-7 7" />
|
|
</svg>
|
|
</a>
|
|
<!-- Source build hint -->
|
|
<div class="flex items-center gap-2 p-2 rounded-lg bg-blue-50 dark:bg-blue-900/20 border border-blue-200 dark:border-blue-800/50">
|
|
<svg class="w-3.5 h-3.5 text-blue-500 dark:text-blue-400 flex-shrink-0" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2">
|
|
<path stroke-linecap="round" stroke-linejoin="round" d="M13 16h-1v-4h-1m1-4h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z" />
|
|
</svg>
|
|
<p class="text-xs text-blue-600 dark:text-blue-400">{{ t('version.sourceModeHint') }}</p>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Priority 4: Update available for release build - show update button -->
|
|
<div v-else-if="hasUpdate && isReleaseBuild" class="space-y-2">
|
|
<!-- Update info card -->
|
|
<div class="flex items-center gap-3 p-3 rounded-lg bg-amber-50 dark:bg-amber-900/20 border border-amber-200 dark:border-amber-800/50">
|
|
<div class="flex-shrink-0 w-8 h-8 rounded-full bg-amber-100 dark:bg-amber-900/50 flex items-center justify-center">
|
|
<svg class="w-4 h-4 text-amber-600 dark:text-amber-400" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2">
|
|
<path stroke-linecap="round" stroke-linejoin="round" d="M4 16v1a3 3 0 003 3h10a3 3 0 003-3v-1m-4-4l-4 4m0 0l-4-4m4 4V4" />
|
|
</svg>
|
|
</div>
|
|
<div class="flex-1 min-w-0">
|
|
<p class="text-sm font-medium text-amber-700 dark:text-amber-300">{{ t('version.updateAvailable') }}</p>
|
|
<p class="text-xs text-amber-600/70 dark:text-amber-400/70">v{{ latestVersion }}</p>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Update button -->
|
|
<button
|
|
@click="handleUpdate"
|
|
:disabled="updating"
|
|
class="w-full flex items-center justify-center gap-2 px-4 py-2 rounded-lg text-sm font-medium text-white bg-primary-500 hover:bg-primary-600 disabled:opacity-50 disabled:cursor-not-allowed transition-colors"
|
|
>
|
|
<svg v-if="updating" class="animate-spin h-4 w-4" fill="none" viewBox="0 0 24 24">
|
|
<circle class="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" stroke-width="4"></circle>
|
|
<path class="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"></path>
|
|
</svg>
|
|
<svg v-else class="w-4 h-4" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2">
|
|
<path stroke-linecap="round" stroke-linejoin="round" d="M4 16v1a3 3 0 003 3h10a3 3 0 003-3v-1m-4-4l-4 4m0 0l-4-4m4 4V4" />
|
|
</svg>
|
|
{{ updating ? t('version.updating') : t('version.updateNow') }}
|
|
</button>
|
|
|
|
<!-- View release link -->
|
|
<a
|
|
v-if="releaseInfo?.html_url && releaseInfo.html_url !== '#'"
|
|
:href="releaseInfo.html_url"
|
|
target="_blank"
|
|
rel="noopener noreferrer"
|
|
class="flex items-center justify-center gap-1 text-xs text-gray-500 dark:text-dark-400 hover:text-gray-700 dark:hover:text-dark-200 transition-colors"
|
|
>
|
|
{{ t('version.viewChangelog') }}
|
|
<svg class="w-3 h-3" fill="none" viewBox="0 0 24 24" stroke="currentColor" stroke-width="2">
|
|
<path stroke-linecap="round" stroke-linejoin="round" d="M10 6H6a2 2 0 00-2 2v10a2 2 0 002 2h10a2 2 0 002-2v-4M14 4h6m0 0v6m0-6L10 14" />
|
|
</svg>
|
|
</a>
|
|
</div>
|
|
|
|
<!-- Priority 5: Up to date - show GitHub link -->
|
|
<a
|
|
v-else-if="releaseInfo?.html_url && releaseInfo.html_url !== '#'"
|
|
:href="releaseInfo.html_url"
|
|
target="_blank"
|
|
rel="noopener noreferrer"
|
|
class="flex items-center justify-center gap-2 py-2 text-sm text-gray-500 dark:text-dark-400 hover:text-gray-700 dark:hover:text-dark-200 transition-colors"
|
|
>
|
|
<svg class="w-4 h-4" fill="currentColor" viewBox="0 0 24 24">
|
|
<path fill-rule="evenodd" clip-rule="evenodd" d="M12 2C6.477 2 2 6.477 2 12c0 4.42 2.865 8.17 6.839 9.49.5.092.682-.217.682-.482 0-.237-.008-.866-.013-1.7-2.782.604-3.369-1.34-3.369-1.34-.454-1.156-1.11-1.464-1.11-1.464-.908-.62.069-.608.069-.608 1.003.07 1.531 1.03 1.531 1.03.892 1.529 2.341 1.087 2.91.831.092-.646.35-1.086.636-1.336-2.22-.253-4.555-1.11-4.555-4.943 0-1.091.39-1.984 1.029-2.683-.103-.253-.446-1.27.098-2.647 0 0 .84-.269 2.75 1.025A9.578 9.578 0 0112 6.836c.85.004 1.705.114 2.504.336 1.909-1.294 2.747-1.025 2.747-1.025.546 1.377.203 2.394.1 2.647.64.699 1.028 1.592 1.028 2.683 0 3.842-2.339 4.687-4.566 4.935.359.309.678.919.678 1.852 0 1.336-.012 2.415-.012 2.743 0 .267.18.578.688.48C19.138 20.167 22 16.418 22 12c0-5.523-4.477-10-10-10z" />
|
|
</svg>
|
|
{{ t('version.viewRelease') }}
|
|
</a>
|
|
</template>
|
|
</div>
|
|
</div>
|
|
</transition>
|
|
</template>
|
|
|
|
<!-- Non-admin: Simple static version text -->
|
|
<span
|
|
v-else-if="version"
|
|
class="text-xs text-gray-500 dark:text-dark-400"
|
|
>
|
|
v{{ version }}
|
|
</span>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import { ref, computed, onMounted, onBeforeUnmount } from 'vue';
|
|
import { useI18n } from 'vue-i18n';
|
|
import { useAuthStore } from '@/stores';
|
|
import { checkUpdates, performUpdate, restartService, type VersionInfo, type ReleaseInfo } from '@/api/admin/system';
|
|
|
|
const { t } = useI18n();
|
|
|
|
const props = defineProps<{
|
|
version?: string;
|
|
}>();
|
|
|
|
const authStore = useAuthStore();
|
|
|
|
const isAdmin = computed(() => authStore.isAdmin);
|
|
|
|
const loading = ref(false);
|
|
const dropdownOpen = ref(false);
|
|
const dropdownRef = ref<HTMLElement | null>(null);
|
|
|
|
const currentVersion = ref('0.1.0');
|
|
const latestVersion = ref('0.1.0');
|
|
const hasUpdate = ref(false);
|
|
const releaseInfo = ref<ReleaseInfo | null>(null);
|
|
const buildType = ref('source'); // "source" or "release"
|
|
|
|
// Update process states
|
|
const updating = ref(false);
|
|
const restarting = ref(false);
|
|
const needRestart = ref(false);
|
|
const updateError = ref('');
|
|
const updateSuccess = ref(false);
|
|
|
|
// Only show update check for release builds (binary/docker deployment)
|
|
const isReleaseBuild = computed(() => buildType.value === 'release');
|
|
|
|
function toggleDropdown() {
|
|
dropdownOpen.value = !dropdownOpen.value;
|
|
}
|
|
|
|
function closeDropdown() {
|
|
dropdownOpen.value = false;
|
|
}
|
|
|
|
async function refreshVersion(force = true) {
|
|
if (!isAdmin.value) return;
|
|
|
|
loading.value = true;
|
|
try {
|
|
const data: VersionInfo = await checkUpdates(force);
|
|
currentVersion.value = data.current_version;
|
|
latestVersion.value = data.latest_version;
|
|
buildType.value = data.build_type || 'source';
|
|
// Show update indicator for all build types
|
|
hasUpdate.value = data.has_update;
|
|
releaseInfo.value = data.release_info || null;
|
|
// Reset update states when refreshing
|
|
updateError.value = '';
|
|
updateSuccess.value = false;
|
|
needRestart.value = false;
|
|
} catch (error) {
|
|
console.error('Failed to check updates:', error);
|
|
} finally {
|
|
loading.value = false;
|
|
}
|
|
}
|
|
|
|
async function handleUpdate() {
|
|
if (updating.value) return;
|
|
|
|
updating.value = true;
|
|
updateError.value = '';
|
|
updateSuccess.value = false;
|
|
|
|
try {
|
|
const result = await performUpdate();
|
|
updateSuccess.value = true;
|
|
needRestart.value = result.need_restart;
|
|
hasUpdate.value = false;
|
|
} catch (error: unknown) {
|
|
const err = error as { response?: { data?: { message?: string } }; message?: string };
|
|
updateError.value = err.response?.data?.message || err.message || t('version.updateFailed');
|
|
} finally {
|
|
updating.value = false;
|
|
}
|
|
}
|
|
|
|
async function handleRestart() {
|
|
if (restarting.value) return;
|
|
|
|
restarting.value = true;
|
|
|
|
try {
|
|
await restartService();
|
|
// Service will restart, page will reload automatically or show disconnected
|
|
} catch (error) {
|
|
// Expected - connection will be lost during restart
|
|
console.log('Service restarting...');
|
|
}
|
|
|
|
// Show restarting state for a while, then reload
|
|
setTimeout(() => {
|
|
window.location.reload();
|
|
}, 3000);
|
|
}
|
|
|
|
function handleClickOutside(event: MouseEvent) {
|
|
const target = event.target as Node;
|
|
const button = (event.target as Element).closest('button');
|
|
if (dropdownRef.value && !dropdownRef.value.contains(target) && !button?.contains(target)) {
|
|
closeDropdown();
|
|
}
|
|
}
|
|
|
|
onMounted(() => {
|
|
if (isAdmin.value) {
|
|
refreshVersion(false);
|
|
} else if (props.version) {
|
|
currentVersion.value = props.version;
|
|
}
|
|
document.addEventListener('click', handleClickOutside);
|
|
});
|
|
|
|
onBeforeUnmount(() => {
|
|
document.removeEventListener('click', handleClickOutside);
|
|
});
|
|
</script>
|
|
|
|
<style scoped>
|
|
.dropdown-enter-active,
|
|
.dropdown-leave-active {
|
|
transition: all 0.2s ease;
|
|
}
|
|
|
|
.dropdown-enter-from,
|
|
.dropdown-leave-to {
|
|
opacity: 0;
|
|
transform: scale(0.95) translateY(-4px);
|
|
}
|
|
|
|
.line-clamp-3 {
|
|
display: -webkit-box;
|
|
-webkit-line-clamp: 3;
|
|
-webkit-box-orient: vertical;
|
|
overflow: hidden;
|
|
}
|
|
</style>
|