FiveM_Snippets/src/updater.ts

160 lines
5.0 KiB
TypeScript

import * as vscode from "vscode";
// ================= LOGGING SETUP =================
export const updaterOutput = vscode.window.createOutputChannel("Updater");
export function logDebug(...args: any[]) {
const message = args
.map(a => (typeof a === "object" ? JSON.stringify(a, null, 2) : a))
.join(" ");
updaterOutput.appendLine(message);
console.log(message); // optional, still shows in DevTools
updaterOutput.show(true);
}
// ================= CONFIG =================
const CONFIG = {
apiBase: "https://git.cstmgames.dev/api/v1",
repo: "CSTMGames/FiveM_Snippets",
extensionId: "cstmgames.fivem_snippets",
checkIntervalMs: 1000 * 60 * 60 * 6, // 6 hours
allowPrerelease: false,
authToken: "", // optional: "your_gitea_token"
};
// ================= UPDATER =================
export async function runUpdater(context: vscode.ExtensionContext) {
// Run once on startup (silent)
await checkForUpdates(context, true);
// Schedule periodic checks
setInterval(() => {
checkForUpdates(context, true);
}, CONFIG.checkIntervalMs);
}
export async function manualUpdateCheck(context: vscode.ExtensionContext) {
await checkForUpdates(context, false);
}
async function checkForUpdates(
context: vscode.ExtensionContext,
silent: boolean
) {
try {
// Read user settings dynamically
const autoInstall = vscode.workspace
.getConfiguration("cstmgames.fivem_snippets")
.get<boolean>("autoUpdate", false);
const now = Date.now();
const lastCheck = context.globalState.get<number>("lastUpdateCheck") || 0;
// Prevent too frequent checks
if (now - lastCheck < CONFIG.checkIntervalMs && silent) return;
context.globalState.update("lastUpdateCheck", now);
// Build Gitea releases URL
const url = `${CONFIG.apiBase}/repos/${CONFIG.repo}/releases`;
// Build headers safely
const headers: Record<string, string> = {};
if (CONFIG.authToken && CONFIG.authToken.length > 0) {
headers["Authorization"] = `token ${CONFIG.authToken}`;
}
logDebug(`[Updater] Checking Gitea releases URL: ${url}`);
logDebug(`[Updater] Headers:`, headers);
const res = await fetch(url, { headers });
logDebug(`[Updater] HTTP status: ${res.status}`);
if (!res.ok) throw new Error(`HTTP ${res.status}`);
const data: any[] = await res.json();
logDebug(`[Updater] Releases data:`, data);
if (!Array.isArray(data) || data.length === 0) {
console.warn("[Updater] No releases found.");
return;
}
// Take the latest release (first in array)
const latestRelease = data[0];
const latestVersion = latestRelease.tag_name.replace(/^v/, "");
const ext = vscode.extensions.getExtension(CONFIG.extensionId);
if (!ext) {
console.warn(`[Updater] Extension ${CONFIG.extensionId} not found`);
return;
}
const currentVersion = ext.packageJSON.version;
logDebug(`[Updater] Current version: ${currentVersion}, Latest version: ${latestVersion}`);
if (latestVersion === currentVersion) {
if (!silent) vscode.window.showInformationMessage("Extension is up to date.");
return;
}
const asset = latestRelease.assets?.find((a: any) => a.name.endsWith(".vsix"));
if (!asset) {
console.warn("[Updater] No VSIX asset found in latest release");
return;
}
logDebug(`[Updater] Latest VSIX URL: ${asset.browser_download_url}`);
if (autoInstall) {
await installVsix(asset.browser_download_url);
return;
}
const choice = await vscode.window.showInformationMessage(
`Update available: ${currentVersion}${latestVersion}`,
"Install",
"Later"
);
if (choice === "Install") {
await installVsix(asset.browser_download_url);
}
} catch (err) {
console.error("[Updater] Error checking updates:", err);
if (!silent) {
vscode.window.showErrorMessage(`Failed to check for updates: ${err}`);
}
}
}
// ================= INSTALL FUNCTION =================
async function installVsix(url: string) {
try {
logDebug(`[Updater] Installing VSIX from URL: ${url}`);
await vscode.commands.executeCommand(
"workbench.extensions.installExtension",
vscode.Uri.parse(url)
);
const reload = await vscode.window.showInformationMessage(
"Extension updated successfully.",
"Reload Now"
);
if (reload === "Reload Now") {
await vscode.commands.executeCommand("workbench.action.reloadWindow");
}
} catch (err) {
console.error("[Updater] Install failed:", err);
vscode.window.showErrorMessage("Failed to install update.");
}
}