2026-03-22 03:30:21 +00:00
|
|
|
/** Composable for app start/stop/restart/uninstall actions */
|
|
|
|
|
|
|
|
|
|
import { ref, onBeforeUnmount } from 'vue'
|
|
|
|
|
import { useAppStore } from '@/stores/app'
|
|
|
|
|
|
|
|
|
|
export function useAppsActions() {
|
|
|
|
|
const store = useAppStore()
|
|
|
|
|
const loadingActions = ref<Record<string, boolean>>({})
|
|
|
|
|
const actionError = ref('')
|
|
|
|
|
let errorTimer: ReturnType<typeof setTimeout> | undefined
|
|
|
|
|
const actionTimers = new Map<string, ReturnType<typeof setTimeout>>()
|
|
|
|
|
|
|
|
|
|
const uninstalling = ref(false)
|
|
|
|
|
const uninstallingApps = ref<Set<string>>(new Set())
|
|
|
|
|
|
|
|
|
|
function showActionError(msg: string) {
|
|
|
|
|
actionError.value = msg
|
|
|
|
|
if (errorTimer) clearTimeout(errorTimer)
|
|
|
|
|
errorTimer = setTimeout(() => { actionError.value = '' }, 5000)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function startApp(id: string) {
|
|
|
|
|
loadingActions.value[id] = true
|
|
|
|
|
try {
|
|
|
|
|
await store.startPackage(id)
|
|
|
|
|
if (actionTimers.has(id)) clearTimeout(actionTimers.get(id)!)
|
|
|
|
|
actionTimers.set(id, setTimeout(() => {
|
|
|
|
|
loadingActions.value[id] = false
|
|
|
|
|
actionTimers.delete(id)
|
|
|
|
|
}, 5000))
|
|
|
|
|
} catch (err) {
|
|
|
|
|
if (import.meta.env.DEV) console.error('Failed to start app:', err)
|
|
|
|
|
showActionError(`Failed to start app: ${err instanceof Error ? err.message : 'Unknown error'}`)
|
|
|
|
|
loadingActions.value[id] = false
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function stopApp(id: string) {
|
|
|
|
|
loadingActions.value[id] = true
|
|
|
|
|
try {
|
|
|
|
|
await store.stopPackage(id)
|
|
|
|
|
if (actionTimers.has(id)) clearTimeout(actionTimers.get(id)!)
|
|
|
|
|
actionTimers.set(id, setTimeout(() => {
|
|
|
|
|
loadingActions.value[id] = false
|
|
|
|
|
actionTimers.delete(id)
|
|
|
|
|
}, 5000))
|
|
|
|
|
} catch (err) {
|
|
|
|
|
if (import.meta.env.DEV) console.error('Failed to stop app:', err)
|
|
|
|
|
showActionError(`Failed to stop app: ${err instanceof Error ? err.message : 'Unknown error'}`)
|
|
|
|
|
loadingActions.value[id] = false
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function restartApp(id: string) {
|
|
|
|
|
loadingActions.value[id] = true
|
|
|
|
|
try {
|
|
|
|
|
await store.restartPackage(id)
|
|
|
|
|
if (actionTimers.has(id)) clearTimeout(actionTimers.get(id)!)
|
|
|
|
|
actionTimers.set(id, setTimeout(() => {
|
|
|
|
|
loadingActions.value[id] = false
|
|
|
|
|
actionTimers.delete(id)
|
|
|
|
|
}, 8000))
|
|
|
|
|
} catch (err) {
|
|
|
|
|
if (import.meta.env.DEV) console.error('Failed to restart app:', err)
|
|
|
|
|
showActionError(`Failed to restart app: ${err instanceof Error ? err.message : 'Unknown error'}`)
|
|
|
|
|
loadingActions.value[id] = false
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function confirmUninstall(appId: string) {
|
|
|
|
|
uninstalling.value = true
|
|
|
|
|
try {
|
|
|
|
|
uninstallingApps.value.add(appId)
|
|
|
|
|
await store.uninstallPackage(appId)
|
fix: overhaul container lifecycle — recovery, health, uninstall, UI state
Container recovery:
- Health monitor: MAX_RESTART_ATTEMPTS 3→10, interval 60s→120s
- Dependency-aware restarts: won't restart services before their deps
- Reset dependent counters when a dependency recovers
- Handle "created" state containers (were invisible to health monitor)
- Added IndeedHub, mempool-api, mysql to tier system
- Crash recovery: podman start timeout 30s→120s with retry
- Podman client: socket timeout 5s→30s, added restart policy
UI state representation:
- Exit code 0 shows "stopped" (gray), not "crashed" (red)
- Exit code 137 shows "killed (OOM)"
- Non-zero exit shows "crashed" (red)
- Added exit_code field to PackageDataEntry
Install/uninstall fixes:
- Install returns error when container doesn't start (was silent success)
- Post-install hooks awaited instead of fire-and-forget tokio::spawn
- Uninstall: graceful rm before force, volume prune, network cleanup
- Uninstall returns error on partial failure (was 200 OK)
Config consistency:
- DB passwords read from /var/lib/archipelago/secrets/ (was hardcoded)
- Bitcoin: added ZMQ ports 28332/28333 for LND block notifications
- IndeedHub port 7777→8190 (was conflicting with strfry)
- Marketplace versions: LND 0.17.4→0.18.4, Mempool 2.5.0→3.0.0
Performance:
- Metrics collector interval 60s→300s (was duplicating health monitor)
- Podman client: proper error propagation instead of unwrap_or_default
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-31 07:03:57 +01:00
|
|
|
// State update comes via WebSocket — no manual deletion needed
|
2026-03-22 03:30:21 +00:00
|
|
|
} catch (err) {
|
|
|
|
|
if (import.meta.env.DEV) console.error('Failed to uninstall app:', err)
|
fix: overhaul container lifecycle — recovery, health, uninstall, UI state
Container recovery:
- Health monitor: MAX_RESTART_ATTEMPTS 3→10, interval 60s→120s
- Dependency-aware restarts: won't restart services before their deps
- Reset dependent counters when a dependency recovers
- Handle "created" state containers (were invisible to health monitor)
- Added IndeedHub, mempool-api, mysql to tier system
- Crash recovery: podman start timeout 30s→120s with retry
- Podman client: socket timeout 5s→30s, added restart policy
UI state representation:
- Exit code 0 shows "stopped" (gray), not "crashed" (red)
- Exit code 137 shows "killed (OOM)"
- Non-zero exit shows "crashed" (red)
- Added exit_code field to PackageDataEntry
Install/uninstall fixes:
- Install returns error when container doesn't start (was silent success)
- Post-install hooks awaited instead of fire-and-forget tokio::spawn
- Uninstall: graceful rm before force, volume prune, network cleanup
- Uninstall returns error on partial failure (was 200 OK)
Config consistency:
- DB passwords read from /var/lib/archipelago/secrets/ (was hardcoded)
- Bitcoin: added ZMQ ports 28332/28333 for LND block notifications
- IndeedHub port 7777→8190 (was conflicting with strfry)
- Marketplace versions: LND 0.17.4→0.18.4, Mempool 2.5.0→3.0.0
Performance:
- Metrics collector interval 60s→300s (was duplicating health monitor)
- Podman client: proper error propagation instead of unwrap_or_default
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-31 07:03:57 +01:00
|
|
|
showActionError(`Failed to uninstall: ${err instanceof Error ? err.message : 'Unknown error'}`)
|
2026-03-22 03:30:21 +00:00
|
|
|
} finally {
|
|
|
|
|
uninstallingApps.value.delete(appId)
|
|
|
|
|
uninstalling.value = false
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
onBeforeUnmount(() => {
|
|
|
|
|
for (const t of actionTimers.values()) clearTimeout(t)
|
|
|
|
|
actionTimers.clear()
|
|
|
|
|
if (errorTimer) clearTimeout(errorTimer)
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
loadingActions,
|
|
|
|
|
actionError,
|
|
|
|
|
uninstalling,
|
|
|
|
|
uninstallingApps,
|
|
|
|
|
showActionError,
|
|
|
|
|
startApp,
|
|
|
|
|
stopApp,
|
|
|
|
|
restartApp,
|
|
|
|
|
confirmUninstall,
|
|
|
|
|
}
|
|
|
|
|
}
|