mirror of
https://github.com/astral-sh/setup-uv.git
synced 2026-06-09 22:22:41 +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
167 lines
5.1 KiB
TypeScript
167 lines
5.1 KiB
TypeScript
import * as fs from "node:fs";
|
|
import * as cache from "@actions/cache";
|
|
import * as core from "@actions/core";
|
|
import * as exec from "@actions/exec";
|
|
import * as pep440 from "@renovatebot/pep440";
|
|
import {
|
|
STATE_CACHE_KEY,
|
|
STATE_CACHE_MATCHED_KEY,
|
|
STATE_PYTHON_CACHE_MATCHED_KEY,
|
|
} from "./cache/restore-cache";
|
|
import { STATE_UV_PATH, STATE_UV_VERSION } from "./utils/constants";
|
|
import { loadInputs, type SetupInputs } from "./utils/inputs";
|
|
import * as log from "./utils/logging";
|
|
|
|
function formatUnexpectedFailure(error: unknown): string {
|
|
if (error instanceof Error) {
|
|
return error.stack ?? error.message;
|
|
}
|
|
return String(error);
|
|
}
|
|
|
|
function failUnexpectedly(event: string, error: unknown): never {
|
|
core.setFailed(`${event}: ${formatUnexpectedFailure(error)}`);
|
|
process.exit(1);
|
|
}
|
|
|
|
process.on("uncaughtException", (error) => {
|
|
failUnexpectedly("Uncaught exception", error);
|
|
});
|
|
|
|
process.on("unhandledRejection", (reason) => {
|
|
failUnexpectedly("Unhandled promise rejection", reason);
|
|
});
|
|
|
|
export async function run(): Promise<void> {
|
|
try {
|
|
const inputs = loadInputs();
|
|
if (inputs.enableCache) {
|
|
if (inputs.saveCache) {
|
|
await saveCache(inputs);
|
|
} else {
|
|
log.info("save-cache is false. Skipping save cache step.");
|
|
}
|
|
// https://github.com/nodejs/node/issues/56645#issuecomment-3924958861
|
|
await new Promise((resolve) => setTimeout(resolve, 100));
|
|
|
|
// node will stay alive if any promises are not resolved,
|
|
// which is a possibility if HTTP requests are dangling
|
|
// due to retries or timeouts. We know that if we got here
|
|
// that all promises that we care about have successfully
|
|
// resolved, so simply exit with success.
|
|
process.exit(0);
|
|
}
|
|
} catch (error) {
|
|
const err = error as Error;
|
|
core.setFailed(err.message);
|
|
}
|
|
}
|
|
|
|
async function saveCache(inputs: SetupInputs): Promise<void> {
|
|
const cacheKey = core.getState(STATE_CACHE_KEY);
|
|
const matchedKey = core.getState(STATE_CACHE_MATCHED_KEY);
|
|
|
|
if (!cacheKey) {
|
|
log.warning("Error retrieving cache key from state.");
|
|
return;
|
|
}
|
|
if (matchedKey === cacheKey) {
|
|
log.info(`Cache hit occurred on key ${cacheKey}, not saving cache.`);
|
|
} else {
|
|
if (inputs.pruneCache) {
|
|
await pruneCache();
|
|
}
|
|
|
|
const actualCachePath = getUvCachePath(inputs);
|
|
if (!fs.existsSync(actualCachePath)) {
|
|
if (inputs.ignoreNothingToCache) {
|
|
log.info(
|
|
"No cacheable uv cache paths were found. Ignoring because ignore-nothing-to-cache is enabled.",
|
|
);
|
|
} else {
|
|
throw new Error(
|
|
`Cache path ${actualCachePath} does not exist on disk. This likely indicates that there are no dependencies to cache. Consider disabling the cache input if it is not needed.`,
|
|
);
|
|
}
|
|
} else {
|
|
await saveCacheToKey(
|
|
cacheKey,
|
|
actualCachePath,
|
|
STATE_CACHE_MATCHED_KEY,
|
|
"uv cache",
|
|
);
|
|
}
|
|
}
|
|
|
|
if (inputs.cachePython) {
|
|
if (!fs.existsSync(inputs.pythonDir)) {
|
|
log.warning(
|
|
`Python cache path ${inputs.pythonDir} does not exist on disk. Skipping Python cache save because no managed Python installation was found. If you want uv to install managed Python instead of using a system interpreter, set UV_PYTHON_PREFERENCE=only-managed.`,
|
|
);
|
|
return;
|
|
}
|
|
|
|
const pythonCacheKey = `${cacheKey}-python`;
|
|
await saveCacheToKey(
|
|
pythonCacheKey,
|
|
inputs.pythonDir,
|
|
STATE_PYTHON_CACHE_MATCHED_KEY,
|
|
"Python cache",
|
|
);
|
|
}
|
|
}
|
|
|
|
async function pruneCache(): Promise<void> {
|
|
const forceSupported = pep440.gte(core.getState(STATE_UV_VERSION), "0.8.24");
|
|
|
|
const options: exec.ExecOptions = {
|
|
silent: false,
|
|
};
|
|
const execArgs = ["cache", "prune", "--ci"];
|
|
if (forceSupported) {
|
|
execArgs.push("--force");
|
|
}
|
|
|
|
log.info("Pruning cache...");
|
|
const uvPath = core.getState(STATE_UV_PATH);
|
|
await exec.exec(uvPath, execArgs, options);
|
|
}
|
|
|
|
function getUvCachePath(inputs: SetupInputs): string {
|
|
if (inputs.cacheLocalPath === undefined) {
|
|
throw new Error(
|
|
"cache-local-path is not set. Cannot save cache without a valid cache path.",
|
|
);
|
|
}
|
|
if (
|
|
process.env.UV_CACHE_DIR &&
|
|
process.env.UV_CACHE_DIR !== inputs.cacheLocalPath.path
|
|
) {
|
|
log.warning(
|
|
`The environment variable UV_CACHE_DIR has been changed to "${process.env.UV_CACHE_DIR}", by an action or step running after astral-sh/setup-uv. This can lead to unexpected behavior. If you expected this to happen set the cache-local-path input to "${process.env.UV_CACHE_DIR}" instead of "${inputs.cacheLocalPath.path}".`,
|
|
);
|
|
return process.env.UV_CACHE_DIR;
|
|
}
|
|
return inputs.cacheLocalPath.path;
|
|
}
|
|
|
|
async function saveCacheToKey(
|
|
cacheKey: string,
|
|
cachePath: string,
|
|
stateKey: string,
|
|
cacheName: string,
|
|
): Promise<void> {
|
|
const matchedKey = core.getState(stateKey);
|
|
|
|
if (matchedKey === cacheKey) {
|
|
log.info(`${cacheName} hit occurred on key ${cacheKey}, not saving cache.`);
|
|
return;
|
|
}
|
|
|
|
log.info(`Including ${cacheName} path: ${cachePath}`);
|
|
await cache.saveCache([cachePath], cacheKey);
|
|
log.info(`${cacheName} saved with key: ${cacheKey}`);
|
|
}
|
|
|
|
run();
|