UNPKG

next

Version:

The React Framework

140 lines (139 loc) 5.87 kB
import os from "os"; import fs from "fs"; import path from "path"; import * as Log from "../build/output/log"; import tar from "next/dist/compiled/tar"; const { fetch } = require("next/dist/compiled/undici"); const { WritableStream } = require("node:stream/web"); import { fileExists } from "./file-exists"; import { getRegistry } from "./helpers/get-registry"; const MAX_VERSIONS_TO_CACHE = 8; // get platform specific cache directory adapted from playwright's handling // https://github.com/microsoft/playwright/blob/7d924470d397975a74a19184c136b3573a974e13/packages/playwright-core/src/utils/registry.ts#L141 async function getCacheDirectory() { let result; const envDefined = process.env["NEXT_SWC_PATH"]; if (envDefined) { result = envDefined; } else { let systemCacheDirectory; if (process.platform === "linux") { systemCacheDirectory = process.env.XDG_CACHE_HOME || path.join(os.homedir(), ".cache"); } else if (process.platform === "darwin") { systemCacheDirectory = path.join(os.homedir(), "Library", "Caches"); } else if (process.platform === "win32") { systemCacheDirectory = process.env.LOCALAPPDATA || path.join(os.homedir(), "AppData", "Local"); } else { /// Attempt to use generic tmp location for un-handled platform if (!systemCacheDirectory) { for (const dir of [ path.join(os.homedir(), ".cache"), path.join(os.tmpdir()) ]){ if (await fileExists(dir)) { systemCacheDirectory = dir; break; } } } if (!systemCacheDirectory) { console.error(new Error("Unsupported platform: " + process.platform)); process.exit(0); } } result = path.join(systemCacheDirectory, "next-swc"); } if (!path.isAbsolute(result)) { // It is important to resolve to the absolute path: // - for unzipping to work correctly; // - so that registry directory matches between installation and execution. // INIT_CWD points to the root of `npm/yarn install` and is probably what // the user meant when typing the relative path. result = path.resolve(process.env["INIT_CWD"] || process.cwd(), result); } return result; } async function extractBinary(outputDirectory, pkgName, tarFileName) { const cacheDirectory = await getCacheDirectory(); const extractFromTar = async ()=>{ await tar.x({ file: path.join(cacheDirectory, tarFileName), cwd: outputDirectory, strip: 1 }); }; if (!await fileExists(path.join(cacheDirectory, tarFileName))) { Log.info(`Downloading swc package ${pkgName}...`); await fs.promises.mkdir(cacheDirectory, { recursive: true }); const tempFile = path.join(cacheDirectory, `${tarFileName}.temp-${Date.now()}`); const registry = getRegistry(); const downloadUrl = `${registry}${pkgName}/-/${tarFileName}`; await fetch(downloadUrl).then((res)=>{ const { ok , body } = res; if (!ok || !body) { Log.error(`Failed to download swc package from ${downloadUrl}`); } if (!ok) { throw new Error(`request failed with status ${res.status}`); } if (!body) { throw new Error("request failed with empty body"); } const cacheWriteStream = fs.createWriteStream(tempFile); return body.pipeTo(new WritableStream({ write (chunk) { cacheWriteStream.write(chunk); }, close () { cacheWriteStream.close(); } })); }); await fs.promises.rename(tempFile, path.join(cacheDirectory, tarFileName)); } await extractFromTar(); const cacheFiles = await fs.promises.readdir(cacheDirectory); if (cacheFiles.length > MAX_VERSIONS_TO_CACHE) { cacheFiles.sort((a, b)=>{ if (a.length < b.length) return -1; return a.localeCompare(b); }); // prune oldest versions in cache for(let i = 0; i++; i < cacheFiles.length - MAX_VERSIONS_TO_CACHE){ await fs.promises.unlink(path.join(cacheDirectory, cacheFiles[i])).catch(()=>{}); } } } export async function downloadNativeNextSwc(version, bindingsDirectory, triplesABI) { for (const triple of triplesABI){ const pkgName = `@next/swc-${triple}`; const tarFileName = `${pkgName.substring(6)}-${version}.tgz`; const outputDirectory = path.join(bindingsDirectory, pkgName); if (await fileExists(outputDirectory)) { // if the package is already downloaded a different // failure occurred than not being present return; } await fs.promises.mkdir(outputDirectory, { recursive: true }); await extractBinary(outputDirectory, pkgName, tarFileName); } } export async function downloadWasmSwc(version, wasmDirectory, variant = "nodejs") { const pkgName = `@next/swc-wasm-${variant}`; const tarFileName = `${pkgName.substring(6)}-${version}.tgz`; const outputDirectory = path.join(wasmDirectory, pkgName); if (await fileExists(outputDirectory)) { // if the package is already downloaded a different // failure occurred than not being present return; } await fs.promises.mkdir(outputDirectory, { recursive: true }); await extractBinary(outputDirectory, pkgName, tarFileName); } //# sourceMappingURL=download-swc.js.map