mirror of
https://github.com/astral-sh/setup-uv.git
synced 2026-06-24 13:32:36 +00:00
3faa3174e6
Adds `uv.lock` as a supported `version-file` source. When `uv` is locked as a dependency in `uv.lock`, the action now installs the exact pinned version, closing the gap reported in #682. This is useful for deterministic CI: the same uv version is used until the lockfile is updated, which avoids "CI worked yesterday, fails today" drift and reduces supply-chain exposure from auto-installing the latest release. The implementation mirrors the existing `version-file` parsers — a new `uv.lock` entry in the parser registry reads the `[[package]]` whose `name = "uv"` and returns its locked `version`. Scoped to explicit `version-file: uv.lock`; workspace auto-detection is left as a possible follow-up to avoid precedence ambiguity with `uv.toml` / `pyproject.toml`. Validation (local, Node 23; dist build is esbuild-deterministic): - `npm run all` → build clean, biome clean, package clean, jest 77/77 - New tests: 3 unit (`uv-lock-file.test.ts`) + 1 integration — exact pin resolves through the full pipeline (`uv.lock` → `0.8.17`) - dist rebuilt + committed (single bundle, no spurious churn) related: #682
108 lines
3.1 KiB
TypeScript
108 lines
3.1 KiB
TypeScript
import fs from "node:fs";
|
|
import { getConfigValueFromTomlContent } from "../utils/config-file";
|
|
import * as log from "../utils/logging";
|
|
import {
|
|
getUvVersionFromParsedPyproject,
|
|
getUvVersionFromRequirementsText,
|
|
parsePyprojectContent,
|
|
} from "./requirements-file";
|
|
import { normalizeVersionSpecifier } from "./specifier";
|
|
import { getUvVersionFromToolVersions } from "./tool-versions-file";
|
|
import type { ParsedVersionFile, VersionFileFormat } from "./types";
|
|
import { getUvVersionFromUvLock } from "./uv-lock-file";
|
|
|
|
interface VersionFileParser {
|
|
format: VersionFileFormat;
|
|
parse(filePath: string): string | undefined;
|
|
supports(filePath: string): boolean;
|
|
}
|
|
|
|
const VERSION_FILE_PARSERS: VersionFileParser[] = [
|
|
{
|
|
format: ".tool-versions",
|
|
parse: (filePath) => getUvVersionFromToolVersions(filePath),
|
|
supports: (filePath) => filePath.endsWith(".tool-versions"),
|
|
},
|
|
{
|
|
format: "uv.toml",
|
|
parse: (filePath) => {
|
|
const fileContent = fs.readFileSync(filePath, "utf-8");
|
|
return getConfigValueFromTomlContent(
|
|
filePath,
|
|
fileContent,
|
|
"required-version",
|
|
);
|
|
},
|
|
supports: (filePath) => filePath.endsWith("uv.toml"),
|
|
},
|
|
{
|
|
format: "pyproject.toml",
|
|
parse: (filePath) => {
|
|
const fileContent = fs.readFileSync(filePath, "utf-8");
|
|
const pyproject = parsePyprojectContent(fileContent);
|
|
const requiredVersion = pyproject.tool?.uv?.["required-version"];
|
|
|
|
if (requiredVersion !== undefined) {
|
|
return requiredVersion;
|
|
}
|
|
|
|
return getUvVersionFromParsedPyproject(pyproject);
|
|
},
|
|
supports: (filePath) => filePath.endsWith("pyproject.toml"),
|
|
},
|
|
{
|
|
format: "uv.lock",
|
|
parse: (filePath) => getUvVersionFromUvLock(filePath),
|
|
supports: (filePath) => filePath.endsWith("uv.lock"),
|
|
},
|
|
{
|
|
format: "requirements",
|
|
parse: (filePath) => {
|
|
const fileContent = fs.readFileSync(filePath, "utf-8");
|
|
return getUvVersionFromRequirementsText(fileContent);
|
|
},
|
|
supports: (filePath) => filePath.endsWith(".txt"),
|
|
},
|
|
];
|
|
|
|
export function getParsedVersionFile(
|
|
filePath: string,
|
|
): ParsedVersionFile | undefined {
|
|
log.info(`Trying to find version for uv in: ${filePath}`);
|
|
|
|
if (!fs.existsSync(filePath)) {
|
|
log.info(`Could not find file: ${filePath}`);
|
|
return undefined;
|
|
}
|
|
|
|
const parser = getVersionFileParser(filePath);
|
|
if (parser === undefined) {
|
|
return undefined;
|
|
}
|
|
|
|
try {
|
|
const specifier = parser.parse(filePath);
|
|
if (specifier === undefined) {
|
|
return undefined;
|
|
}
|
|
|
|
const normalizedSpecifier = normalizeVersionSpecifier(specifier);
|
|
log.info(`Found version for uv in ${filePath}: ${normalizedSpecifier}`);
|
|
return {
|
|
format: parser.format,
|
|
specifier: normalizedSpecifier,
|
|
};
|
|
} catch (error) {
|
|
log.warning(`Error while parsing ${filePath}: ${(error as Error).message}`);
|
|
return undefined;
|
|
}
|
|
}
|
|
|
|
export function getUvVersionFromFile(filePath: string): string | undefined {
|
|
return getParsedVersionFile(filePath)?.specifier;
|
|
}
|
|
|
|
function getVersionFileParser(filePath: string): VersionFileParser | undefined {
|
|
return VERSION_FILE_PARSERS.find((parser) => parser.supports(filePath));
|
|
}
|