mirror of
https://github.com/astral-sh/setup-uv.git
synced 2026-03-13 21:35:08 +00:00
Some checks failed
test / test-act (push) Failing after 18s
Release Drafter / ✏️ Draft release (push) Waiting to run
test / test-tool-install (macos-14) (push) Waiting to run
test / test-tool-install (macos-latest) (push) Waiting to run
test / test-tool-install (windows-latest) (push) Waiting to run
test / test-cache-key-os-version (macos-14, macos-14) (push) Waiting to run
test / test-default-version (macos-14) (push) Waiting to run
test / test-default-version (macos-latest) (push) Waiting to run
test / test-default-version (windows-latest) (push) Waiting to run
test / test-checksum (map[checksum:a70cbfbf3bb5c08b2f84963b4f12c94e08fbb2468ba418a3bfe1066fbe9e7218 os:macos-latest]) (push) Waiting to run
test / test-python-version (macos-latest) (push) Waiting to run
test / test-activate-environment (windows-latest) (push) Waiting to run
test / test-cache-key-os-version (windows-2022, windows-2022) (push) Waiting to run
test / test-python-version (windows-latest) (push) Waiting to run
test / test-activate-environment (macos-latest) (push) Waiting to run
test / test-cache-key-os-version (macos-15, macos-15) (push) Waiting to run
test / test-activate-environment-custom-path (macos-latest) (push) Waiting to run
test / test-activate-environment-custom-path (windows-latest) (push) Waiting to run
test / test-cache-key-os-version (ubuntu-24.04, ubuntu-24.04) (push) Waiting to run
test / test-cache-key-os-version (windows-2025, windows-2025) (push) Waiting to run
test / test-setup-cache (auto, windows-latest) (push) Waiting to run
test / test-setup-cache (false, windows-latest) (push) Waiting to run
test / test-setup-cache (true, windows-latest) (push) Waiting to run
test / test-restore-cache (auto, ubuntu-latest) (push) Blocked by required conditions
test / test-restore-cache (auto, windows-latest) (push) Blocked by required conditions
test / test-restore-cache (false, ubuntu-latest) (push) Blocked by required conditions
test / test-restore-cache (false, windows-latest) (push) Blocked by required conditions
test / test-cache-local (map[expected-cache-dir:D:\a\_temp\setup-uv-cache os:windows-latest]) (push) Waiting to run
test / test-python-install-dir (map[expected-python-dir:D:\a\_temp\uv-python-dir os:windows-latest]) (push) Waiting to run
test / all-tests-passed (push) Blocked by required conditions
test / test-restore-cache (true, ubuntu-latest) (push) Blocked by required conditions
test / test-restore-cache (true, windows-latest) (push) Blocked by required conditions
CodeQL / Analyze (TypeScript) (push) Failing after 6s
test / lint (push) Failing after 6s
test / test-default-version (ubuntu-latest) (push) Failing after 6s
test / test-uv-no-modify-path (push) Failing after 10s
test / test-specific-version (map[expected-version:0.1.0 resolution-strategy:lowest version-input:>=0.1.0,<0.2]) (push) Failing after 10s
test / test-specific-version (map[expected-version:0.1.45 resolution-strategy:highest version-input:>=0.1,<0.2]) (push) Failing after 10s
test / test-specific-version (map[expected-version:0.3.0 version-input:0.3.0]) (push) Failing after 10s
test / test-specific-version (map[expected-version:0.3.2 version-input:0.3.2]) (push) Failing after 10s
test / test-specific-version (map[expected-version:0.3.5 version-input:0.3.x]) (push) Failing after 12s
test / test-specific-version (map[expected-version:0.3.5 version-input:0.3]) (push) Failing after 12s
test / test-specific-version (map[expected-version:0.4.25 resolution-strategy:lowest version-input:>=0.4.25,<0.5]) (push) Failing after 12s
test / test-specific-version (map[expected-version:0.4.25 resolution-strategy:lowest version-input:>=0.4.25]) (push) Failing after 12s
test / test-specific-version (map[expected-version:0.4.30 version-input:>=0.4.25,<0.5]) (push) Failing after 12s
test / test-latest-version (>=0.8) (push) Failing after 12s
test / test-latest-version (latest) (push) Failing after 13s
test / test-from-working-directory-version (map[expected-version:0.5.14 working-directory:__tests__/fixtures/pyproject-toml-project]) (push) Failing after 18s
test / test-from-working-directory-version (map[expected-version:0.5.15 working-directory:__tests__/fixtures/uv-toml-project]) (push) Failing after 18s
test / test-version-file-version (map[expected-version:0.5.15 version-file:__tests__/fixtures/.tool-versions]) (push) Failing after 18s
test / test-version-file-version (map[expected-version:0.6.17 version-file:__tests__/fixtures/uv-in-requirements-txt-project/requirements.txt]) (push) Failing after 18s
test / test-version-file-version (map[expected-version:0.8.3 version-file:__tests__/fixtures/uv-in-requirements-hash-txt-project/requirements.txt]) (push) Failing after 18s
test / test-malformed-pyproject-file-fallback (push) Failing after 17s
test / test-checksum (map[checksum:4d9279ad5ca596b1e2d703901d508430eb07564dc4d8837de9e2fca9c90f8ecd os:ubuntu-latest]) (push) Failing after 18s
test / test-with-explicit-token (push) Failing after 17s
test / test-uvx (push) Failing after 19s
test / test-tool-install (ubuntu-latest) (push) Failing after 20s
test / test-python-version (ubuntu-latest) (push) Failing after 20s
test / test-activate-environment (ubuntu-latest) (push) Failing after 20s
test / test-activate-environment-custom-path (ubuntu-latest) (push) Failing after 20s
test / test-cache-key-os-version (ubuntu-22.04, ubuntu-22.04) (push) Failing after 18s
test / test-setup-cache (auto, ubuntu-latest) (push) Failing after 18s
test / test-setup-cache (false, ubuntu-latest) (push) Failing after 18s
test / test-setup-cache (true, ubuntu-latest) (push) Failing after 18s
test / test-restore-cache-requirements-txt (push) Has been skipped
test / test-setup-cache-dependency-glob (push) Failing after 18s
test / test-setup-cache-requirements-txt (push) Failing after 18s
test / test-restore-cache-dependency-glob (push) Has been skipped
test / test-setup-cache-save-cache-false (push) Failing after 18s
test / test-restore-cache-save-cache-false (push) Has been skipped
test / test-setup-cache-restore-cache-false (push) Failing after 17s
test / test-restore-cache-restore-cache-false (push) Has been skipped
test / test-cache-local (map[expected-cache-dir:/home/runner/work/_temp/setup-uv-cache os:ubuntu-latest]) (push) Failing after 17s
test / test-cache-local-cache-disabled (push) Failing after 17s
test / test-cache-local-cache-disabled-but-explicit-path (push) Failing after 17s
test / test-no-python-version (push) Failing after 17s
test / test-debian-unstable (push) Failing after 31s
test / test-musl (push) Failing after 30s
test / test-custom-manifest-file (push) Failing after 18s
test / test-absolute-path (push) Failing after 18s
test / test-relative-path (push) Failing after 18s
test / test-cache-prune-force (push) Failing after 18s
test / test-cache-dir-from-file (push) Failing after 18s
test / test-cache-python-missing-managed-install-dir (push) Failing after 19s
test / test-cache-python-installs (push) Failing after 19s
test / test-python-install-dir (map[expected-python-dir:/home/runner/work/_temp/uv-python-dir os:ubuntu-latest]) (push) Failing after 19s
test / test-restore-python-installs (push) Has been skipped
test / validate-typings (push) Failing after 21s
Closes: #777 Closes: #325
257 lines
7.8 KiB
TypeScript
257 lines
7.8 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 * as semver from "semver";
|
|
import { TOOL_CACHE_NAME, VERSIONS_NDJSON_URL } from "../utils/constants";
|
|
import type { Architecture, Platform } from "../utils/platforms";
|
|
import { validateChecksum } from "./checksum/checksum";
|
|
import {
|
|
getAllVersions as getAllManifestVersions,
|
|
getLatestKnownVersion as getLatestVersionInManifest,
|
|
getManifestArtifact,
|
|
} from "./version-manifest";
|
|
import {
|
|
getAllVersions as getAllVersionsFromNdjson,
|
|
getArtifact as getArtifactFromNdjson,
|
|
getLatestVersion as getLatestVersionFromNdjson,
|
|
} from "./versions-client";
|
|
|
|
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 downloadVersionFromNdjson(
|
|
platform: Platform,
|
|
arch: Architecture,
|
|
version: string,
|
|
checkSum: string | undefined,
|
|
githubToken: string,
|
|
): Promise<{ version: string; cachedToolDir: string }> {
|
|
const artifact = await getArtifactFromNdjson(version, arch, platform);
|
|
|
|
if (!artifact) {
|
|
throw new Error(
|
|
`Could not find artifact for version ${version}, arch ${arch}, platform ${platform} in ${VERSIONS_NDJSON_URL} .`,
|
|
);
|
|
}
|
|
|
|
// For the default astral-sh/versions source, checksum validation relies on
|
|
// user input or the built-in KNOWN_CHECKSUMS table, not NDJSON sha256 values.
|
|
return await downloadVersion(
|
|
artifact.url,
|
|
`uv-${arch}-${platform}`,
|
|
platform,
|
|
arch,
|
|
version,
|
|
checkSum,
|
|
githubToken,
|
|
);
|
|
}
|
|
|
|
export async function downloadVersionFromManifest(
|
|
manifestUrl: string,
|
|
platform: Platform,
|
|
arch: Architecture,
|
|
version: string,
|
|
checkSum: string | undefined,
|
|
githubToken: string,
|
|
): Promise<{ version: string; cachedToolDir: string }> {
|
|
const artifact = await getManifestArtifact(
|
|
manifestUrl,
|
|
version,
|
|
arch,
|
|
platform,
|
|
);
|
|
if (!artifact) {
|
|
throw new Error(
|
|
`manifest-file does not contain version ${version}, arch ${arch}, platform ${platform}.`,
|
|
);
|
|
}
|
|
|
|
return await downloadVersion(
|
|
artifact.downloadUrl,
|
|
`uv-${arch}-${platform}`,
|
|
platform,
|
|
arch,
|
|
version,
|
|
resolveChecksum(checkSum, artifact.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;
|
|
if (platform === "pc-windows-msvc") {
|
|
// On windows extracting the zip does not create an intermediate directory.
|
|
try {
|
|
// Try tar first as it's much faster, but only bsdtar supports zip files,
|
|
// so this may fail if another tar, like gnu tar, ends up being used.
|
|
uvDir = await tc.extractTar(downloadPath, undefined, "x");
|
|
} catch (err) {
|
|
core.info(
|
|
`Extracting with tar failed, falling back to zip extraction: ${(err as Error).message}`,
|
|
);
|
|
const extension = getExtension(platform);
|
|
const fullPathWithExtension = `${downloadPath}${extension}`;
|
|
await fs.copyFile(downloadPath, fullPathWithExtension);
|
|
uvDir = await tc.extractZip(fullPathWithExtension);
|
|
}
|
|
} 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 resolveChecksum(
|
|
checkSum: string | undefined,
|
|
manifestChecksum?: string,
|
|
): string | undefined {
|
|
return checkSum !== undefined && checkSum !== ""
|
|
? checkSum
|
|
: manifestChecksum;
|
|
}
|
|
|
|
function getExtension(platform: Platform): string {
|
|
return platform === "pc-windows-msvc" ? ".zip" : ".tar.gz";
|
|
}
|
|
|
|
export async function resolveVersion(
|
|
versionInput: string,
|
|
manifestUrl: string | undefined,
|
|
resolutionStrategy: "highest" | "lowest" = "highest",
|
|
): Promise<string> {
|
|
core.debug(`Resolving version: ${versionInput}`);
|
|
let version: string;
|
|
const isSimpleMinimumVersionSpecifier =
|
|
versionInput.includes(">") && !versionInput.includes(",");
|
|
const resolveVersionSpecifierToLatest =
|
|
isSimpleMinimumVersionSpecifier && resolutionStrategy === "highest";
|
|
if (resolveVersionSpecifierToLatest) {
|
|
core.info("Found minimum version specifier, using latest version");
|
|
}
|
|
if (manifestUrl !== undefined) {
|
|
version =
|
|
versionInput === "latest" || resolveVersionSpecifierToLatest
|
|
? await getLatestVersionInManifest(manifestUrl)
|
|
: versionInput;
|
|
} else {
|
|
version =
|
|
versionInput === "latest" || resolveVersionSpecifierToLatest
|
|
? await getLatestVersionFromNdjson()
|
|
: versionInput;
|
|
}
|
|
if (tc.isExplicitVersion(version)) {
|
|
core.debug(`Version ${version} is an explicit version.`);
|
|
if (resolveVersionSpecifierToLatest) {
|
|
if (!pep440.satisfies(version, versionInput)) {
|
|
throw new Error(`No version found for ${versionInput}`);
|
|
}
|
|
}
|
|
return version;
|
|
}
|
|
|
|
const availableVersions = await getAvailableVersions(manifestUrl);
|
|
core.debug(`Available versions: ${availableVersions}`);
|
|
const resolvedVersion =
|
|
resolutionStrategy === "lowest"
|
|
? minSatisfying(availableVersions, version)
|
|
: maxSatisfying(availableVersions, version);
|
|
|
|
if (resolvedVersion === undefined) {
|
|
throw new Error(`No version found for ${version}`);
|
|
}
|
|
|
|
return resolvedVersion;
|
|
}
|
|
|
|
async function getAvailableVersions(
|
|
manifestUrl: string | undefined,
|
|
): Promise<string[]> {
|
|
if (manifestUrl !== undefined) {
|
|
core.info(
|
|
`Getting available versions from manifest-file ${manifestUrl} ...`,
|
|
);
|
|
return await getAllManifestVersions(manifestUrl);
|
|
}
|
|
|
|
core.info(`Getting available versions from ${VERSIONS_NDJSON_URL} ...`);
|
|
return await getAllVersionsFromNdjson();
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
function minSatisfying(
|
|
versions: string[],
|
|
version: string,
|
|
): string | undefined {
|
|
// For semver, we need to use a different approach since tc.evaluateVersions only returns max
|
|
// Let's use semver directly for min satisfying
|
|
const minSemver = semver.minSatisfying(versions, version);
|
|
if (minSemver !== null) {
|
|
core.debug(`Found a version that satisfies the semver range: ${minSemver}`);
|
|
return minSemver;
|
|
}
|
|
const minPep440 = pep440.minSatisfying(versions, version);
|
|
if (minPep440 !== null) {
|
|
core.debug(
|
|
`Found a version that satisfies the pep440 specifier: ${minPep440}`,
|
|
);
|
|
return minPep440;
|
|
}
|
|
return undefined;
|
|
}
|