123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479 |
- import fs from "fs-extra";
- import zlib from "zlib";
- import tar from "tar";
- import path from "path";
- import AdmZip from "adm-zip";
- import fetch from "node-fetch";
- import proxyAgent from "https-proxy-agent";
- import { execSync } from "child_process";
- const cwd = process.cwd();
- const TEMP_DIR = path.join(cwd, "node_modules/.verge");
- const FORCE = process.argv.includes("--force");
- const PLATFORM_MAP = {
- "x86_64-pc-windows-msvc": "win32",
- "i686-pc-windows-msvc": "win32",
- "aarch64-pc-windows-msvc": "win32",
- "x86_64-apple-darwin": "darwin",
- "aarch64-apple-darwin": "darwin",
- "x86_64-unknown-linux-gnu": "linux",
- "i686-unknown-linux-gnu": "linux",
- "aarch64-unknown-linux-gnu": "linux",
- "armv7-unknown-linux-gnueabihf": "linux",
- "riscv64gc-unknown-linux-gnu": "linux",
- "loongarch64-unknown-linux-gnu": "linux",
- };
- const ARCH_MAP = {
- "x86_64-pc-windows-msvc": "x64",
- "i686-pc-windows-msvc": "ia32",
- "aarch64-pc-windows-msvc": "arm64",
- "x86_64-apple-darwin": "x64",
- "aarch64-apple-darwin": "arm64",
- "x86_64-unknown-linux-gnu": "x64",
- "i686-unknown-linux-gnu": "ia32",
- "aarch64-unknown-linux-gnu": "arm64",
- "armv7-unknown-linux-gnueabihf": "arm",
- "riscv64gc-unknown-linux-gnu": "riscv64",
- "loongarch64-unknown-linux-gnu": "loong64",
- };
- const arg1 = process.argv.slice(2)[0];
- const arg2 = process.argv.slice(2)[1];
- const target = arg1 === "--force" ? arg2 : arg1;
- const { platform, arch } = target
- ? { platform: PLATFORM_MAP[target], arch: ARCH_MAP[target] }
- : process;
- const SIDECAR_HOST = target
- ? target
- : execSync("rustc -vV")
- .toString()
- .match(/(?<=host: ).+(?=\s*)/g)[0];
- /* ======= clash meta alpha======= */
- const META_ALPHA_VERSION_URL =
- "https://github.com/MetaCubeX/mihomo/releases/download/Prerelease-Alpha/version.txt";
- const META_ALPHA_URL_PREFIX = `https://github.com/MetaCubeX/mihomo/releases/download/Prerelease-Alpha`;
- let META_ALPHA_VERSION;
- const META_ALPHA_MAP = {
- "win32-x64": "mihomo-windows-amd64-compatible",
- "win32-ia32": "mihomo-windows-386",
- "win32-arm64": "mihomo-windows-arm64",
- "darwin-x64": "mihomo-darwin-amd64-compatible",
- "darwin-arm64": "mihomo-darwin-arm64",
- "linux-x64": "mihomo-linux-amd64-compatible",
- "linux-ia32": "mihomo-linux-386",
- "linux-arm64": "mihomo-linux-arm64",
- "linux-arm": "mihomo-linux-armv7",
- "linux-riscv64": "mihomo-linux-riscv64",
- "linux-loong64": "mihomo-linux-loong64",
- };
- // Fetch the latest alpha release version from the version.txt file
- async function getLatestAlphaVersion() {
- const options = {};
- const httpProxy =
- process.env.HTTP_PROXY ||
- process.env.http_proxy ||
- process.env.HTTPS_PROXY ||
- process.env.https_proxy;
- if (httpProxy) {
- options.agent = proxyAgent(httpProxy);
- }
- try {
- const response = await fetch(META_ALPHA_VERSION_URL, {
- ...options,
- method: "GET",
- });
- let v = await response.text();
- META_ALPHA_VERSION = v.trim(); // Trim to remove extra whitespaces
- console.log(`Latest alpha version: ${META_ALPHA_VERSION}`);
- } catch (error) {
- console.error("Error fetching latest alpha version:", error.message);
- process.exit(1);
- }
- }
- /* ======= clash meta stable ======= */
- const META_VERSION_URL =
- "https://github.com/MetaCubeX/mihomo/releases/latest/download/version.txt";
- const META_URL_PREFIX = `https://github.com/MetaCubeX/mihomo/releases/download`;
- let META_VERSION;
- const META_MAP = {
- "win32-x64": "mihomo-windows-amd64-compatible",
- "win32-ia32": "mihomo-windows-386",
- "win32-arm64": "mihomo-windows-arm64",
- "darwin-x64": "mihomo-darwin-amd64-compatible",
- "darwin-arm64": "mihomo-darwin-arm64",
- "linux-x64": "mihomo-linux-amd64-compatible",
- "linux-ia32": "mihomo-linux-386",
- "linux-arm64": "mihomo-linux-arm64",
- "linux-arm": "mihomo-linux-armv7",
- "linux-riscv64": "mihomo-linux-riscv64",
- "linux-loong64": "mihomo-linux-loong64",
- };
- // Fetch the latest release version from the version.txt file
- async function getLatestReleaseVersion() {
- const options = {};
- const httpProxy =
- process.env.HTTP_PROXY ||
- process.env.http_proxy ||
- process.env.HTTPS_PROXY ||
- process.env.https_proxy;
- if (httpProxy) {
- options.agent = proxyAgent(httpProxy);
- }
- try {
- const response = await fetch(META_VERSION_URL, {
- ...options,
- method: "GET",
- });
- let v = await response.text();
- META_VERSION = v.trim(); // Trim to remove extra whitespaces
- console.log(`Latest release version: ${META_VERSION}`);
- } catch (error) {
- console.error("Error fetching latest release version:", error.message);
- process.exit(1);
- }
- }
- /*
- * check available
- */
- if (!META_MAP[`${platform}-${arch}`]) {
- throw new Error(
- `clash meta alpha unsupported platform "${platform}-${arch}"`
- );
- }
- if (!META_ALPHA_MAP[`${platform}-${arch}`]) {
- throw new Error(
- `clash meta alpha unsupported platform "${platform}-${arch}"`
- );
- }
- /**
- * core info
- */
- function clashMetaAlpha() {
- const name = META_ALPHA_MAP[`${platform}-${arch}`];
- const isWin = platform === "win32";
- const urlExt = isWin ? "zip" : "gz";
- const downloadURL = `${META_ALPHA_URL_PREFIX}/${name}-${META_ALPHA_VERSION}.${urlExt}`;
- const exeFile = `${name}${isWin ? ".exe" : ""}`;
- const zipFile = `${name}-${META_ALPHA_VERSION}.${urlExt}`;
- return {
- name: "verge-mihomo-alpha",
- targetFile: `verge-mihomo-alpha-${SIDECAR_HOST}${isWin ? ".exe" : ""}`,
- exeFile,
- zipFile,
- downloadURL,
- };
- }
- function clashMeta() {
- const name = META_MAP[`${platform}-${arch}`];
- const isWin = platform === "win32";
- const urlExt = isWin ? "zip" : "gz";
- const downloadURL = `${META_URL_PREFIX}/${META_VERSION}/${name}-${META_VERSION}.${urlExt}`;
- const exeFile = `${name}${isWin ? ".exe" : ""}`;
- const zipFile = `${name}-${META_VERSION}.${urlExt}`;
- return {
- name: "verge-mihomo",
- targetFile: `verge-mihomo-${SIDECAR_HOST}${isWin ? ".exe" : ""}`,
- exeFile,
- zipFile,
- downloadURL,
- };
- }
- /**
- * download sidecar and rename
- */
- async function resolveSidecar(binInfo) {
- const { name, targetFile, zipFile, exeFile, downloadURL } = binInfo;
- const sidecarDir = path.join(cwd, "src-tauri", "sidecar");
- const sidecarPath = path.join(sidecarDir, targetFile);
- await fs.mkdirp(sidecarDir);
- if (!FORCE && (await fs.pathExists(sidecarPath))) return;
- const tempDir = path.join(TEMP_DIR, name);
- const tempZip = path.join(tempDir, zipFile);
- const tempExe = path.join(tempDir, exeFile);
- await fs.mkdirp(tempDir);
- try {
- if (!(await fs.pathExists(tempZip))) {
- await downloadFile(downloadURL, tempZip);
- }
- if (zipFile.endsWith(".zip")) {
- const zip = new AdmZip(tempZip);
- zip.getEntries().forEach((entry) => {
- console.log(`[DEBUG]: "${name}" entry name`, entry.entryName);
- });
- zip.extractAllTo(tempDir, true);
- await fs.rename(tempExe, sidecarPath);
- console.log(`[INFO]: "${name}" unzip finished`);
- } else if (zipFile.endsWith(".tgz")) {
- // tgz
- await fs.mkdirp(tempDir);
- await tar.extract({
- cwd: tempDir,
- file: tempZip,
- //strip: 1, // 可能需要根据实际的 .tgz 文件结构调整
- });
- const files = await fs.readdir(tempDir);
- console.log(`[DEBUG]: "${name}" files in tempDir:`, files);
- const extractedFile = files.find((file) => file.startsWith("虚空终端-"));
- if (extractedFile) {
- const extractedFilePath = path.join(tempDir, extractedFile);
- await fs.rename(extractedFilePath, sidecarPath);
- console.log(`[INFO]: "${name}" file renamed to "${sidecarPath}"`);
- execSync(`chmod 755 ${sidecarPath}`);
- console.log(`[INFO]: "${name}" chmod binary finished`);
- } else {
- throw new Error(`Expected file not found in ${tempDir}`);
- }
- } else {
- // gz
- const readStream = fs.createReadStream(tempZip);
- const writeStream = fs.createWriteStream(sidecarPath);
- await new Promise((resolve, reject) => {
- const onError = (error) => {
- console.error(`[ERROR]: "${name}" gz failed:`, error.message);
- reject(error);
- };
- readStream
- .pipe(zlib.createGunzip().on("error", onError))
- .pipe(writeStream)
- .on("finish", () => {
- console.log(`[INFO]: "${name}" gunzip finished`);
- execSync(`chmod 755 ${sidecarPath}`);
- console.log(`[INFO]: "${name}" chmod binary finished`);
- resolve();
- })
- .on("error", onError);
- });
- }
- } catch (err) {
- // 需要删除文件
- await fs.remove(sidecarPath);
- throw err;
- } finally {
- // delete temp dir
- await fs.remove(tempDir);
- }
- }
- /**
- * download the file to the resources dir
- */
- async function resolveResource(binInfo) {
- const { file, downloadURL } = binInfo;
- const resDir = path.join(cwd, "src-tauri/resources");
- const targetPath = path.join(resDir, file);
- if (!FORCE && (await fs.pathExists(targetPath))) return;
- await fs.mkdirp(resDir);
- await downloadFile(downloadURL, targetPath);
- console.log(`[INFO]: ${file} finished`);
- }
- /**
- * download file and save to `path`
- */
- async function downloadFile(url, path) {
- const options = {};
- const httpProxy =
- process.env.HTTP_PROXY ||
- process.env.http_proxy ||
- process.env.HTTPS_PROXY ||
- process.env.https_proxy;
- if (httpProxy) {
- options.agent = proxyAgent(httpProxy);
- }
- const response = await fetch(url, {
- ...options,
- method: "GET",
- headers: { "Content-Type": "application/octet-stream" },
- });
- const buffer = await response.arrayBuffer();
- await fs.writeFile(path, new Uint8Array(buffer));
- console.log(`[INFO]: download finished "${url}"`);
- }
- // SimpleSC.dll
- const resolvePlugin = async () => {
- const url =
- "https://nsis.sourceforge.io/mediawiki/images/e/ef/NSIS_Simple_Service_Plugin_Unicode_1.30.zip";
- const tempDir = path.join(TEMP_DIR, "SimpleSC");
- const tempZip = path.join(
- tempDir,
- "NSIS_Simple_Service_Plugin_Unicode_1.30.zip"
- );
- const tempDll = path.join(tempDir, "SimpleSC.dll");
- const pluginDir = path.join(process.env.APPDATA, "Local/NSIS");
- const pluginPath = path.join(pluginDir, "SimpleSC.dll");
- await fs.mkdirp(pluginDir);
- await fs.mkdirp(tempDir);
- if (!FORCE && (await fs.pathExists(pluginPath))) return;
- try {
- if (!(await fs.pathExists(tempZip))) {
- await downloadFile(url, tempZip);
- }
- const zip = new AdmZip(tempZip);
- zip.getEntries().forEach((entry) => {
- console.log(`[DEBUG]: "SimpleSC" entry name`, entry.entryName);
- });
- zip.extractAllTo(tempDir, true);
- await fs.copyFile(tempDll, pluginPath);
- console.log(`[INFO]: "SimpleSC" unzip finished`);
- } finally {
- await fs.remove(tempDir);
- }
- };
- // service chmod
- const resolveServicePermission = async () => {
- const serviceExecutables = [
- "clash-verge-service",
- "install-service",
- "uninstall-service",
- ];
- const resDir = path.join(cwd, "src-tauri/resources");
- for (let f of serviceExecutables) {
- const targetPath = path.join(resDir, f);
- if (await fs.pathExists(targetPath)) {
- execSync(`chmod 755 ${targetPath}`);
- console.log(`[INFO]: "${targetPath}" chmod finished`);
- }
- }
- };
- /**
- * main
- */
- const SERVICE_URL = `https://github.com/clash-verge-rev/clash-verge-service/releases/download/${SIDECAR_HOST}`;
- const resolveService = () => {
- let ext = platform === "win32" ? ".exe" : "";
- resolveResource({
- file: "clash-verge-service" + ext,
- downloadURL: `${SERVICE_URL}/clash-verge-service${ext}`,
- });
- };
- const resolveInstall = () => {
- let ext = platform === "win32" ? ".exe" : "";
- resolveResource({
- file: "install-service" + ext,
- downloadURL: `${SERVICE_URL}/install-service${ext}`,
- });
- };
- const resolveUninstall = () => {
- let ext = platform === "win32" ? ".exe" : "";
- resolveResource({
- file: "uninstall-service" + ext,
- downloadURL: `${SERVICE_URL}/uninstall-service${ext}`,
- });
- };
- const resolveMmdb = () =>
- resolveResource({
- file: "Country.mmdb",
- downloadURL: `https://github.com/MetaCubeX/meta-rules-dat/releases/download/latest/country.mmdb`,
- });
- const resolveGeosite = () =>
- resolveResource({
- file: "geosite.dat",
- downloadURL: `https://github.com/MetaCubeX/meta-rules-dat/releases/download/latest/geosite.dat`,
- });
- const resolveGeoIP = () =>
- resolveResource({
- file: "geoip.dat",
- downloadURL: `https://github.com/MetaCubeX/meta-rules-dat/releases/download/latest/geoip.dat`,
- });
- const resolveEnableLoopback = () =>
- resolveResource({
- file: "enableLoopback.exe",
- downloadURL: `https://github.com/Kuingsmile/uwp-tool/releases/download/latest/enableLoopback.exe`,
- });
- const tasks = [
- // { name: "clash", func: resolveClash, retry: 5 },
- {
- name: "verge-mihomo-alpha",
- func: () =>
- getLatestAlphaVersion().then(() => resolveSidecar(clashMetaAlpha())),
- retry: 5,
- },
- {
- name: "verge-mihomo",
- func: () =>
- getLatestReleaseVersion().then(() => resolveSidecar(clashMeta())),
- retry: 5,
- },
- { name: "plugin", func: resolvePlugin, retry: 5, winOnly: true },
- { name: "service", func: resolveService, retry: 5 },
- { name: "install", func: resolveInstall, retry: 5 },
- { name: "uninstall", func: resolveUninstall, retry: 5 },
- { name: "mmdb", func: resolveMmdb, retry: 5 },
- { name: "geosite", func: resolveGeosite, retry: 5 },
- { name: "geoip", func: resolveGeoIP, retry: 5 },
- {
- name: "enableLoopback",
- func: resolveEnableLoopback,
- retry: 5,
- winOnly: true,
- },
- {
- name: "service_chmod",
- func: resolveServicePermission,
- retry: 1,
- unixOnly: true,
- },
- ];
- async function runTask() {
- const task = tasks.shift();
- if (!task) return;
- if (task.winOnly && platform !== "win32") return runTask();
- if (task.linuxOnly && platform !== "linux") return runTask();
- if (task.unixOnly && platform === "win32") return runTask();
- for (let i = 0; i < task.retry; i++) {
- try {
- await task.func();
- break;
- } catch (err) {
- console.error(`[ERROR]: task::${task.name} try ${i} ==`, err.message);
- if (i === task.retry - 1) throw err;
- }
- }
- return runTask();
- }
- runTask();
- runTask();
|