mirror of
https://github.com/astral-sh/setup-uv.git
synced 2025-12-17 11:05:59 +00:00
254 lines
7.4 KiB
TypeScript
254 lines
7.4 KiB
TypeScript
import { promises as fs } from "node:fs";
|
|
import * as path from "node:path";
|
|
import * as core from "@actions/core";
|
|
import * as tc from "@actions/tool-cache";
|
|
import * as pep440 from "@renovatebot/pep440";
|
|
import { OWNER, REPO, TOOL_CACHE_NAME } from "../utils/constants";
|
|
import { Octokit } from "../utils/octokit";
|
|
import type { Architecture, Platform } from "../utils/platforms";
|
|
import { validateChecksum } from "./checksum/checksum";
|
|
import {
|
|
getDownloadUrl,
|
|
getLatestKnownVersion as getLatestVersionInManifest,
|
|
} from "./version-manifest";
|
|
|
|
export function tryGetFromToolCache(
|
|
arch: Architecture,
|
|
version: string,
|
|
): { version: string; installedPath: string | undefined } {
|
|
core.debug(`Trying to get uv from tool cache for ${version}...`);
|
|
const cachedVersions = tc.findAllVersions(TOOL_CACHE_NAME, arch);
|
|
core.debug(`Cached versions: ${cachedVersions}`);
|
|
let resolvedVersion = tc.evaluateVersions(cachedVersions, version);
|
|
if (resolvedVersion === "") {
|
|
resolvedVersion = version;
|
|
}
|
|
const installedPath = tc.find(TOOL_CACHE_NAME, resolvedVersion, arch);
|
|
return { installedPath, version: resolvedVersion };
|
|
}
|
|
|
|
export async function downloadVersionFromGithub(
|
|
serverUrl: string,
|
|
platform: Platform,
|
|
arch: Architecture,
|
|
version: string,
|
|
checkSum: string | undefined,
|
|
githubToken: string,
|
|
): Promise<{ version: string; cachedToolDir: string }> {
|
|
const artifact = `uv-${arch}-${platform}`;
|
|
const extension = getExtension(platform);
|
|
const downloadUrl = `${serverUrl}/${OWNER}/${REPO}/releases/download/${version}/${artifact}${extension}`;
|
|
return await downloadVersion(
|
|
downloadUrl,
|
|
artifact,
|
|
platform,
|
|
arch,
|
|
version,
|
|
checkSum,
|
|
githubToken,
|
|
);
|
|
}
|
|
|
|
export async function downloadVersionFromManifest(
|
|
manifestUrl: string | undefined,
|
|
platform: Platform,
|
|
arch: Architecture,
|
|
version: string,
|
|
checkSum: string | undefined,
|
|
githubToken: string,
|
|
): Promise<{ version: string; cachedToolDir: string }> {
|
|
const downloadUrl = await getDownloadUrl(
|
|
manifestUrl,
|
|
version,
|
|
arch,
|
|
platform,
|
|
);
|
|
if (!downloadUrl) {
|
|
core.info(
|
|
`manifest-file does not contain version ${version}, arch ${arch}, platform ${platform}. Falling back to GitHub releases.`,
|
|
);
|
|
return await downloadVersionFromGithub(
|
|
"https://github.com",
|
|
platform,
|
|
arch,
|
|
version,
|
|
checkSum,
|
|
githubToken,
|
|
);
|
|
}
|
|
return await downloadVersion(
|
|
downloadUrl,
|
|
`uv-${arch}-${platform}`,
|
|
platform,
|
|
arch,
|
|
version,
|
|
checkSum,
|
|
githubToken,
|
|
);
|
|
}
|
|
|
|
async function downloadVersion(
|
|
downloadUrl: string,
|
|
artifactName: string,
|
|
platform: Platform,
|
|
arch: Architecture,
|
|
version: string,
|
|
checkSum: string | undefined,
|
|
githubToken: string,
|
|
): Promise<{ version: string; cachedToolDir: string }> {
|
|
core.info(`Downloading uv from "${downloadUrl}" ...`);
|
|
const downloadPath = await tc.downloadTool(
|
|
downloadUrl,
|
|
undefined,
|
|
githubToken,
|
|
);
|
|
await validateChecksum(checkSum, downloadPath, arch, platform, version);
|
|
|
|
let uvDir: string;
|
|
const extension = getExtension(platform);
|
|
if (platform === "pc-windows-msvc") {
|
|
const fullPathWithExtension = `${downloadPath}${extension}`;
|
|
await fs.copyFile(downloadPath, fullPathWithExtension);
|
|
uvDir = await tc.extractZip(fullPathWithExtension);
|
|
// On windows extracting the zip does not create an intermediate directory
|
|
} else {
|
|
const extractedDir = await tc.extractTar(downloadPath);
|
|
uvDir = path.join(extractedDir, artifactName);
|
|
}
|
|
const cachedToolDir = await tc.cacheDir(
|
|
uvDir,
|
|
TOOL_CACHE_NAME,
|
|
version,
|
|
arch,
|
|
);
|
|
return { cachedToolDir, version: version };
|
|
}
|
|
|
|
function getExtension(platform: Platform): string {
|
|
return platform === "pc-windows-msvc" ? ".zip" : ".tar.gz";
|
|
}
|
|
|
|
export async function resolveVersion(
|
|
versionInput: string,
|
|
manifestFile: string | undefined,
|
|
githubToken: string,
|
|
): Promise<string> {
|
|
core.debug(`Resolving version: ${versionInput}`);
|
|
let version: string;
|
|
if (manifestFile) {
|
|
version =
|
|
versionInput === "latest"
|
|
? await getLatestVersionInManifest(manifestFile)
|
|
: versionInput;
|
|
} else {
|
|
version =
|
|
versionInput === "latest"
|
|
? await getLatestVersion(githubToken)
|
|
: versionInput;
|
|
}
|
|
if (tc.isExplicitVersion(version)) {
|
|
core.debug(`Version ${version} is an explicit version.`);
|
|
return version;
|
|
}
|
|
const availableVersions = await getAvailableVersions(githubToken);
|
|
core.debug(`Available versions: ${availableVersions}`);
|
|
const resolvedVersion = maxSatisfying(availableVersions, version);
|
|
if (resolvedVersion === undefined) {
|
|
throw new Error(`No version found for ${version}`);
|
|
}
|
|
return resolvedVersion;
|
|
}
|
|
|
|
async function getAvailableVersions(githubToken: string): Promise<string[]> {
|
|
core.info("Getting available versions from GitHub API...");
|
|
try {
|
|
const octokit = new Octokit({
|
|
auth: githubToken,
|
|
});
|
|
return await getReleaseTagNames(octokit);
|
|
} catch (err) {
|
|
if ((err as Error).message.includes("Bad credentials")) {
|
|
core.info(
|
|
"No (valid) GitHub token provided. Falling back to anonymous. Requests might be rate limited.",
|
|
);
|
|
const octokit = new Octokit();
|
|
return await getReleaseTagNames(octokit);
|
|
}
|
|
throw err;
|
|
}
|
|
}
|
|
|
|
async function getReleaseTagNames(
|
|
octokit: InstanceType<typeof Octokit>,
|
|
): Promise<string[]> {
|
|
const response = await octokit.paginate(octokit.rest.repos.listReleases, {
|
|
owner: OWNER,
|
|
repo: REPO,
|
|
});
|
|
const releaseTagNames = response.map((release) => release.tag_name);
|
|
if (releaseTagNames.length === 0) {
|
|
throw Error(
|
|
"Github API request failed while getting releases. Check the GitHub status page for outages. Try again later.",
|
|
);
|
|
}
|
|
return releaseTagNames;
|
|
}
|
|
|
|
async function getLatestVersion(githubToken: string) {
|
|
core.info("Getting latest version from GitHub API...");
|
|
const octokit = new Octokit({
|
|
auth: githubToken,
|
|
});
|
|
|
|
let latestRelease: { tag_name: string } | undefined;
|
|
try {
|
|
latestRelease = await getLatestRelease(octokit);
|
|
} catch (err) {
|
|
if ((err as Error).message.includes("Bad credentials")) {
|
|
core.info(
|
|
"No (valid) GitHub token provided. Falling back to anonymous. Requests might be rate limited.",
|
|
);
|
|
const octokit = new Octokit();
|
|
latestRelease = await getLatestRelease(octokit);
|
|
} else {
|
|
core.error(
|
|
"Github API request failed while getting latest release. Check the GitHub status page for outages. Try again later.",
|
|
);
|
|
throw err;
|
|
}
|
|
}
|
|
|
|
if (!latestRelease) {
|
|
throw new Error("Could not determine latest release.");
|
|
}
|
|
core.debug(`Latest version: ${latestRelease.tag_name}`);
|
|
return latestRelease.tag_name;
|
|
}
|
|
|
|
async function getLatestRelease(octokit: InstanceType<typeof Octokit>) {
|
|
const { data: latestRelease } = await octokit.rest.repos.getLatestRelease({
|
|
owner: OWNER,
|
|
repo: REPO,
|
|
});
|
|
return latestRelease;
|
|
}
|
|
|
|
function maxSatisfying(
|
|
versions: string[],
|
|
version: string,
|
|
): string | undefined {
|
|
const maxSemver = tc.evaluateVersions(versions, version);
|
|
if (maxSemver !== "") {
|
|
core.debug(`Found a version that satisfies the semver range: ${maxSemver}`);
|
|
return maxSemver;
|
|
}
|
|
const maxPep440 = pep440.maxSatisfying(versions, version);
|
|
if (maxPep440 !== null) {
|
|
core.debug(
|
|
`Found a version that satisfies the pep440 specifier: ${maxPep440}`,
|
|
);
|
|
return maxPep440;
|
|
}
|
|
return undefined;
|
|
}
|