import fetch from "node-fetch"; import { getOctokit, context } from "@actions/github"; import { resolveUpdateLog } from "./updatelog.mjs"; // Add stable update JSON filenames const UPDATE_TAG_NAME = "updater"; const UPDATE_JSON_FILE = "update.json"; const UPDATE_JSON_PROXY = "update-proxy.json"; // Add alpha update JSON filenames const ALPHA_TAG_NAME = "updater-alpha"; const ALPHA_UPDATE_JSON_FILE = "update-alpha.json"; const ALPHA_UPDATE_JSON_PROXY = "update-alpha-proxy.json"; /// generate update.json /// upload to update tag's release asset async function resolveUpdater() { if (process.env.GITHUB_TOKEN === undefined) { throw new Error("GITHUB_TOKEN is required"); } const options = { owner: context.repo.owner, repo: context.repo.repo }; const github = getOctokit(process.env.GITHUB_TOKEN); const { data: tags } = await github.rest.repos.listTags({ ...options, per_page: 20, // Increase to have more chances to find alpha tags page: 1, }); // Get the latest stable tag and alpha tag const stableTag = tags.find( (t) => t.name.startsWith("v") && !t.name.includes("alpha"), ); const alphaTag = tags.find((t) => t.name.includes("alpha")); console.log("Stable tag:", stableTag); console.log("Alpha tag:", alphaTag); console.log(); // Process stable release await processRelease(github, options, stableTag, false); // Process alpha release if found if (alphaTag) { await processRelease(github, options, alphaTag, true); } } // Process a release (stable or alpha) and generate update files async function processRelease(github, options, tag, isAlpha) { if (!tag) return; const { data: release } = await github.rest.repos.getReleaseByTag({ ...options, tag: tag.name, }); const updateData = { name: tag.name, notes: await resolveUpdateLog(tag.name).catch( () => "No changelog available", ), pub_date: new Date().toISOString(), platforms: { win64: { signature: "", url: "" }, // compatible with older formats linux: { signature: "", url: "" }, // compatible with older formats darwin: { signature: "", url: "" }, // compatible with older formats "darwin-aarch64": { signature: "", url: "" }, "darwin-intel": { signature: "", url: "" }, "darwin-x86_64": { signature: "", url: "" }, "linux-x86_64": { signature: "", url: "" }, "linux-x86": { signature: "", url: "" }, "linux-i686": { signature: "", url: "" }, "linux-aarch64": { signature: "", url: "" }, "linux-armv7": { signature: "", url: "" }, "windows-x86_64": { signature: "", url: "" }, "windows-aarch64": { signature: "", url: "" }, "windows-x86": { signature: "", url: "" }, "windows-i686": { signature: "", url: "" }, }, }; const promises = release.assets.map(async (asset) => { const { name, browser_download_url } = asset; // Process all the platform URL and signature data // win64 url if (name.endsWith("x64-setup.exe")) { updateData.platforms.win64.url = browser_download_url; updateData.platforms["windows-x86_64"].url = browser_download_url; } // win64 signature if (name.endsWith("x64-setup.exe.sig")) { const sig = await getSignature(browser_download_url); updateData.platforms.win64.signature = sig; updateData.platforms["windows-x86_64"].signature = sig; } // win32 url if (name.endsWith("x86-setup.exe")) { updateData.platforms["windows-x86"].url = browser_download_url; updateData.platforms["windows-i686"].url = browser_download_url; } // win32 signature if (name.endsWith("x86-setup.exe.sig")) { const sig = await getSignature(browser_download_url); updateData.platforms["windows-x86"].signature = sig; updateData.platforms["windows-i686"].signature = sig; } // win arm url if (name.endsWith("arm64-setup.exe")) { updateData.platforms["windows-aarch64"].url = browser_download_url; } // win arm signature if (name.endsWith("arm64-setup.exe.sig")) { const sig = await getSignature(browser_download_url); updateData.platforms["windows-aarch64"].signature = sig; } // darwin url (intel) if (name.endsWith(".app.tar.gz") && !name.includes("aarch")) { updateData.platforms.darwin.url = browser_download_url; updateData.platforms["darwin-intel"].url = browser_download_url; updateData.platforms["darwin-x86_64"].url = browser_download_url; } // darwin signature (intel) if (name.endsWith(".app.tar.gz.sig") && !name.includes("aarch")) { const sig = await getSignature(browser_download_url); updateData.platforms.darwin.signature = sig; updateData.platforms["darwin-intel"].signature = sig; updateData.platforms["darwin-x86_64"].signature = sig; } // darwin url (aarch) if (name.endsWith("aarch64.app.tar.gz")) { updateData.platforms["darwin-aarch64"].url = browser_download_url; // 使linux可以检查更新 updateData.platforms.linux.url = browser_download_url; updateData.platforms["linux-x86_64"].url = browser_download_url; updateData.platforms["linux-x86"].url = browser_download_url; updateData.platforms["linux-i686"].url = browser_download_url; updateData.platforms["linux-aarch64"].url = browser_download_url; updateData.platforms["linux-armv7"].url = browser_download_url; } // darwin signature (aarch) if (name.endsWith("aarch64.app.tar.gz.sig")) { const sig = await getSignature(browser_download_url); updateData.platforms["darwin-aarch64"].signature = sig; updateData.platforms.linux.signature = sig; updateData.platforms["linux-x86_64"].signature = sig; updateData.platforms["linux-x86"].url = browser_download_url; updateData.platforms["linux-i686"].url = browser_download_url; updateData.platforms["linux-aarch64"].signature = sig; updateData.platforms["linux-armv7"].signature = sig; } }); await Promise.allSettled(promises); console.log(updateData); // maybe should test the signature as well // delete the null field Object.entries(updateData.platforms).forEach(([key, value]) => { if (!value.url) { console.log(`[Error]: failed to parse release for "${key}"`); delete updateData.platforms[key]; } }); // Generate a proxy update file for accelerated GitHub resources const updateDataNew = JSON.parse(JSON.stringify(updateData)); Object.entries(updateDataNew.platforms).forEach(([key, value]) => { if (value.url) { updateDataNew.platforms[key].url = "https://download.clashverge.dev/" + value.url; } else { console.log(`[Error]: updateDataNew.platforms.${key} is null`); } }); // Get the appropriate updater release based on isAlpha flag const releaseTag = isAlpha ? ALPHA_TAG_NAME : UPDATE_TAG_NAME; try { const { data: updateRelease } = await github.rest.repos.getReleaseByTag({ ...options, tag: releaseTag, }); // File names based on release type const jsonFile = isAlpha ? ALPHA_UPDATE_JSON_FILE : UPDATE_JSON_FILE; const proxyFile = isAlpha ? ALPHA_UPDATE_JSON_PROXY : UPDATE_JSON_PROXY; // Delete existing assets with these names for (let asset of updateRelease.assets) { if (asset.name === jsonFile) { await github.rest.repos.deleteReleaseAsset({ ...options, asset_id: asset.id, }); } if (asset.name === proxyFile) { await github.rest.repos .deleteReleaseAsset({ ...options, asset_id: asset.id }) .catch(console.error); // do not break the pipeline } } // Upload new assets await github.rest.repos.uploadReleaseAsset({ ...options, release_id: updateRelease.id, name: jsonFile, data: JSON.stringify(updateData, null, 2), }); await github.rest.repos.uploadReleaseAsset({ ...options, release_id: updateRelease.id, name: proxyFile, data: JSON.stringify(updateDataNew, null, 2), }); console.log( `Successfully uploaded ${isAlpha ? "alpha" : "stable"} update files to ${releaseTag}`, ); } catch (error) { console.error( `Failed to process ${isAlpha ? "alpha" : "stable"} release:`, error.message, ); } } // get the signature file content async function getSignature(url) { const response = await fetch(url, { method: "GET", headers: { "Content-Type": "application/octet-stream" }, }); return response.text(); } resolveUpdater().catch(console.error);