mirror of
https://github.com/astral-sh/setup-uv.git
synced 2026-06-10 14:42:22 +00:00
a92cb43098
test / test-latest-version (latest) (push) Failing after 35s
test / test-specific-version (map[expected-version:0.3.5 version-input:0.3.x]) (push) Failing after 36s
test / test-from-working-directory-version (map[expected-version:0.5.14 working-directory:__tests__/fixtures/pyproject-toml-project]) (push) Failing after 37s
test / test-specific-version (map[expected-version:0.3.2 version-input:0.3.2]) (push) Failing after 40s
test / test-specific-version (map[expected-version:0.4.30 version-input:>=0.4.25,<0.5]) (push) Failing after 44s
test / test-uv-no-modify-path (push) Failing after 50s
test / test-latest-version (>=0.8) (push) Failing after 49s
test / test-specific-version (map[expected-version:0.1.0 resolution-strategy:lowest version-input:>=0.1.0,<0.2]) (push) Failing after 51s
test / test-specific-version (map[expected-version:0.4.25 resolution-strategy:lowest version-input:>=0.4.25]) (push) Failing after 50s
test / test-specific-version (map[expected-version:0.3.5 version-input:0.3]) (push) Failing after 50s
test / test-default-version (ubuntu-latest) (push) Failing after 51s
test / test-specific-version (map[expected-version:0.1.45 resolution-strategy:highest version-input:>=0.1,<0.2]) (push) Failing after 52s
test / test-specific-version (map[expected-version:0.3.0 version-input:0.3.0]) (push) Failing after 53s
test / test-specific-version (map[expected-version:0.4.25 resolution-strategy:lowest version-input:>=0.4.25,<0.5]) (push) Failing after 53s
test / test-from-working-directory-version (map[expected-version:0.5.15 working-directory:__tests__/fixtures/uv-toml-project]) (push) Failing after 31s
test / test-version-file-version (map[expected-version:0.5.15 version-file:__tests__/fixtures/.tool-versions]) (push) Failing after 33s
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 46s
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 49s
test / test-malformed-pyproject-file-fallback (push) Failing after 49s
test / test-uvx (push) Failing after 47s
test / test-python-version (ubuntu-latest) (push) Failing after 48s
test / test-checksum (map[checksum:4d9279ad5ca596b1e2d703901d508430eb07564dc4d8837de9e2fca9c90f8ecd os:ubuntu-latest]) (push) Failing after 52s
test / test-with-explicit-token (push) Failing after 53s
test / test-activate-environment (ubuntu-latest) (push) Failing after 53s
test / test-tool-install (ubuntu-latest) (push) Failing after 54s
test / test-activate-environment-custom-path (ubuntu-latest) (push) Failing after 53s
test / test-debian-unstable (push) Failing after 55s
test / test-musl (push) Failing after 43s
test / test-activate-environment-no-project (push) Failing after 58s
test / test-cache-key-os-version (ubuntu-22.04, ubuntu-22.04) (push) Failing after 55s
test / test-setup-cache (auto, ubuntu-latest) (push) Failing after 39s
test / test-setup-cache (false, ubuntu-latest) (push) Failing after 40s
test / lint (push) Failing after 2m10s
test / test-setup-cache (true, ubuntu-latest) (push) Failing after 42s
test / test-setup-cache-requirements-txt (push) Failing after 50s
test / test-setup-cache-dependency-glob (push) Failing after 49s
test / test-restore-cache-requirements-txt (push) Has been skipped
test / test-restore-cache-dependency-glob (push) Has been skipped
test / test-setup-cache-save-cache-false (push) Failing after 51s
test / test-setup-cache-restore-cache-false (push) Failing after 51s
test / test-restore-cache-save-cache-false (push) Has been skipped
test / test-restore-cache-restore-cache-false (push) Has been skipped
test / test-cache-local-cache-disabled (push) Failing after 51s
test / test-cache-local (map[expected-cache-dir:/home/runner/work/_temp/setup-uv-cache os:ubuntu-latest]) (push) Failing after 53s
test / test-cache-local-cache-disabled-but-explicit-path (push) Failing after 52s
test / test-custom-manifest-file (push) Failing after 50s
test / test-download-from-astral-mirror-false (push) Failing after 49s
test / test-no-python-version (push) Failing after 53s
test / test-cache-prune-force (push) Failing after 38s
test / test-absolute-path (push) Failing after 43s
test / test-cache-python-missing-managed-install-dir (push) Failing after 32s
test / test-cache-dir-from-file (push) Failing after 40s
test / test-python-install-dir (map[expected-python-dir:/home/runner/work/_temp/uv-python-dir os:ubuntu-latest]) (push) Failing after 24s
test / test-relative-path (push) Failing after 47s
test / test-act (push) Failing after 21s
test / test-cache-python-installs (push) Failing after 26s
test / test-restore-python-installs (push) Has been skipped
test / validate-typings (push) Successful in 49s
CodeQL / Analyze (TypeScript) (push) Failing after 18m18s
test / test-tool-install (windows-latest) (push) Has been cancelled
Release Drafter / ✏️ Draft release (push) Has been cancelled
test / test-default-version (macos-14) (push) Has been cancelled
test / test-default-version (macos-latest) (push) Has been cancelled
test / test-default-version (windows-latest) (push) Has been cancelled
test / test-checksum (map[checksum:a70cbfbf3bb5c08b2f84963b4f12c94e08fbb2468ba418a3bfe1066fbe9e7218 os:macos-latest]) (push) Has been cancelled
test / test-tool-install (macos-latest) (push) Has been cancelled
test / test-python-version (macos-latest) (push) Has been cancelled
test / test-python-version (windows-latest) (push) Has been cancelled
test / test-activate-environment (macos-latest) (push) Has been cancelled
test / test-activate-environment (windows-latest) (push) Has been cancelled
test / test-cache-key-os-version (macos-14, macos-14) (push) Has been cancelled
test / test-tool-install (macos-14) (push) Has been cancelled
test / test-activate-environment-custom-path (macos-latest) (push) Has been cancelled
test / test-activate-environment-custom-path (windows-latest) (push) Has been cancelled
test / test-cache-key-os-version (macos-15, macos-15) (push) Has been cancelled
test / test-cache-key-os-version (ubuntu-24.04, ubuntu-24.04) (push) Has been cancelled
test / test-cache-key-os-version (windows-2022, windows-2022) (push) Has been cancelled
test / test-cache-key-os-version (windows-2025, windows-2025) (push) Has been cancelled
test / test-setup-cache (auto, windows-latest) (push) Has been cancelled
test / test-setup-cache (false, windows-latest) (push) Has been cancelled
test / test-setup-cache (true, windows-latest) (push) Has been cancelled
test / test-cache-local (map[expected-cache-dir:D:\a\_temp\setup-uv-cache os:windows-latest]) (push) Has been cancelled
test / test-python-install-dir (map[expected-python-dir:D:\a\_temp\uv-python-dir os:windows-latest]) (push) Has been cancelled
test / test-restore-cache (false, windows-latest) (push) Has been cancelled
test / test-restore-cache (auto, ubuntu-latest) (push) Has been cancelled
test / test-restore-cache (auto, windows-latest) (push) Has been cancelled
test / test-restore-cache (false, ubuntu-latest) (push) Has been cancelled
test / test-restore-cache (true, ubuntu-latest) (push) Has been cancelled
test / test-restore-cache (true, windows-latest) (push) Has been cancelled
test / all-tests-passed (push) Has been cancelled
## Summary Adds a new `quiet` input (default: `false`) that suppresses `info`-level log output when set to `true`. Only warnings and errors are shown. Contributes to: #868
341 lines
10 KiB
TypeScript
341 lines
10 KiB
TypeScript
import path from "node:path";
|
|
import * as core from "@actions/core";
|
|
import { getConfigValueFromTomlFile } from "./config-file";
|
|
import * as log from "./logging";
|
|
|
|
export enum CacheLocalSource {
|
|
Input,
|
|
Config,
|
|
Env,
|
|
Default,
|
|
}
|
|
|
|
export interface CacheLocalPath {
|
|
path: string;
|
|
source: CacheLocalSource;
|
|
}
|
|
|
|
export type ResolutionStrategy = "highest" | "lowest";
|
|
|
|
export interface SetupInputs {
|
|
workingDirectory: string;
|
|
version: string;
|
|
versionFile: string;
|
|
pythonVersion: string;
|
|
activateEnvironment: boolean;
|
|
noProject: boolean;
|
|
venvPath: string;
|
|
checksum: string;
|
|
enableCache: boolean;
|
|
restoreCache: boolean;
|
|
saveCache: boolean;
|
|
cacheSuffix: string;
|
|
cacheLocalPath?: CacheLocalPath;
|
|
cacheDependencyGlob: string;
|
|
pruneCache: boolean;
|
|
cachePython: boolean;
|
|
ignoreNothingToCache: boolean;
|
|
ignoreEmptyWorkdir: boolean;
|
|
toolBinDir?: string;
|
|
toolDir?: string;
|
|
pythonDir: string;
|
|
githubToken: string;
|
|
manifestFile?: string;
|
|
downloadFromAstralMirror: boolean;
|
|
addProblemMatchers: boolean;
|
|
quiet: boolean;
|
|
resolutionStrategy: ResolutionStrategy;
|
|
}
|
|
|
|
export function loadInputs(): SetupInputs {
|
|
const workingDirectory = core.getInput("working-directory");
|
|
const version = core.getInput("version");
|
|
const versionFile = getVersionFile(workingDirectory);
|
|
const pythonVersion = core.getInput("python-version");
|
|
const activateEnvironment = core.getBooleanInput("activate-environment");
|
|
const noProject = core.getBooleanInput("no-project");
|
|
const venvPath = getVenvPath(workingDirectory, activateEnvironment);
|
|
const checksum = core.getInput("checksum");
|
|
const enableCache = getEnableCache();
|
|
const restoreCache = core.getInput("restore-cache") === "true";
|
|
const saveCache = core.getInput("save-cache") === "true";
|
|
const cacheSuffix = core.getInput("cache-suffix") || "";
|
|
const cacheLocalPath = getCacheLocalPath(
|
|
workingDirectory,
|
|
versionFile,
|
|
enableCache,
|
|
);
|
|
const cacheDependencyGlob = getCacheDependencyGlob(workingDirectory);
|
|
const pruneCache = core.getInput("prune-cache") === "true";
|
|
const cachePython = core.getInput("cache-python") === "true";
|
|
const ignoreNothingToCache =
|
|
core.getInput("ignore-nothing-to-cache") === "true";
|
|
const ignoreEmptyWorkdir = core.getInput("ignore-empty-workdir") === "true";
|
|
const toolBinDir = getToolBinDir(workingDirectory);
|
|
const toolDir = getToolDir(workingDirectory);
|
|
const pythonDir = getUvPythonDir();
|
|
const githubToken = core.getInput("github-token");
|
|
const manifestFile = getManifestFile();
|
|
const downloadFromAstralMirror =
|
|
core.getInput("download-from-astral-mirror") === "true";
|
|
const addProblemMatchers = core.getInput("add-problem-matchers") === "true";
|
|
const quiet = core.getInput("quiet") === "true";
|
|
const resolutionStrategy = getResolutionStrategy();
|
|
|
|
return {
|
|
activateEnvironment,
|
|
addProblemMatchers,
|
|
cacheDependencyGlob,
|
|
cacheLocalPath,
|
|
cachePython,
|
|
cacheSuffix,
|
|
checksum,
|
|
downloadFromAstralMirror,
|
|
enableCache,
|
|
githubToken,
|
|
ignoreEmptyWorkdir,
|
|
ignoreNothingToCache,
|
|
manifestFile,
|
|
noProject,
|
|
pruneCache,
|
|
pythonDir,
|
|
pythonVersion,
|
|
quiet,
|
|
resolutionStrategy,
|
|
restoreCache,
|
|
saveCache,
|
|
toolBinDir,
|
|
toolDir,
|
|
venvPath,
|
|
version,
|
|
versionFile,
|
|
workingDirectory,
|
|
};
|
|
}
|
|
|
|
function getVersionFile(workingDirectory: string): string {
|
|
const versionFileInput = core.getInput("version-file");
|
|
if (versionFileInput !== "") {
|
|
const tildeExpanded = expandTilde(versionFileInput);
|
|
return resolveRelativePath(workingDirectory, tildeExpanded);
|
|
}
|
|
return versionFileInput;
|
|
}
|
|
|
|
function getVenvPath(
|
|
workingDirectory: string,
|
|
activateEnvironment: boolean,
|
|
): string {
|
|
const venvPathInput = core.getInput("venv-path");
|
|
if (venvPathInput !== "") {
|
|
if (!activateEnvironment) {
|
|
log.warning("venv-path is only used when activate-environment is true");
|
|
}
|
|
const tildeExpanded = expandTilde(venvPathInput);
|
|
return normalizePath(resolveRelativePath(workingDirectory, tildeExpanded));
|
|
}
|
|
return normalizePath(resolveRelativePath(workingDirectory, ".venv"));
|
|
}
|
|
|
|
function getEnableCache(): boolean {
|
|
const enableCacheInput = core.getInput("enable-cache");
|
|
if (enableCacheInput === "auto") {
|
|
return process.env.RUNNER_ENVIRONMENT === "github-hosted";
|
|
}
|
|
return enableCacheInput === "true";
|
|
}
|
|
|
|
function getToolBinDir(workingDirectory: string): string | undefined {
|
|
const toolBinDirInput = core.getInput("tool-bin-dir");
|
|
if (toolBinDirInput !== "") {
|
|
const tildeExpanded = expandTilde(toolBinDirInput);
|
|
return resolveRelativePath(workingDirectory, tildeExpanded);
|
|
}
|
|
if (process.platform === "win32") {
|
|
if (process.env.RUNNER_TEMP !== undefined) {
|
|
return `${process.env.RUNNER_TEMP}${path.sep}uv-tool-bin-dir`;
|
|
}
|
|
throw Error(
|
|
"Could not determine UV_TOOL_BIN_DIR. Please make sure RUNNER_TEMP is set or provide the tool-bin-dir input",
|
|
);
|
|
}
|
|
return undefined;
|
|
}
|
|
|
|
function getToolDir(workingDirectory: string): string | undefined {
|
|
const toolDirInput = core.getInput("tool-dir");
|
|
if (toolDirInput !== "") {
|
|
const tildeExpanded = expandTilde(toolDirInput);
|
|
return resolveRelativePath(workingDirectory, tildeExpanded);
|
|
}
|
|
if (process.platform === "win32") {
|
|
if (process.env.RUNNER_TEMP !== undefined) {
|
|
return `${process.env.RUNNER_TEMP}${path.sep}uv-tool-dir`;
|
|
}
|
|
throw Error(
|
|
"Could not determine UV_TOOL_DIR. Please make sure RUNNER_TEMP is set or provide the tool-dir input",
|
|
);
|
|
}
|
|
return undefined;
|
|
}
|
|
|
|
function getCacheLocalPath(
|
|
workingDirectory: string,
|
|
versionFile: string,
|
|
enableCache: boolean,
|
|
): CacheLocalPath | undefined {
|
|
const cacheLocalPathInput = core.getInput("cache-local-path");
|
|
if (cacheLocalPathInput !== "") {
|
|
const tildeExpanded = expandTilde(cacheLocalPathInput);
|
|
return {
|
|
path: resolveRelativePath(workingDirectory, tildeExpanded),
|
|
source: CacheLocalSource.Input,
|
|
};
|
|
}
|
|
const cacheDirFromConfig = getCacheDirFromConfig(
|
|
workingDirectory,
|
|
versionFile,
|
|
);
|
|
if (cacheDirFromConfig !== undefined) {
|
|
return { path: cacheDirFromConfig, source: CacheLocalSource.Config };
|
|
}
|
|
if (process.env.UV_CACHE_DIR !== undefined) {
|
|
log.info(`UV_CACHE_DIR is already set to ${process.env.UV_CACHE_DIR}`);
|
|
return { path: process.env.UV_CACHE_DIR, source: CacheLocalSource.Env };
|
|
}
|
|
if (enableCache) {
|
|
if (process.env.RUNNER_ENVIRONMENT === "github-hosted") {
|
|
if (process.env.RUNNER_TEMP !== undefined) {
|
|
return {
|
|
path: `${process.env.RUNNER_TEMP}${path.sep}setup-uv-cache`,
|
|
source: CacheLocalSource.Default,
|
|
};
|
|
}
|
|
throw Error(
|
|
"Could not determine UV_CACHE_DIR. Please make sure RUNNER_TEMP is set or provide the cache-local-path input",
|
|
);
|
|
}
|
|
if (process.platform === "win32") {
|
|
return {
|
|
path: `${process.env.APPDATA}${path.sep}uv${path.sep}cache`,
|
|
source: CacheLocalSource.Default,
|
|
};
|
|
}
|
|
return {
|
|
path: `${process.env.HOME}${path.sep}.cache${path.sep}uv`,
|
|
source: CacheLocalSource.Default,
|
|
};
|
|
}
|
|
}
|
|
|
|
function getCacheDirFromConfig(
|
|
workingDirectory: string,
|
|
versionFile: string,
|
|
): string | undefined {
|
|
for (const filePath of [versionFile, "uv.toml", "pyproject.toml"]) {
|
|
const resolvedPath = resolveRelativePath(workingDirectory, filePath);
|
|
try {
|
|
const cacheDir = getConfigValueFromTomlFile(resolvedPath, "cache-dir");
|
|
if (cacheDir !== undefined) {
|
|
log.info(`Found cache-dir in ${resolvedPath}: ${cacheDir}`);
|
|
return cacheDir;
|
|
}
|
|
} catch (err) {
|
|
const message = (err as Error).message;
|
|
log.warning(`Error while parsing ${filePath}: ${message}`);
|
|
return undefined;
|
|
}
|
|
}
|
|
return undefined;
|
|
}
|
|
|
|
export function getUvPythonDir(): string {
|
|
if (process.env.UV_PYTHON_INSTALL_DIR !== undefined) {
|
|
log.info(
|
|
`UV_PYTHON_INSTALL_DIR is already set to ${process.env.UV_PYTHON_INSTALL_DIR}`,
|
|
);
|
|
return process.env.UV_PYTHON_INSTALL_DIR;
|
|
}
|
|
if (process.env.RUNNER_ENVIRONMENT !== "github-hosted") {
|
|
if (process.platform === "win32") {
|
|
return `${process.env.APPDATA}${path.sep}uv${path.sep}python`;
|
|
}
|
|
return `${process.env.HOME}${path.sep}.local${path.sep}share${path.sep}uv${path.sep}python`;
|
|
}
|
|
if (process.env.RUNNER_TEMP !== undefined) {
|
|
return `${process.env.RUNNER_TEMP}${path.sep}uv-python-dir`;
|
|
}
|
|
throw Error(
|
|
"Could not determine UV_PYTHON_INSTALL_DIR. Please make sure RUNNER_TEMP is set or provide the UV_PYTHON_INSTALL_DIR environment variable",
|
|
);
|
|
}
|
|
|
|
function getCacheDependencyGlob(workingDirectory: string): string {
|
|
const cacheDependencyGlobInput = core.getInput("cache-dependency-glob");
|
|
if (cacheDependencyGlobInput !== "") {
|
|
return cacheDependencyGlobInput
|
|
.split("\n")
|
|
.map((part) => part.trim())
|
|
.map((part) => expandTilde(part))
|
|
.map((part) => resolveRelativePath(workingDirectory, part))
|
|
.join("\n");
|
|
}
|
|
return cacheDependencyGlobInput;
|
|
}
|
|
|
|
function expandTilde(input: string): string {
|
|
if (input.startsWith("~")) {
|
|
return `${process.env.HOME}${input.substring(1)}`;
|
|
}
|
|
return input;
|
|
}
|
|
|
|
function normalizePath(inputPath: string): string {
|
|
const normalized = path.normalize(inputPath);
|
|
const root = path.parse(normalized).root;
|
|
|
|
// Remove any trailing path separators, except when the whole path is the root.
|
|
let trimmed = normalized;
|
|
while (trimmed.length > root.length && trimmed.endsWith(path.sep)) {
|
|
trimmed = trimmed.slice(0, -1);
|
|
}
|
|
|
|
return trimmed;
|
|
}
|
|
|
|
function resolveRelativePath(
|
|
workingDirectory: string,
|
|
inputPath: string,
|
|
): string {
|
|
const hasNegation = inputPath.startsWith("!");
|
|
const pathWithoutNegation = hasNegation ? inputPath.substring(1) : inputPath;
|
|
|
|
const resolvedPath = path.resolve(workingDirectory, pathWithoutNegation);
|
|
|
|
core.debug(
|
|
`Resolving relative path ${inputPath} to ${hasNegation ? "!" : ""}${resolvedPath}`,
|
|
);
|
|
return hasNegation ? `!${resolvedPath}` : resolvedPath;
|
|
}
|
|
|
|
function getManifestFile(): string | undefined {
|
|
const manifestFileInput = core.getInput("manifest-file");
|
|
if (manifestFileInput !== "") {
|
|
return manifestFileInput;
|
|
}
|
|
return undefined;
|
|
}
|
|
|
|
function getResolutionStrategy(): ResolutionStrategy {
|
|
const resolutionStrategyInput = core.getInput("resolution-strategy");
|
|
if (resolutionStrategyInput === "lowest") {
|
|
return "lowest";
|
|
}
|
|
if (resolutionStrategyInput === "highest" || resolutionStrategyInput === "") {
|
|
return "highest";
|
|
}
|
|
throw new Error(
|
|
`Invalid resolution-strategy: ${resolutionStrategyInput}. Must be 'highest' or 'lowest'.`,
|
|
);
|
|
}
|