Add support for semver version ranges (#82)

Closes: #38
This commit is contained in:
Kevin Stillhammer 2024-09-18 11:29:09 +02:00 committed by GitHub
parent 8e09161b4b
commit ce0062aac7
WARNING! Although there is a key with this ID in the database it does not verify this commit! This commit is SUSPICIOUS.
GPG key ID: B5690EEEBB952194
5 changed files with 4259 additions and 32 deletions

View file

@ -40,7 +40,7 @@ jobs:
strategy: strategy:
matrix: matrix:
os: [ubuntu-latest, macos-latest, macos-14, oracle-aarch64] os: [ubuntu-latest, macos-latest, macos-14, oracle-aarch64]
uv-version: ["latest", "0.3.0", "0.3.2"] uv-version: ["latest", "0.3.0", "0.3.2", "0.3", "0.3.x", ">=0.3.0"]
steps: steps:
- uses: actions/checkout@v4 - uses: actions/checkout@v4
- name: Install version ${{ matrix.uv-version }} - name: Install version ${{ matrix.uv-version }}
@ -49,6 +49,25 @@ jobs:
version: ${{ matrix.uv-version }} version: ${{ matrix.uv-version }}
- run: uv sync - run: uv sync
working-directory: __tests__/fixtures/uv-project working-directory: __tests__/fixtures/uv-project
test-semver-range:
runs-on: ${{ matrix.os }}
strategy:
matrix:
os: [ubuntu-latest, oracle-aarch64]
steps:
- uses: actions/checkout@v4
- name: Install version 0.3
id: setup-uv
uses: ./
with:
version: "0.3"
- name: Correct version gets installed
run: |
if [ "$UV_VERSION" != "0.3.5" ]; then
exit 1
fi
env:
UV_VERSION: ${{ steps.setup-uv.outputs.uv-version }}
test-checksum: test-checksum:
runs-on: ${{ matrix.os }} runs-on: ${{ matrix.os }}
strategy: strategy:

View file

@ -50,6 +50,25 @@ For an example workflow, see
version: "0.4.4" version: "0.4.4"
``` ```
### Install a version by supplying a semver range
You can also specify a [semver range](https://github.com/npm/node-semver?tab=readme-ov-file#ranges)
to install the latest version that satisfies the range.
```yaml
- name: Install a semver range of uv
uses: astral-sh/setup-uv@v2
with:
version: ">=0.3.0"
```
```yaml
- name: Pinning a minor version of uv
uses: astral-sh/setup-uv@v2
with:
version: "0.3.x"
```
### Validate checksum ### Validate checksum
You can also specify a checksum to validate the downloaded file. Checksums up to the default version You can also specify a checksum to validate the downloaded file. Checksums up to the default version

4163
dist/setup/index.js generated vendored

File diff suppressed because it is too large Load diff

View file

@ -4,15 +4,21 @@ import * as path from "path";
import { OWNER, REPO, TOOL_CACHE_NAME } from "../utils/constants"; import { OWNER, REPO, TOOL_CACHE_NAME } from "../utils/constants";
import { Architecture, Platform } from "../utils/platforms"; import { Architecture, Platform } from "../utils/platforms";
import { validateChecksum } from "./checksum/checksum"; import { validateChecksum } from "./checksum/checksum";
import * as github from "@actions/github";
export function tryGetFromToolCache( export function tryGetFromToolCache(
arch: Architecture, arch: Architecture,
version: string, version: string,
): string | undefined { ): { version: string; installedPath: string | undefined } {
core.debug(`Trying to get uv from tool cache for ${version}...`); core.debug(`Trying to get uv from tool cache for ${version}...`);
const cachedVersions = tc.findAllVersions(TOOL_CACHE_NAME, arch); const cachedVersions = tc.findAllVersions(TOOL_CACHE_NAME, arch);
core.debug(`Cached versions: ${cachedVersions}`); core.debug(`Cached versions: ${cachedVersions}`);
return tc.find(TOOL_CACHE_NAME, version, arch); let resolvedVersion = tc.evaluateVersions(cachedVersions, version);
if (resolvedVersion === "") {
resolvedVersion = version;
}
const installedPath = tc.find(TOOL_CACHE_NAME, resolvedVersion, arch);
return { version: resolvedVersion, installedPath };
} }
export async function downloadVersion( export async function downloadVersion(
@ -20,10 +26,11 @@ export async function downloadVersion(
arch: Architecture, arch: Architecture,
version: string, version: string,
checkSum: string | undefined, checkSum: string | undefined,
githubToken: string | undefined, githubToken: string,
): Promise<string> { ): Promise<{ version: string; cachedToolDir: string }> {
const resolvedVersion = await resolveVersion(version, githubToken);
const artifact = `uv-${arch}-${platform}`; const artifact = `uv-${arch}-${platform}`;
let downloadUrl = `https://github.com/${OWNER}/${REPO}/releases/download/${version}/${artifact}`; let downloadUrl = `https://github.com/${OWNER}/${REPO}/releases/download/${resolvedVersion}/${artifact}`;
if (platform === "pc-windows-msvc") { if (platform === "pc-windows-msvc") {
downloadUrl += ".zip"; downloadUrl += ".zip";
} else { } else {
@ -36,7 +43,13 @@ export async function downloadVersion(
undefined, undefined,
githubToken, githubToken,
); );
await validateChecksum(checkSum, downloadPath, arch, platform, version); await validateChecksum(
checkSum,
downloadPath,
arch,
platform,
resolvedVersion,
);
let uvDir: string; let uvDir: string;
if (platform === "pc-windows-msvc") { if (platform === "pc-windows-msvc") {
@ -46,6 +59,37 @@ export async function downloadVersion(
const extractedDir = await tc.extractTar(downloadPath); const extractedDir = await tc.extractTar(downloadPath);
uvDir = path.join(extractedDir, artifact); uvDir = path.join(extractedDir, artifact);
} }
const cachedToolDir = await tc.cacheDir(
return await tc.cacheDir(uvDir, TOOL_CACHE_NAME, version, arch); uvDir,
TOOL_CACHE_NAME,
resolvedVersion,
arch,
);
return { version: resolvedVersion, cachedToolDir };
}
async function resolveVersion(
version: string,
githubToken: string,
): Promise<string> {
if (tc.isExplicitVersion(version)) {
core.debug(`Version ${version} is an explicit version.`);
return version;
}
const availableVersions = await getAvailableVersions(githubToken);
const resolvedVersion = tc.evaluateVersions(availableVersions, version);
if (resolvedVersion === "") {
throw new Error(`No version found for ${version}`);
}
return resolvedVersion;
}
async function getAvailableVersions(githubToken: string): Promise<string[]> {
const octokit = github.getOctokit(githubToken);
const response = await octokit.paginate(octokit.rest.repos.listReleases, {
owner: OWNER,
repo: REPO,
});
return response.map((release) => release.tag_name);
} }

View file

@ -41,8 +41,8 @@ async function run(): Promise<void> {
); );
addUvToPath(setupResult.uvDir); addUvToPath(setupResult.uvDir);
core.setOutput("uv-version", version); core.setOutput("uv-version", setupResult.version);
core.info(`Successfully installed uv version ${version}`); core.info(`Successfully installed uv version ${setupResult.version}`);
addMatchers(); addMatchers();
setCacheDir(cacheLocalPath); setCacheDir(cacheLocalPath);
@ -50,10 +50,10 @@ async function run(): Promise<void> {
if (enableCache) { if (enableCache) {
await restoreCache(setupResult.version); await restoreCache(setupResult.version);
} }
process.exit(0);
} catch (err) { } catch (err) {
core.setFailed((err as Error).message); core.setFailed((err as Error).message);
} }
process.exit(0);
} }
async function setupUv( async function setupUv(
@ -61,29 +61,37 @@ async function setupUv(
arch: Architecture, arch: Architecture,
versionInput: string, versionInput: string,
checkSum: string | undefined, checkSum: string | undefined,
githubToken: string | undefined, githubToken: string,
): Promise<{ uvDir: string; version: string }> { ): Promise<{ uvDir: string; version: string }> {
let installedPath: string | undefined; let installedPath: string | undefined;
let cachedToolDir: string; let cachedToolDir: string;
let version: string; let version: string;
if (versionInput === "latest") { if (versionInput === "latest") {
const result = await downloadLatest(platform, arch, checkSum, githubToken); const latestResult = await downloadLatest(
version = result.version; platform,
cachedToolDir = result.cachedToolDir; arch,
checkSum,
githubToken,
);
version = latestResult.version;
cachedToolDir = latestResult.cachedToolDir;
} else { } else {
version = versionInput; const toolCacheResult = tryGetFromToolCache(arch, versionInput);
installedPath = tryGetFromToolCache(arch, versionInput); version = toolCacheResult.version;
installedPath = toolCacheResult.installedPath;
if (installedPath) { if (installedPath) {
core.info(`Found uv in tool-cache for ${versionInput}`); core.info(`Found uv in tool-cache for ${versionInput}`);
return { uvDir: installedPath, version }; return { uvDir: installedPath, version };
} }
cachedToolDir = await downloadVersion( const versionResult = await downloadVersion(
platform, platform,
arch, arch,
versionInput, versionInput,
checkSum, checkSum,
githubToken, githubToken,
); );
cachedToolDir = versionResult.cachedToolDir;
version = versionResult.version;
} }
return { uvDir: cachedToolDir, version }; return { uvDir: cachedToolDir, version };