diff --git a/src/setup-uv.ts b/src/setup-uv.ts index ee7e77c..26c3695 100644 --- a/src/setup-uv.ts +++ b/src/setup-uv.ts @@ -28,7 +28,7 @@ import { } from "./utils/inputs"; import * as exec from "@actions/exec"; import fs from "node:fs"; -import { getUvVersionFromConfigFile } from "./utils/pyproject"; +import { getUvVersionFromConfigFile, getPythonVersionFromPyProject } from "./utils/pyproject"; async function run(): Promise { detectEmptyWorkdir(); @@ -164,23 +164,85 @@ function setToolDir(): void { } async function setupPython(): Promise { - if (pythonVersion !== "") { - core.exportVariable("UV_PYTHON", pythonVersion); - core.info(`Set UV_PYTHON to ${pythonVersion}`); - const options: exec.ExecOptions = { - silent: !core.isDebug(), - }; - const execArgs = ["venv", "--python", pythonVersion]; + const options: exec.ExecOptions = { + silent: !core.isDebug(), + }; - core.info("Activating python venv..."); + // Case (1): No Python version and no pyproject.toml file + if (pythonVersion === "" && pyProjectFile === "") { + core.info("No Python setup required."); + return; + } + + // Case (2): Python version is provided and no pyproject.toml file + else if (pythonVersion !== "" && pyProjectFile === "") { + const execArgs = ["pin", "python", pythonVersion]; + core.info(`Pinning Python version to ${pythonVersion}...`); await exec.exec("uv", execArgs, options); + } - let venvBinPath = ".venv/bin"; - if (process.platform === "win32") { - venvBinPath = ".venv/Scripts"; + // Case (3): No Python version and pyproject.toml file + else if (pythonVersion === "" && pyProjectFile !== "") { + const extractedPythonVersion = getPythonVersionFromPyProject(pyProjectFile); + + if (!extractedPythonVersion){ + core.warning( + `Could not find python version in pyproject.toml. Won't setup python.`, + ); + return; } - core.addPath(path.resolve(venvBinPath)); - core.exportVariable("VIRTUAL_ENV", path.resolve(".venv")); + + const execArgs = ["pin", "python", extractedPythonVersion, "--project", pyProjectFile]; + core.info(`Pinning Python version to ${extractedPythonVersion}...`); + await exec.exec("uv", execArgs, options); + } + + // Case (4): Pin python version using uv pin if python version is provided and pyproject.toml file is present + if (pythonVersion !== "" && pyProjectFile !== "") { + const execArgs = ["pin", "python", pythonVersion, "--project", pyProjectFile]; + core.info(`Pinning Python version to ${pythonVersion}...`); + await exec.exec("uv", execArgs, options); + } + + // Extract the pinned python version + let pinnedPythonVersion = getPinnedPythonVersion(); + if (!pinnedPythonVersion) { + core.setFailed("Failed to determine pinned Python version after uv pin."); + return; + } + + // Setup and activate venv + // Set UV_PYHTON to the pinned python version + core.exportVariable("UV_PYTHON", pinnedPythonVersion); + core.info(`Setting UV_PYTHON to ${pinnedPythonVersion}`); + + core.info("Activating Python venv..."); + const execArgs = ["venv"]; + await exec.exec("uv", execArgs, options); + + let venvBinPath = ".venv/bin"; + if (process.platform === "win32") { + venvBinPath = ".venv/Scripts"; + } + core.addPath(path.resolve(venvBinPath)); + core.exportVariable("VIRTUAL_ENV", path.resolve(".venv")); +} + +function getPinnedPythonVersion(): string | undefined { + const pythonVersionFile = ".python-version"; + + if (!fs.existsSync(pythonVersionFile)) { + core.warning(`No .python-version file found after uv pin.`); + return undefined; + } + + try { + const version = fs.readFileSync(pythonVersionFile, "utf-8").trim(); + core.info(`Detected pinned Python version from .python-version: ${version}`); + return version; + } catch (error) { + core.warning(`Failed to read .python-version: ${error}`); + return undefined; } } diff --git a/src/utils/pyproject.ts b/src/utils/pyproject.ts index 4dd1ff0..9c031a7 100644 --- a/src/utils/pyproject.ts +++ b/src/utils/pyproject.ts @@ -44,3 +44,31 @@ function getRequiredVersion(filePath: string): string | undefined { }; return tomlContent["required-version"]; } + +export function getPythonVersionFromPyProject( + filePath: string, +): string | undefined { + if (!fs.existsSync(filePath)) { + core.warning(`Could not find pyproject.toml file: ${filePath}`); + return undefined; + } + + let pythonVersionConstraint: string | undefined; + try { + const fileContent = fs.readFileSync(filePath, "utf-8"); + const tomlContent = toml.parse(fileContent) as { + tool?: { uv?: { "requires-python"?: string } }; + }; + + pythonVersionConstraint = tomlContent?.tool?.uv?.["requires-python"]; + + if (pythonVersionConstraint) { + core.info(`Extracted Python version constraint from ${filePath}: ${pythonVersionConstraint}`); + } + } catch (err) { + const message = (err as Error).message; + core.warning(`Error while parsing ${filePath} for Python version: ${message}`); + } + + return pythonVersionConstraint; +}