mirror of
https://github.com/astral-sh/setup-uv.git
synced 2026-01-13 06:52:16 +00:00
add outputs python-version and python-cache-hit (#728)
This commit splits up the "normal" cache containing the dependencies and the "python" cache containing the python binaries. This will lead to a one-time invalidation of caches. Closes: #713
This commit is contained in:
committed by
GitHub
parent
11050edb83
commit
3ae150cc9d
11
.github/workflows/test.yml
vendored
11
.github/workflows/test.yml
vendored
@@ -325,6 +325,7 @@ jobs:
|
|||||||
with:
|
with:
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
- name: Install latest version
|
- name: Install latest version
|
||||||
|
id: setup-uv
|
||||||
uses: ./
|
uses: ./
|
||||||
with:
|
with:
|
||||||
python-version: 3.13.1t
|
python-version: 3.13.1t
|
||||||
@@ -335,6 +336,14 @@ jobs:
|
|||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
shell: bash
|
shell: bash
|
||||||
|
- name: Verify output python-version is correct
|
||||||
|
run: |
|
||||||
|
if [ "$PYTHON_VERSION" != "3.13.1t" ]; then
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
shell: bash
|
||||||
|
env:
|
||||||
|
PYTHON_VERSION: ${{ steps.setup-uv.outputs.python-version }}
|
||||||
- run: uv sync
|
- run: uv sync
|
||||||
working-directory: __tests__/fixtures/uv-project
|
working-directory: __tests__/fixtures/uv-project
|
||||||
|
|
||||||
@@ -986,7 +995,7 @@ jobs:
|
|||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
env:
|
env:
|
||||||
CACHE_HIT: ${{ steps.restore.outputs.cache-hit }}
|
CACHE_HIT: ${{ steps.restore.outputs.python-cache-hit }}
|
||||||
- run: uv sync --managed-python
|
- run: uv sync --managed-python
|
||||||
working-directory: __tests__/fixtures/uv-project
|
working-directory: __tests__/fixtures/uv-project
|
||||||
|
|
||||||
|
|||||||
@@ -125,6 +125,8 @@ Have a look under [Advanced Configuration](#advanced-configuration) for detailed
|
|||||||
- `uvx-path`: The path to the installed uvx binary.
|
- `uvx-path`: The path to the installed uvx binary.
|
||||||
- `cache-hit`: A boolean value to indicate a cache entry was found.
|
- `cache-hit`: A boolean value to indicate a cache entry was found.
|
||||||
- `venv`: Path to the activated venv if activate-environment is true.
|
- `venv`: Path to the activated venv if activate-environment is true.
|
||||||
|
- `python-version`: The Python version that was set.
|
||||||
|
- `python-cache-hit`: A boolean value to indicate the Python cache entry was found.
|
||||||
|
|
||||||
### Python version
|
### Python version
|
||||||
|
|
||||||
|
|||||||
@@ -69,3 +69,7 @@ outputs:
|
|||||||
type: string
|
type: string
|
||||||
venv:
|
venv:
|
||||||
type: string
|
type: string
|
||||||
|
python-version:
|
||||||
|
type: string
|
||||||
|
python-cache-hit:
|
||||||
|
type: boolean
|
||||||
|
|||||||
@@ -93,6 +93,10 @@ outputs:
|
|||||||
description: "The cache key used for storing/restoring the cache"
|
description: "The cache key used for storing/restoring the cache"
|
||||||
venv:
|
venv:
|
||||||
description: "Path to the activated venv if activate-environment is true"
|
description: "Path to the activated venv if activate-environment is true"
|
||||||
|
python-version:
|
||||||
|
description: "The Python version that was set."
|
||||||
|
python-cache-hit:
|
||||||
|
description: "A boolean value to indicate the Python cache entry was found"
|
||||||
runs:
|
runs:
|
||||||
using: "node24"
|
using: "node24"
|
||||||
main: "dist/setup/index.js"
|
main: "dist/setup/index.js"
|
||||||
|
|||||||
153
dist/save-cache/index.js
generated
vendored
153
dist/save-cache/index.js
generated
vendored
@@ -90599,46 +90599,52 @@ var __importStar = (this && this.__importStar) || (function () {
|
|||||||
};
|
};
|
||||||
})();
|
})();
|
||||||
Object.defineProperty(exports, "__esModule", ({ value: true }));
|
Object.defineProperty(exports, "__esModule", ({ value: true }));
|
||||||
exports.STATE_CACHE_MATCHED_KEY = exports.STATE_CACHE_KEY = void 0;
|
exports.STATE_PYTHON_CACHE_MATCHED_KEY = exports.STATE_CACHE_MATCHED_KEY = exports.STATE_CACHE_KEY = void 0;
|
||||||
exports.restoreCache = restoreCache;
|
exports.restoreCache = restoreCache;
|
||||||
const cache = __importStar(__nccwpck_require__(5116));
|
const cache = __importStar(__nccwpck_require__(5116));
|
||||||
const core = __importStar(__nccwpck_require__(7484));
|
const core = __importStar(__nccwpck_require__(7484));
|
||||||
const exec = __importStar(__nccwpck_require__(5236));
|
|
||||||
const hash_files_1 = __nccwpck_require__(9660);
|
const hash_files_1 = __nccwpck_require__(9660);
|
||||||
const inputs_1 = __nccwpck_require__(9612);
|
const inputs_1 = __nccwpck_require__(9612);
|
||||||
const platforms_1 = __nccwpck_require__(8361);
|
const platforms_1 = __nccwpck_require__(8361);
|
||||||
exports.STATE_CACHE_KEY = "cache-key";
|
exports.STATE_CACHE_KEY = "cache-key";
|
||||||
exports.STATE_CACHE_MATCHED_KEY = "cache-matched-key";
|
exports.STATE_CACHE_MATCHED_KEY = "cache-matched-key";
|
||||||
|
exports.STATE_PYTHON_CACHE_MATCHED_KEY = "python-cache-matched-key";
|
||||||
const CACHE_VERSION = "2";
|
const CACHE_VERSION = "2";
|
||||||
async function restoreCache() {
|
async function restoreCache(pythonVersion) {
|
||||||
const cacheKey = await computeKeys();
|
const cacheKey = await computeKeys(pythonVersion);
|
||||||
core.saveState(exports.STATE_CACHE_KEY, cacheKey);
|
core.saveState(exports.STATE_CACHE_KEY, cacheKey);
|
||||||
core.setOutput("cache-key", cacheKey);
|
core.setOutput("cache-key", cacheKey);
|
||||||
if (!inputs_1.restoreCache) {
|
if (!inputs_1.restoreCache) {
|
||||||
core.info("restore-cache is false. Skipping restore cache step.");
|
core.info("restore-cache is false. Skipping restore cache step.");
|
||||||
|
core.setOutput("python-cache-hit", false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
let matchedKey;
|
|
||||||
core.info(`Trying to restore uv cache from GitHub Actions cache with key: ${cacheKey}`);
|
|
||||||
if (inputs_1.cacheLocalPath === undefined) {
|
if (inputs_1.cacheLocalPath === undefined) {
|
||||||
throw new Error("cache-local-path is not set. Cannot restore cache without a valid cache path.");
|
throw new Error("cache-local-path is not set. Cannot restore cache without a valid cache path.");
|
||||||
}
|
}
|
||||||
const cachePaths = [inputs_1.cacheLocalPath.path];
|
await restoreCacheFromKey(cacheKey, inputs_1.cacheLocalPath.path, exports.STATE_CACHE_MATCHED_KEY, "cache-hit");
|
||||||
if (inputs_1.cachePython) {
|
if (inputs_1.cachePython) {
|
||||||
cachePaths.push(inputs_1.pythonDir);
|
await restoreCacheFromKey(`${cacheKey}-python`, inputs_1.pythonDir, exports.STATE_PYTHON_CACHE_MATCHED_KEY, "python-cache-hit");
|
||||||
}
|
}
|
||||||
|
else {
|
||||||
|
core.setOutput("python-cache-hit", false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
async function restoreCacheFromKey(cacheKey, cachePath, stateKey, outputKey) {
|
||||||
|
core.info(`Trying to restore cache from GitHub Actions cache with key: ${cacheKey}`);
|
||||||
|
let matchedKey;
|
||||||
try {
|
try {
|
||||||
matchedKey = await cache.restoreCache(cachePaths, cacheKey);
|
matchedKey = await cache.restoreCache([cachePath], cacheKey);
|
||||||
}
|
}
|
||||||
catch (err) {
|
catch (err) {
|
||||||
const message = err.message;
|
const message = err.message;
|
||||||
core.warning(message);
|
core.warning(message);
|
||||||
core.setOutput("cache-hit", false);
|
core.setOutput(outputKey, false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
handleMatchResult(matchedKey, cacheKey);
|
handleMatchResult(matchedKey, cacheKey, stateKey, outputKey);
|
||||||
}
|
}
|
||||||
async function computeKeys() {
|
async function computeKeys(pythonVersion) {
|
||||||
let cacheDependencyPathHash = "-";
|
let cacheDependencyPathHash = "-";
|
||||||
if (inputs_1.cacheDependencyGlob !== "") {
|
if (inputs_1.cacheDependencyGlob !== "") {
|
||||||
core.info(`Searching files using cache dependency glob: ${inputs_1.cacheDependencyGlob.split("\n").join(",")}`);
|
core.info(`Searching files using cache dependency glob: ${inputs_1.cacheDependencyGlob.split("\n").join(",")}`);
|
||||||
@@ -90651,50 +90657,22 @@ async function computeKeys() {
|
|||||||
cacheDependencyPathHash = "-no-dependency-glob";
|
cacheDependencyPathHash = "-no-dependency-glob";
|
||||||
}
|
}
|
||||||
const suffix = inputs_1.cacheSuffix ? `-${inputs_1.cacheSuffix}` : "";
|
const suffix = inputs_1.cacheSuffix ? `-${inputs_1.cacheSuffix}` : "";
|
||||||
const pythonVersion = await getPythonVersion();
|
const version = pythonVersion ?? "unknown";
|
||||||
const platform = await (0, platforms_1.getPlatform)();
|
const platform = await (0, platforms_1.getPlatform)();
|
||||||
const osNameVersion = (0, platforms_1.getOSNameVersion)();
|
const osNameVersion = (0, platforms_1.getOSNameVersion)();
|
||||||
const pruned = inputs_1.pruneCache ? "-pruned" : "";
|
const pruned = inputs_1.pruneCache ? "-pruned" : "";
|
||||||
const python = inputs_1.cachePython ? "-py" : "";
|
const python = inputs_1.cachePython ? "-py" : "";
|
||||||
return `setup-uv-${CACHE_VERSION}-${(0, platforms_1.getArch)()}-${platform}-${osNameVersion}-${pythonVersion}${pruned}${python}${cacheDependencyPathHash}${suffix}`;
|
return `setup-uv-${CACHE_VERSION}-${(0, platforms_1.getArch)()}-${platform}-${osNameVersion}-${version}${pruned}${python}${cacheDependencyPathHash}${suffix}`;
|
||||||
}
|
}
|
||||||
async function getPythonVersion() {
|
function handleMatchResult(matchedKey, primaryKey, stateKey, outputKey) {
|
||||||
if (inputs_1.pythonVersion !== "") {
|
|
||||||
return inputs_1.pythonVersion;
|
|
||||||
}
|
|
||||||
let output = "";
|
|
||||||
const options = {
|
|
||||||
listeners: {
|
|
||||||
stdout: (data) => {
|
|
||||||
output += data.toString();
|
|
||||||
},
|
|
||||||
},
|
|
||||||
silent: !core.isDebug(),
|
|
||||||
};
|
|
||||||
try {
|
|
||||||
const execArgs = ["python", "find", "--directory", inputs_1.workingDirectory];
|
|
||||||
await exec.exec("uv", execArgs, options);
|
|
||||||
const pythonPath = output.trim();
|
|
||||||
output = "";
|
|
||||||
await exec.exec(pythonPath, ["--version"], options);
|
|
||||||
// output is like "Python 3.8.10"
|
|
||||||
return output.split(" ")[1].trim();
|
|
||||||
}
|
|
||||||
catch (error) {
|
|
||||||
const err = error;
|
|
||||||
core.debug(`Failed to get python version from uv. Error: ${err.message}`);
|
|
||||||
return "unknown";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
function handleMatchResult(matchedKey, primaryKey) {
|
|
||||||
if (!matchedKey) {
|
if (!matchedKey) {
|
||||||
core.info(`No GitHub Actions cache found for key: ${primaryKey}`);
|
core.info(`No GitHub Actions cache found for key: ${primaryKey}`);
|
||||||
core.setOutput("cache-hit", false);
|
core.setOutput(outputKey, false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
core.saveState(exports.STATE_CACHE_MATCHED_KEY, matchedKey);
|
core.saveState(stateKey, matchedKey);
|
||||||
core.info(`uv cache restored from GitHub Actions cache with key: ${matchedKey}`);
|
core.info(`cache restored from GitHub Actions cache with key: ${matchedKey}`);
|
||||||
core.setOutput("cache-hit", true);
|
core.setOutput(outputKey, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -90868,46 +90846,17 @@ async function saveCache() {
|
|||||||
}
|
}
|
||||||
if (matchedKey === cacheKey) {
|
if (matchedKey === cacheKey) {
|
||||||
core.info(`Cache hit occurred on key ${cacheKey}, not saving cache.`);
|
core.info(`Cache hit occurred on key ${cacheKey}, not saving cache.`);
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
if (inputs_1.pruneCache) {
|
else {
|
||||||
await pruneCache();
|
if (inputs_1.pruneCache) {
|
||||||
|
await pruneCache();
|
||||||
|
}
|
||||||
|
const actualCachePath = getUvCachePath();
|
||||||
|
await saveCacheToKey(cacheKey, actualCachePath, restore_cache_1.STATE_CACHE_MATCHED_KEY, "uv cache", `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.`);
|
||||||
}
|
}
|
||||||
if (inputs_1.cacheLocalPath === undefined) {
|
|
||||||
throw new Error("cache-local-path is not set. Cannot save cache without a valid cache path.");
|
|
||||||
}
|
|
||||||
let actualCachePath = inputs_1.cacheLocalPath.path;
|
|
||||||
if (process.env.UV_CACHE_DIR &&
|
|
||||||
process.env.UV_CACHE_DIR !== inputs_1.cacheLocalPath.path) {
|
|
||||||
core.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_1.cacheLocalPath.path}".`);
|
|
||||||
actualCachePath = process.env.UV_CACHE_DIR;
|
|
||||||
}
|
|
||||||
core.info(`Saving cache path: ${actualCachePath}`);
|
|
||||||
if (!fs.existsSync(actualCachePath) && !inputs_1.ignoreNothingToCache) {
|
|
||||||
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.`);
|
|
||||||
}
|
|
||||||
const cachePaths = [actualCachePath];
|
|
||||||
if (inputs_1.cachePython) {
|
if (inputs_1.cachePython) {
|
||||||
core.info(`Including Python cache path: ${inputs_1.pythonDir}`);
|
const pythonCacheKey = `${cacheKey}-python`;
|
||||||
if (!fs.existsSync(inputs_1.pythonDir) && !inputs_1.ignoreNothingToCache) {
|
await saveCacheToKey(pythonCacheKey, inputs_1.pythonDir, restore_cache_1.STATE_PYTHON_CACHE_MATCHED_KEY, "Python cache", `Python cache path ${inputs_1.pythonDir} does not exist on disk. This likely indicates that there are no Python installations to cache. Consider disabling the cache input if it is not needed.`);
|
||||||
throw new Error(`Python cache path ${inputs_1.pythonDir} 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.`);
|
|
||||||
}
|
|
||||||
cachePaths.push(inputs_1.pythonDir);
|
|
||||||
}
|
|
||||||
core.info(`Final cache paths: ${cachePaths.join(", ")}`);
|
|
||||||
try {
|
|
||||||
await cache.saveCache(cachePaths, cacheKey);
|
|
||||||
core.info(`cache saved with the key: ${cacheKey}`);
|
|
||||||
}
|
|
||||||
catch (e) {
|
|
||||||
if (e instanceof Error &&
|
|
||||||
e.message ===
|
|
||||||
"Path Validation Error: Path(s) specified in the action for caching do(es) not exist, hence no cache is being saved.") {
|
|
||||||
core.info("No cacheable paths were found. Ignoring because ignore-nothing-to-save is enabled.");
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
async function pruneCache() {
|
async function pruneCache() {
|
||||||
@@ -90923,6 +90872,42 @@ async function pruneCache() {
|
|||||||
const uvPath = core.getState(constants_1.STATE_UV_PATH);
|
const uvPath = core.getState(constants_1.STATE_UV_PATH);
|
||||||
await exec.exec(uvPath, execArgs, options);
|
await exec.exec(uvPath, execArgs, options);
|
||||||
}
|
}
|
||||||
|
function getUvCachePath() {
|
||||||
|
if (inputs_1.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_1.cacheLocalPath.path) {
|
||||||
|
core.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_1.cacheLocalPath.path}".`);
|
||||||
|
return process.env.UV_CACHE_DIR;
|
||||||
|
}
|
||||||
|
return inputs_1.cacheLocalPath.path;
|
||||||
|
}
|
||||||
|
async function saveCacheToKey(cacheKey, cachePath, stateKey, cacheName, pathNotExistErrorMessage) {
|
||||||
|
const matchedKey = core.getState(stateKey);
|
||||||
|
if (matchedKey === cacheKey) {
|
||||||
|
core.info(`${cacheName} hit occurred on key ${cacheKey}, not saving cache.`);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
core.info(`Including ${cacheName} path: ${cachePath}`);
|
||||||
|
if (!fs.existsSync(cachePath) && !inputs_1.ignoreNothingToCache) {
|
||||||
|
throw new Error(pathNotExistErrorMessage);
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
await cache.saveCache([cachePath], cacheKey);
|
||||||
|
core.info(`${cacheName} saved with key: ${cacheKey}`);
|
||||||
|
}
|
||||||
|
catch (e) {
|
||||||
|
if (e instanceof Error &&
|
||||||
|
e.message ===
|
||||||
|
"Path Validation Error: Path(s) specified in the action for caching do(es) not exist, hence no cache is being saved.") {
|
||||||
|
core.info(`No cacheable ${cacheName} paths were found. Ignoring because ignore-nothing-to-save is enabled.`);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
run();
|
run();
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
104
dist/setup/index.js
generated
vendored
104
dist/setup/index.js
generated
vendored
@@ -91502,46 +91502,52 @@ var __importStar = (this && this.__importStar) || (function () {
|
|||||||
};
|
};
|
||||||
})();
|
})();
|
||||||
Object.defineProperty(exports, "__esModule", ({ value: true }));
|
Object.defineProperty(exports, "__esModule", ({ value: true }));
|
||||||
exports.STATE_CACHE_MATCHED_KEY = exports.STATE_CACHE_KEY = void 0;
|
exports.STATE_PYTHON_CACHE_MATCHED_KEY = exports.STATE_CACHE_MATCHED_KEY = exports.STATE_CACHE_KEY = void 0;
|
||||||
exports.restoreCache = restoreCache;
|
exports.restoreCache = restoreCache;
|
||||||
const cache = __importStar(__nccwpck_require__(5116));
|
const cache = __importStar(__nccwpck_require__(5116));
|
||||||
const core = __importStar(__nccwpck_require__(7484));
|
const core = __importStar(__nccwpck_require__(7484));
|
||||||
const exec = __importStar(__nccwpck_require__(5236));
|
|
||||||
const hash_files_1 = __nccwpck_require__(9660);
|
const hash_files_1 = __nccwpck_require__(9660);
|
||||||
const inputs_1 = __nccwpck_require__(9612);
|
const inputs_1 = __nccwpck_require__(9612);
|
||||||
const platforms_1 = __nccwpck_require__(8361);
|
const platforms_1 = __nccwpck_require__(8361);
|
||||||
exports.STATE_CACHE_KEY = "cache-key";
|
exports.STATE_CACHE_KEY = "cache-key";
|
||||||
exports.STATE_CACHE_MATCHED_KEY = "cache-matched-key";
|
exports.STATE_CACHE_MATCHED_KEY = "cache-matched-key";
|
||||||
|
exports.STATE_PYTHON_CACHE_MATCHED_KEY = "python-cache-matched-key";
|
||||||
const CACHE_VERSION = "2";
|
const CACHE_VERSION = "2";
|
||||||
async function restoreCache() {
|
async function restoreCache(pythonVersion) {
|
||||||
const cacheKey = await computeKeys();
|
const cacheKey = await computeKeys(pythonVersion);
|
||||||
core.saveState(exports.STATE_CACHE_KEY, cacheKey);
|
core.saveState(exports.STATE_CACHE_KEY, cacheKey);
|
||||||
core.setOutput("cache-key", cacheKey);
|
core.setOutput("cache-key", cacheKey);
|
||||||
if (!inputs_1.restoreCache) {
|
if (!inputs_1.restoreCache) {
|
||||||
core.info("restore-cache is false. Skipping restore cache step.");
|
core.info("restore-cache is false. Skipping restore cache step.");
|
||||||
|
core.setOutput("python-cache-hit", false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
let matchedKey;
|
|
||||||
core.info(`Trying to restore uv cache from GitHub Actions cache with key: ${cacheKey}`);
|
|
||||||
if (inputs_1.cacheLocalPath === undefined) {
|
if (inputs_1.cacheLocalPath === undefined) {
|
||||||
throw new Error("cache-local-path is not set. Cannot restore cache without a valid cache path.");
|
throw new Error("cache-local-path is not set. Cannot restore cache without a valid cache path.");
|
||||||
}
|
}
|
||||||
const cachePaths = [inputs_1.cacheLocalPath.path];
|
await restoreCacheFromKey(cacheKey, inputs_1.cacheLocalPath.path, exports.STATE_CACHE_MATCHED_KEY, "cache-hit");
|
||||||
if (inputs_1.cachePython) {
|
if (inputs_1.cachePython) {
|
||||||
cachePaths.push(inputs_1.pythonDir);
|
await restoreCacheFromKey(`${cacheKey}-python`, inputs_1.pythonDir, exports.STATE_PYTHON_CACHE_MATCHED_KEY, "python-cache-hit");
|
||||||
}
|
}
|
||||||
|
else {
|
||||||
|
core.setOutput("python-cache-hit", false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
async function restoreCacheFromKey(cacheKey, cachePath, stateKey, outputKey) {
|
||||||
|
core.info(`Trying to restore cache from GitHub Actions cache with key: ${cacheKey}`);
|
||||||
|
let matchedKey;
|
||||||
try {
|
try {
|
||||||
matchedKey = await cache.restoreCache(cachePaths, cacheKey);
|
matchedKey = await cache.restoreCache([cachePath], cacheKey);
|
||||||
}
|
}
|
||||||
catch (err) {
|
catch (err) {
|
||||||
const message = err.message;
|
const message = err.message;
|
||||||
core.warning(message);
|
core.warning(message);
|
||||||
core.setOutput("cache-hit", false);
|
core.setOutput(outputKey, false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
handleMatchResult(matchedKey, cacheKey);
|
handleMatchResult(matchedKey, cacheKey, stateKey, outputKey);
|
||||||
}
|
}
|
||||||
async function computeKeys() {
|
async function computeKeys(pythonVersion) {
|
||||||
let cacheDependencyPathHash = "-";
|
let cacheDependencyPathHash = "-";
|
||||||
if (inputs_1.cacheDependencyGlob !== "") {
|
if (inputs_1.cacheDependencyGlob !== "") {
|
||||||
core.info(`Searching files using cache dependency glob: ${inputs_1.cacheDependencyGlob.split("\n").join(",")}`);
|
core.info(`Searching files using cache dependency glob: ${inputs_1.cacheDependencyGlob.split("\n").join(",")}`);
|
||||||
@@ -91554,50 +91560,22 @@ async function computeKeys() {
|
|||||||
cacheDependencyPathHash = "-no-dependency-glob";
|
cacheDependencyPathHash = "-no-dependency-glob";
|
||||||
}
|
}
|
||||||
const suffix = inputs_1.cacheSuffix ? `-${inputs_1.cacheSuffix}` : "";
|
const suffix = inputs_1.cacheSuffix ? `-${inputs_1.cacheSuffix}` : "";
|
||||||
const pythonVersion = await getPythonVersion();
|
const version = pythonVersion ?? "unknown";
|
||||||
const platform = await (0, platforms_1.getPlatform)();
|
const platform = await (0, platforms_1.getPlatform)();
|
||||||
const osNameVersion = (0, platforms_1.getOSNameVersion)();
|
const osNameVersion = (0, platforms_1.getOSNameVersion)();
|
||||||
const pruned = inputs_1.pruneCache ? "-pruned" : "";
|
const pruned = inputs_1.pruneCache ? "-pruned" : "";
|
||||||
const python = inputs_1.cachePython ? "-py" : "";
|
const python = inputs_1.cachePython ? "-py" : "";
|
||||||
return `setup-uv-${CACHE_VERSION}-${(0, platforms_1.getArch)()}-${platform}-${osNameVersion}-${pythonVersion}${pruned}${python}${cacheDependencyPathHash}${suffix}`;
|
return `setup-uv-${CACHE_VERSION}-${(0, platforms_1.getArch)()}-${platform}-${osNameVersion}-${version}${pruned}${python}${cacheDependencyPathHash}${suffix}`;
|
||||||
}
|
}
|
||||||
async function getPythonVersion() {
|
function handleMatchResult(matchedKey, primaryKey, stateKey, outputKey) {
|
||||||
if (inputs_1.pythonVersion !== "") {
|
|
||||||
return inputs_1.pythonVersion;
|
|
||||||
}
|
|
||||||
let output = "";
|
|
||||||
const options = {
|
|
||||||
listeners: {
|
|
||||||
stdout: (data) => {
|
|
||||||
output += data.toString();
|
|
||||||
},
|
|
||||||
},
|
|
||||||
silent: !core.isDebug(),
|
|
||||||
};
|
|
||||||
try {
|
|
||||||
const execArgs = ["python", "find", "--directory", inputs_1.workingDirectory];
|
|
||||||
await exec.exec("uv", execArgs, options);
|
|
||||||
const pythonPath = output.trim();
|
|
||||||
output = "";
|
|
||||||
await exec.exec(pythonPath, ["--version"], options);
|
|
||||||
// output is like "Python 3.8.10"
|
|
||||||
return output.split(" ")[1].trim();
|
|
||||||
}
|
|
||||||
catch (error) {
|
|
||||||
const err = error;
|
|
||||||
core.debug(`Failed to get python version from uv. Error: ${err.message}`);
|
|
||||||
return "unknown";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
function handleMatchResult(matchedKey, primaryKey) {
|
|
||||||
if (!matchedKey) {
|
if (!matchedKey) {
|
||||||
core.info(`No GitHub Actions cache found for key: ${primaryKey}`);
|
core.info(`No GitHub Actions cache found for key: ${primaryKey}`);
|
||||||
core.setOutput("cache-hit", false);
|
core.setOutput(outputKey, false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
core.saveState(exports.STATE_CACHE_MATCHED_KEY, matchedKey);
|
core.saveState(stateKey, matchedKey);
|
||||||
core.info(`uv cache restored from GitHub Actions cache with key: ${matchedKey}`);
|
core.info(`cache restored from GitHub Actions cache with key: ${matchedKey}`);
|
||||||
core.setOutput("cache-hit", true);
|
core.setOutput(outputKey, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -96267,6 +96245,34 @@ const constants_1 = __nccwpck_require__(6156);
|
|||||||
const inputs_1 = __nccwpck_require__(9612);
|
const inputs_1 = __nccwpck_require__(9612);
|
||||||
const platforms_1 = __nccwpck_require__(8361);
|
const platforms_1 = __nccwpck_require__(8361);
|
||||||
const resolve_1 = __nccwpck_require__(6772);
|
const resolve_1 = __nccwpck_require__(6772);
|
||||||
|
async function getPythonVersion() {
|
||||||
|
if (inputs_1.pythonVersion !== "") {
|
||||||
|
return inputs_1.pythonVersion;
|
||||||
|
}
|
||||||
|
let output = "";
|
||||||
|
const options = {
|
||||||
|
listeners: {
|
||||||
|
stdout: (data) => {
|
||||||
|
output += data.toString();
|
||||||
|
},
|
||||||
|
},
|
||||||
|
silent: !core.isDebug(),
|
||||||
|
};
|
||||||
|
try {
|
||||||
|
const execArgs = ["python", "find", "--directory", inputs_1.workingDirectory];
|
||||||
|
await exec.exec("uv", execArgs, options);
|
||||||
|
const pythonPath = output.trim();
|
||||||
|
output = "";
|
||||||
|
await exec.exec(pythonPath, ["--version"], options);
|
||||||
|
// output is like "Python 3.8.10"
|
||||||
|
return output.split(" ")[1].trim();
|
||||||
|
}
|
||||||
|
catch (error) {
|
||||||
|
const err = error;
|
||||||
|
core.debug(`Failed to get python version from uv. Error: ${err.message}`);
|
||||||
|
return "unknown";
|
||||||
|
}
|
||||||
|
}
|
||||||
async function run() {
|
async function run() {
|
||||||
detectEmptyWorkdir();
|
detectEmptyWorkdir();
|
||||||
const platform = await (0, platforms_1.getPlatform)();
|
const platform = await (0, platforms_1.getPlatform)();
|
||||||
@@ -96290,8 +96296,10 @@ async function run() {
|
|||||||
core.setOutput("uv-version", setupResult.version);
|
core.setOutput("uv-version", setupResult.version);
|
||||||
core.saveState(constants_1.STATE_UV_VERSION, setupResult.version);
|
core.saveState(constants_1.STATE_UV_VERSION, setupResult.version);
|
||||||
core.info(`Successfully installed uv version ${setupResult.version}`);
|
core.info(`Successfully installed uv version ${setupResult.version}`);
|
||||||
|
const pythonVersion = await getPythonVersion();
|
||||||
|
core.setOutput("python-version", pythonVersion);
|
||||||
if (inputs_1.enableCache) {
|
if (inputs_1.enableCache) {
|
||||||
await (0, restore_cache_1.restoreCache)();
|
await (0, restore_cache_1.restoreCache)(pythonVersion);
|
||||||
}
|
}
|
||||||
// https://github.com/nodejs/node/issues/56645#issuecomment-3077594952
|
// https://github.com/nodejs/node/issues/56645#issuecomment-3077594952
|
||||||
await new Promise((resolve) => setTimeout(resolve, 50));
|
await new Promise((resolve) => setTimeout(resolve, 50));
|
||||||
|
|||||||
99
src/cache/restore-cache.ts
vendored
99
src/cache/restore-cache.ts
vendored
@@ -1,6 +1,5 @@
|
|||||||
import * as cache from "@actions/cache";
|
import * as cache from "@actions/cache";
|
||||||
import * as core from "@actions/core";
|
import * as core from "@actions/core";
|
||||||
import * as exec from "@actions/exec";
|
|
||||||
import { hashFiles } from "../hash/hash-files";
|
import { hashFiles } from "../hash/hash-files";
|
||||||
import {
|
import {
|
||||||
cacheDependencyGlob,
|
cacheDependencyGlob,
|
||||||
@@ -9,52 +8,75 @@ import {
|
|||||||
cacheSuffix,
|
cacheSuffix,
|
||||||
pruneCache,
|
pruneCache,
|
||||||
pythonDir,
|
pythonDir,
|
||||||
pythonVersion as pythonVersionInput,
|
|
||||||
restoreCache as shouldRestoreCache,
|
restoreCache as shouldRestoreCache,
|
||||||
workingDirectory,
|
|
||||||
} from "../utils/inputs";
|
} from "../utils/inputs";
|
||||||
import { getArch, getOSNameVersion, getPlatform } from "../utils/platforms";
|
import { getArch, getOSNameVersion, getPlatform } from "../utils/platforms";
|
||||||
|
|
||||||
export const STATE_CACHE_KEY = "cache-key";
|
export const STATE_CACHE_KEY = "cache-key";
|
||||||
export const STATE_CACHE_MATCHED_KEY = "cache-matched-key";
|
export const STATE_CACHE_MATCHED_KEY = "cache-matched-key";
|
||||||
|
export const STATE_PYTHON_CACHE_MATCHED_KEY = "python-cache-matched-key";
|
||||||
|
|
||||||
const CACHE_VERSION = "2";
|
const CACHE_VERSION = "2";
|
||||||
|
|
||||||
export async function restoreCache(): Promise<void> {
|
export async function restoreCache(pythonVersion?: string): Promise<void> {
|
||||||
const cacheKey = await computeKeys();
|
const cacheKey = await computeKeys(pythonVersion);
|
||||||
core.saveState(STATE_CACHE_KEY, cacheKey);
|
core.saveState(STATE_CACHE_KEY, cacheKey);
|
||||||
core.setOutput("cache-key", cacheKey);
|
core.setOutput("cache-key", cacheKey);
|
||||||
|
|
||||||
if (!shouldRestoreCache) {
|
if (!shouldRestoreCache) {
|
||||||
core.info("restore-cache is false. Skipping restore cache step.");
|
core.info("restore-cache is false. Skipping restore cache step.");
|
||||||
|
core.setOutput("python-cache-hit", false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
let matchedKey: string | undefined;
|
|
||||||
core.info(
|
|
||||||
`Trying to restore uv cache from GitHub Actions cache with key: ${cacheKey}`,
|
|
||||||
);
|
|
||||||
if (cacheLocalPath === undefined) {
|
if (cacheLocalPath === undefined) {
|
||||||
throw new Error(
|
throw new Error(
|
||||||
"cache-local-path is not set. Cannot restore cache without a valid cache path.",
|
"cache-local-path is not set. Cannot restore cache without a valid cache path.",
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
const cachePaths = [cacheLocalPath.path];
|
|
||||||
|
await restoreCacheFromKey(
|
||||||
|
cacheKey,
|
||||||
|
cacheLocalPath.path,
|
||||||
|
STATE_CACHE_MATCHED_KEY,
|
||||||
|
"cache-hit",
|
||||||
|
);
|
||||||
|
|
||||||
if (cachePython) {
|
if (cachePython) {
|
||||||
cachePaths.push(pythonDir);
|
await restoreCacheFromKey(
|
||||||
|
`${cacheKey}-python`,
|
||||||
|
pythonDir,
|
||||||
|
STATE_PYTHON_CACHE_MATCHED_KEY,
|
||||||
|
"python-cache-hit",
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
core.setOutput("python-cache-hit", false);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function restoreCacheFromKey(
|
||||||
|
cacheKey: string,
|
||||||
|
cachePath: string,
|
||||||
|
stateKey: string,
|
||||||
|
outputKey: string,
|
||||||
|
): Promise<void> {
|
||||||
|
core.info(
|
||||||
|
`Trying to restore cache from GitHub Actions cache with key: ${cacheKey}`,
|
||||||
|
);
|
||||||
|
let matchedKey: string | undefined;
|
||||||
try {
|
try {
|
||||||
matchedKey = await cache.restoreCache(cachePaths, cacheKey);
|
matchedKey = await cache.restoreCache([cachePath], cacheKey);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
const message = (err as Error).message;
|
const message = (err as Error).message;
|
||||||
core.warning(message);
|
core.warning(message);
|
||||||
core.setOutput("cache-hit", false);
|
core.setOutput(outputKey, false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
handleMatchResult(matchedKey, cacheKey);
|
handleMatchResult(matchedKey, cacheKey, stateKey, outputKey);
|
||||||
}
|
}
|
||||||
|
|
||||||
async function computeKeys(): Promise<string> {
|
async function computeKeys(pythonVersion?: string): Promise<string> {
|
||||||
let cacheDependencyPathHash = "-";
|
let cacheDependencyPathHash = "-";
|
||||||
if (cacheDependencyGlob !== "") {
|
if (cacheDependencyGlob !== "") {
|
||||||
core.info(
|
core.info(
|
||||||
@@ -71,58 +93,27 @@ async function computeKeys(): Promise<string> {
|
|||||||
cacheDependencyPathHash = "-no-dependency-glob";
|
cacheDependencyPathHash = "-no-dependency-glob";
|
||||||
}
|
}
|
||||||
const suffix = cacheSuffix ? `-${cacheSuffix}` : "";
|
const suffix = cacheSuffix ? `-${cacheSuffix}` : "";
|
||||||
const pythonVersion = await getPythonVersion();
|
const version = pythonVersion ?? "unknown";
|
||||||
const platform = await getPlatform();
|
const platform = await getPlatform();
|
||||||
const osNameVersion = getOSNameVersion();
|
const osNameVersion = getOSNameVersion();
|
||||||
const pruned = pruneCache ? "-pruned" : "";
|
const pruned = pruneCache ? "-pruned" : "";
|
||||||
const python = cachePython ? "-py" : "";
|
const python = cachePython ? "-py" : "";
|
||||||
return `setup-uv-${CACHE_VERSION}-${getArch()}-${platform}-${osNameVersion}-${pythonVersion}${pruned}${python}${cacheDependencyPathHash}${suffix}`;
|
return `setup-uv-${CACHE_VERSION}-${getArch()}-${platform}-${osNameVersion}-${version}${pruned}${python}${cacheDependencyPathHash}${suffix}`;
|
||||||
}
|
|
||||||
|
|
||||||
async function getPythonVersion(): Promise<string> {
|
|
||||||
if (pythonVersionInput !== "") {
|
|
||||||
return pythonVersionInput;
|
|
||||||
}
|
|
||||||
|
|
||||||
let output = "";
|
|
||||||
const options: exec.ExecOptions = {
|
|
||||||
listeners: {
|
|
||||||
stdout: (data: Buffer) => {
|
|
||||||
output += data.toString();
|
|
||||||
},
|
|
||||||
},
|
|
||||||
silent: !core.isDebug(),
|
|
||||||
};
|
|
||||||
|
|
||||||
try {
|
|
||||||
const execArgs = ["python", "find", "--directory", workingDirectory];
|
|
||||||
await exec.exec("uv", execArgs, options);
|
|
||||||
const pythonPath = output.trim();
|
|
||||||
|
|
||||||
output = "";
|
|
||||||
await exec.exec(pythonPath, ["--version"], options);
|
|
||||||
// output is like "Python 3.8.10"
|
|
||||||
return output.split(" ")[1].trim();
|
|
||||||
} catch (error) {
|
|
||||||
const err = error as Error;
|
|
||||||
core.debug(`Failed to get python version from uv. Error: ${err.message}`);
|
|
||||||
return "unknown";
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function handleMatchResult(
|
function handleMatchResult(
|
||||||
matchedKey: string | undefined,
|
matchedKey: string | undefined,
|
||||||
primaryKey: string,
|
primaryKey: string,
|
||||||
|
stateKey: string,
|
||||||
|
outputKey: string,
|
||||||
): void {
|
): void {
|
||||||
if (!matchedKey) {
|
if (!matchedKey) {
|
||||||
core.info(`No GitHub Actions cache found for key: ${primaryKey}`);
|
core.info(`No GitHub Actions cache found for key: ${primaryKey}`);
|
||||||
core.setOutput("cache-hit", false);
|
core.setOutput(outputKey, false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
core.saveState(STATE_CACHE_MATCHED_KEY, matchedKey);
|
core.saveState(stateKey, matchedKey);
|
||||||
core.info(
|
core.info(`cache restored from GitHub Actions cache with key: ${matchedKey}`);
|
||||||
`uv cache restored from GitHub Actions cache with key: ${matchedKey}`,
|
core.setOutput(outputKey, true);
|
||||||
);
|
|
||||||
core.setOutput("cache-hit", true);
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ import * as pep440 from "@renovatebot/pep440";
|
|||||||
import {
|
import {
|
||||||
STATE_CACHE_KEY,
|
STATE_CACHE_KEY,
|
||||||
STATE_CACHE_MATCHED_KEY,
|
STATE_CACHE_MATCHED_KEY,
|
||||||
|
STATE_PYTHON_CACHE_MATCHED_KEY,
|
||||||
} from "./cache/restore-cache";
|
} from "./cache/restore-cache";
|
||||||
import { STATE_UV_PATH, STATE_UV_VERSION } from "./utils/constants";
|
import { STATE_UV_PATH, STATE_UV_VERSION } from "./utils/constants";
|
||||||
import {
|
import {
|
||||||
@@ -52,63 +53,30 @@ async function saveCache(): Promise<void> {
|
|||||||
}
|
}
|
||||||
if (matchedKey === cacheKey) {
|
if (matchedKey === cacheKey) {
|
||||||
core.info(`Cache hit occurred on key ${cacheKey}, not saving cache.`);
|
core.info(`Cache hit occurred on key ${cacheKey}, not saving cache.`);
|
||||||
return;
|
} else {
|
||||||
}
|
if (shouldPruneCache) {
|
||||||
|
await pruneCache();
|
||||||
|
}
|
||||||
|
|
||||||
if (shouldPruneCache) {
|
const actualCachePath = getUvCachePath();
|
||||||
await pruneCache();
|
await saveCacheToKey(
|
||||||
}
|
cacheKey,
|
||||||
|
actualCachePath,
|
||||||
if (cacheLocalPath === undefined) {
|
STATE_CACHE_MATCHED_KEY,
|
||||||
throw new Error(
|
"uv cache",
|
||||||
"cache-local-path is not set. Cannot save cache without a valid cache path.",
|
|
||||||
);
|
|
||||||
}
|
|
||||||
let actualCachePath = cacheLocalPath.path;
|
|
||||||
if (
|
|
||||||
process.env.UV_CACHE_DIR &&
|
|
||||||
process.env.UV_CACHE_DIR !== cacheLocalPath.path
|
|
||||||
) {
|
|
||||||
core.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 "${cacheLocalPath.path}".`,
|
|
||||||
);
|
|
||||||
actualCachePath = process.env.UV_CACHE_DIR;
|
|
||||||
}
|
|
||||||
|
|
||||||
core.info(`Saving cache path: ${actualCachePath}`);
|
|
||||||
if (!fs.existsSync(actualCachePath) && !ignoreNothingToCache) {
|
|
||||||
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.`,
|
`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.`,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
const cachePaths = [actualCachePath];
|
|
||||||
if (cachePython) {
|
if (cachePython) {
|
||||||
core.info(`Including Python cache path: ${pythonDir}`);
|
const pythonCacheKey = `${cacheKey}-python`;
|
||||||
if (!fs.existsSync(pythonDir) && !ignoreNothingToCache) {
|
await saveCacheToKey(
|
||||||
throw new Error(
|
pythonCacheKey,
|
||||||
`Python cache path ${pythonDir} 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.`,
|
pythonDir,
|
||||||
);
|
STATE_PYTHON_CACHE_MATCHED_KEY,
|
||||||
}
|
"Python cache",
|
||||||
cachePaths.push(pythonDir);
|
`Python cache path ${pythonDir} does not exist on disk. This likely indicates that there are no Python installations to cache. Consider disabling the cache input if it is not needed.`,
|
||||||
}
|
);
|
||||||
|
|
||||||
core.info(`Final cache paths: ${cachePaths.join(", ")}`);
|
|
||||||
try {
|
|
||||||
await cache.saveCache(cachePaths, cacheKey);
|
|
||||||
core.info(`cache saved with the key: ${cacheKey}`);
|
|
||||||
} catch (e) {
|
|
||||||
if (
|
|
||||||
e instanceof Error &&
|
|
||||||
e.message ===
|
|
||||||
"Path Validation Error: Path(s) specified in the action for caching do(es) not exist, hence no cache is being saved."
|
|
||||||
) {
|
|
||||||
core.info(
|
|
||||||
"No cacheable paths were found. Ignoring because ignore-nothing-to-save is enabled.",
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -128,4 +96,61 @@ async function pruneCache(): Promise<void> {
|
|||||||
await exec.exec(uvPath, execArgs, options);
|
await exec.exec(uvPath, execArgs, options);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function getUvCachePath(): string {
|
||||||
|
if (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 !== cacheLocalPath.path
|
||||||
|
) {
|
||||||
|
core.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 "${cacheLocalPath.path}".`,
|
||||||
|
);
|
||||||
|
return process.env.UV_CACHE_DIR;
|
||||||
|
}
|
||||||
|
return cacheLocalPath.path;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function saveCacheToKey(
|
||||||
|
cacheKey: string,
|
||||||
|
cachePath: string,
|
||||||
|
stateKey: string,
|
||||||
|
cacheName: string,
|
||||||
|
pathNotExistErrorMessage: string,
|
||||||
|
): Promise<void> {
|
||||||
|
const matchedKey = core.getState(stateKey);
|
||||||
|
|
||||||
|
if (matchedKey === cacheKey) {
|
||||||
|
core.info(
|
||||||
|
`${cacheName} hit occurred on key ${cacheKey}, not saving cache.`,
|
||||||
|
);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
core.info(`Including ${cacheName} path: ${cachePath}`);
|
||||||
|
if (!fs.existsSync(cachePath) && !ignoreNothingToCache) {
|
||||||
|
throw new Error(pathNotExistErrorMessage);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
await cache.saveCache([cachePath], cacheKey);
|
||||||
|
core.info(`${cacheName} saved with key: ${cacheKey}`);
|
||||||
|
} catch (e) {
|
||||||
|
if (
|
||||||
|
e instanceof Error &&
|
||||||
|
e.message ===
|
||||||
|
"Path Validation Error: Path(s) specified in the action for caching do(es) not exist, hence no cache is being saved."
|
||||||
|
) {
|
||||||
|
core.info(
|
||||||
|
`No cacheable ${cacheName} paths were found. Ignoring because ignore-nothing-to-save is enabled.`,
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
run();
|
run();
|
||||||
|
|||||||
@@ -36,6 +36,37 @@ import {
|
|||||||
} from "./utils/platforms";
|
} from "./utils/platforms";
|
||||||
import { getUvVersionFromFile } from "./version/resolve";
|
import { getUvVersionFromFile } from "./version/resolve";
|
||||||
|
|
||||||
|
async function getPythonVersion(): Promise<string> {
|
||||||
|
if (pythonVersion !== "") {
|
||||||
|
return pythonVersion;
|
||||||
|
}
|
||||||
|
|
||||||
|
let output = "";
|
||||||
|
const options: exec.ExecOptions = {
|
||||||
|
listeners: {
|
||||||
|
stdout: (data: Buffer) => {
|
||||||
|
output += data.toString();
|
||||||
|
},
|
||||||
|
},
|
||||||
|
silent: !core.isDebug(),
|
||||||
|
};
|
||||||
|
|
||||||
|
try {
|
||||||
|
const execArgs = ["python", "find", "--directory", workingDirectory];
|
||||||
|
await exec.exec("uv", execArgs, options);
|
||||||
|
const pythonPath = output.trim();
|
||||||
|
|
||||||
|
output = "";
|
||||||
|
await exec.exec(pythonPath, ["--version"], options);
|
||||||
|
// output is like "Python 3.8.10"
|
||||||
|
return output.split(" ")[1].trim();
|
||||||
|
} catch (error) {
|
||||||
|
const err = error as Error;
|
||||||
|
core.debug(`Failed to get python version from uv. Error: ${err.message}`);
|
||||||
|
return "unknown";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
async function run(): Promise<void> {
|
async function run(): Promise<void> {
|
||||||
detectEmptyWorkdir();
|
detectEmptyWorkdir();
|
||||||
const platform = await getPlatform();
|
const platform = await getPlatform();
|
||||||
@@ -63,8 +94,11 @@ async function run(): Promise<void> {
|
|||||||
core.saveState(STATE_UV_VERSION, setupResult.version);
|
core.saveState(STATE_UV_VERSION, setupResult.version);
|
||||||
core.info(`Successfully installed uv version ${setupResult.version}`);
|
core.info(`Successfully installed uv version ${setupResult.version}`);
|
||||||
|
|
||||||
|
const pythonVersion = await getPythonVersion();
|
||||||
|
core.setOutput("python-version", pythonVersion);
|
||||||
|
|
||||||
if (enableCache) {
|
if (enableCache) {
|
||||||
await restoreCache();
|
await restoreCache(pythonVersion);
|
||||||
}
|
}
|
||||||
// https://github.com/nodejs/node/issues/56645#issuecomment-3077594952
|
// https://github.com/nodejs/node/issues/56645#issuecomment-3077594952
|
||||||
await new Promise((resolve) => setTimeout(resolve, 50));
|
await new Promise((resolve) => setTimeout(resolve, 50));
|
||||||
|
|||||||
Reference in New Issue
Block a user