3
0
Fork 0
mirror of https://github.com/Swatinem/rust-cache synced 2025-04-05 21:24:07 +00:00

merge all the caches and simplify

This commit is contained in:
Arpad Borsos 2020-10-03 18:39:38 +02:00
parent f77cb1be47
commit e8e3c57b3b
8 changed files with 197 additions and 259 deletions

View file

@ -9,20 +9,12 @@ sensible defaults.
- uses: Swatinem/rust-cache@v1 - uses: Swatinem/rust-cache@v1
``` ```
### Registry Cache ### Cache Details
The cache currently caches the following directories:
- `~/.cargo/registry/index` - `~/.cargo/registry/index`
- `~/.cargo/registry/cache` - `~/.cargo/registry/cache`
This cache is automatically keyed by hashing the `Cargo.lock` / `Cargo.toml`
files. Before persisting, the cache is cleaned of intermediate artifacts and
unneeded dependencies.
**TODO**: The `~/.cargo/git/db` database is not yet persisted, support will be
added at a later point.
### Target Cache
- `./target` - `./target`
This cache is automatically keyed by: This cache is automatically keyed by:
@ -31,7 +23,13 @@ This cache is automatically keyed by:
- the rustc release / host / hash, and - the rustc release / host / hash, and
- a hash of the `Cargo.lock` / `Cargo.toml` files. - a hash of the `Cargo.lock` / `Cargo.toml` files.
Before persisting, the cache is cleaned of anything that is not a needed An additional input `key` can be provided if the builtin keys are not sufficient.
dependency. In particular, no caching of workspace crates will be done. For
Before persisting, the cache is cleaned of intermediate artifacts and
anything that is not a workspace dependency.
In particular, no caching of workspace crates will be done. For
this reason, this action will automatically set `CARGO_INCREMENTAL=0` to this reason, this action will automatically set `CARGO_INCREMENTAL=0` to
disable incremental compilation. disable incremental compilation.
**TODO**: The `~/.cargo/git/db` database is not yet persisted, support will be
added at a later point.

View file

@ -1,9 +1,9 @@
name: "Rust Cache" name: "Rust Cache"
description: "A GitHub Action that implements smart caching for rust/cargo projects" description: "A GitHub Action that implements smart caching for rust/cargo projects with sensible defaults."
author: "Arpad Borsos <arpad.borsos@googlemail.com>" author: "Arpad Borsos <arpad.borsos@googlemail.com>"
inputs: inputs:
key: key:
description: "An explicit key for restoring and saving the target cache" description: "An additional key for the cache"
required: false required: false
runs: runs:
using: "node12" using: "node12"

93
dist/restore/index.js vendored
View file

@ -54612,6 +54612,8 @@ var __asyncValues = (undefined && undefined.__asyncValues) || function (o) {
const stateKey = "RUST_CACHE_KEY";
const stateHash = "RUST_CACHE_HASH";
const home = external_os_default().homedir(); const home = external_os_default().homedir();
const paths = { const paths = {
index: external_path_default().join(home, ".cargo/registry/index"), index: external_path_default().join(home, ".cargo/registry/index"),
@ -54623,39 +54625,31 @@ const RefKey = "GITHUB_REF";
function isValidEvent() { function isValidEvent() {
return RefKey in process.env && Boolean(process.env[RefKey]); return RefKey in process.env && Boolean(process.env[RefKey]);
} }
async function getCaches() { async function getCacheConfig() {
const rustKey = await getRustKey(); let lockHash = core.getState(stateHash);
let lockHash = core.getState("lockHash");
if (!lockHash) { if (!lockHash) {
lockHash = await getLockfileHash(); lockHash = await getLockfileHash();
core.saveState("lockHash", lockHash); core.saveState(stateHash, lockHash);
} }
let targetKey = core.getInput("key"); let key = `v0-rust-`;
if (targetKey) { let inputKey = core.getInput("key");
targetKey = `${targetKey}-`; if (inputKey) {
key += `${inputKey}-`;
} }
const job = process.env.GITHUB_JOB; const job = process.env.GITHUB_JOB;
if (job) { if (job) {
targetKey = `${job}-${targetKey}`; key += `${job}-`;
} }
const registry = `v0-registry`; key += await getRustKey();
const target = `v0-target-${targetKey}${rustKey}`;
return { return {
registry: { paths: [
name: "Registry", paths.index,
paths: [ paths.cache,
paths.index, // TODO: paths.git,
paths.cache, paths.target,
], ],
key: `${registry}-`, key: `${key}-${lockHash}`,
restoreKeys: [registry], restoreKeys: [key],
},
target: {
name: "Target",
paths: [paths.target],
key: `${target}-${lockHash}`,
restoreKeys: [target],
},
}; };
} }
async function getRustKey() { async function getRustKey() {
@ -54680,15 +54674,6 @@ async function getCmdOutput(cmd, args = [], options = {}) {
} }, options)); } }, options));
return stdout; return stdout;
} }
async function getRegistryName() {
const globber = await glob.create(`${paths.index}/**/.last-updated`, { followSymbolicLinks: false });
const files = await globber.glob();
if (files.length > 1) {
core.warning(`got multiple registries: "${files.join('", "')}"`);
}
const first = files.shift();
return external_path_default().basename(external_path_default().dirname(first));
}
async function getLockfileHash() { async function getLockfileHash() {
var e_1, _a; var e_1, _a;
const globber = await glob.create("**/Cargo.toml\n**/Cargo.lock", { followSymbolicLinks: false }); const globber = await glob.create("**/Cargo.toml\n**/Cargo.lock", { followSymbolicLinks: false });
@ -54723,30 +54708,26 @@ async function run() {
} }
try { try {
core.exportVariable("CARGO_INCREMENTAL", 0); core.exportVariable("CARGO_INCREMENTAL", 0);
const caches = await getCaches(); const start = Date.now();
for (const [type, { name, paths, key, restoreKeys }] of Object.entries(caches)) { const { paths, key, restoreKeys } = await getCacheConfig();
const start = Date.now(); core.info(`Restoring paths:\n ${paths.join("\n ")}.`);
core.startGroup(`Restoring ${name}`); core.info(`Using keys:\n ${[key, ...restoreKeys].join("\n ")}`);
core.info(`Restoring paths:\n ${paths.join("\n ")}.`); try {
core.info(`Using keys:\n ${[key, ...restoreKeys].join("\n ")}`); const restoreKey = await cache.restoreCache(paths, key, restoreKeys);
try { if (restoreKey) {
const restoreKey = await cache.restoreCache(paths, key, restoreKeys); core.info(`Restored from cache key "${restoreKey}".`);
if (restoreKey) { core.saveState(stateKey, restoreKey);
core.info(`Restored from cache key "${restoreKey}".`);
core.saveState(`CACHEKEY-${type}`, restoreKey);
}
else {
core.info("No cache found.");
}
} }
catch (e) { else {
core.info(`[warning] ${e.message}`); core.info("No cache found.");
} }
const duration = Math.round((Date.now() - start) / 1000); }
if (duration) { catch (e) {
core.info(`Took ${duration}s.`); core.info(`[warning] ${e.message}`);
} }
core.endGroup(); const duration = Math.round((Date.now() - start) / 1000);
if (duration) {
core.info(`Took ${duration}s.`);
} }
} }
catch (e) { catch (e) {

132
dist/save/index.js vendored
View file

@ -54585,13 +54585,13 @@ var glob = __webpack_require__(8090);
var external_crypto_ = __webpack_require__(6417); var external_crypto_ = __webpack_require__(6417);
var external_crypto_default = /*#__PURE__*/__webpack_require__.n(external_crypto_); var external_crypto_default = /*#__PURE__*/__webpack_require__.n(external_crypto_);
// EXTERNAL MODULE: ./node_modules/@actions/io/lib/io.js
var io = __webpack_require__(7436);
// EXTERNAL MODULE: external "fs" // EXTERNAL MODULE: external "fs"
var external_fs_ = __webpack_require__(5747); var external_fs_ = __webpack_require__(5747);
var external_fs_default = /*#__PURE__*/__webpack_require__.n(external_fs_); var external_fs_default = /*#__PURE__*/__webpack_require__.n(external_fs_);
// EXTERNAL MODULE: ./node_modules/@actions/io/lib/io.js
var io = __webpack_require__(7436);
// EXTERNAL MODULE: external "os" // EXTERNAL MODULE: external "os"
var external_os_ = __webpack_require__(2087); var external_os_ = __webpack_require__(2087);
var external_os_default = /*#__PURE__*/__webpack_require__.n(external_os_); var external_os_default = /*#__PURE__*/__webpack_require__.n(external_os_);
@ -54615,6 +54615,8 @@ var __asyncValues = (undefined && undefined.__asyncValues) || function (o) {
const stateKey = "RUST_CACHE_KEY";
const stateHash = "RUST_CACHE_HASH";
const home = external_os_default().homedir(); const home = external_os_default().homedir();
const paths = { const paths = {
index: external_path_default().join(home, ".cargo/registry/index"), index: external_path_default().join(home, ".cargo/registry/index"),
@ -54626,39 +54628,31 @@ const RefKey = "GITHUB_REF";
function isValidEvent() { function isValidEvent() {
return RefKey in process.env && Boolean(process.env[RefKey]); return RefKey in process.env && Boolean(process.env[RefKey]);
} }
async function getCaches() { async function getCacheConfig() {
const rustKey = await getRustKey(); let lockHash = core.getState(stateHash);
let lockHash = core.getState("lockHash");
if (!lockHash) { if (!lockHash) {
lockHash = await getLockfileHash(); lockHash = await getLockfileHash();
core.saveState("lockHash", lockHash); core.saveState(stateHash, lockHash);
} }
let targetKey = core.getInput("key"); let key = `v0-rust-`;
if (targetKey) { let inputKey = core.getInput("key");
targetKey = `${targetKey}-`; if (inputKey) {
key += `${inputKey}-`;
} }
const job = process.env.GITHUB_JOB; const job = process.env.GITHUB_JOB;
if (job) { if (job) {
targetKey = `${job}-${targetKey}`; key += `${job}-`;
} }
const registry = `v0-registry`; key += await getRustKey();
const target = `v0-target-${targetKey}${rustKey}`;
return { return {
registry: { paths: [
name: "Registry", paths.index,
paths: [ paths.cache,
paths.index, // TODO: paths.git,
paths.cache, paths.target,
], ],
key: `${registry}-`, key: `${key}-${lockHash}`,
restoreKeys: [registry], restoreKeys: [key],
},
target: {
name: "Target",
paths: [paths.target],
key: `${target}-${lockHash}`,
restoreKeys: [target],
},
}; };
} }
async function getRustKey() { async function getRustKey() {
@ -54683,15 +54677,6 @@ async function getCmdOutput(cmd, args = [], options = {}) {
} }, options)); } }, options));
return stdout; return stdout;
} }
async function getRegistryName() {
const globber = await glob.create(`${paths.index}/**/.last-updated`, { followSymbolicLinks: false });
const files = await globber.glob();
if (files.length > 1) {
core.warning(`got multiple registries: "${files.join('", "')}"`);
}
const first = files.shift();
return external_path_default().basename(external_path_default().dirname(first));
}
async function getLockfileHash() { async function getLockfileHash() {
var e_1, _a; var e_1, _a;
const globber = await glob.create("**/Cargo.toml\n**/Cargo.lock", { followSymbolicLinks: false }); const globber = await glob.create("**/Cargo.toml\n**/Cargo.lock", { followSymbolicLinks: false });
@ -54731,50 +54716,35 @@ var save_asyncValues = (undefined && undefined.__asyncValues) || function (o) {
async function run() { async function run() {
if (!isValidEvent()) { if (!isValidEvent()) {
return; return;
} }
try { try {
const caches = await getCaches(); const start = Date.now();
let upToDate = true; const { paths: savePaths, key } = await getCacheConfig();
for (const [type, { key }] of Object.entries(caches)) { if (core.getState(stateKey) === key) {
if (core.getState(`CACHEKEY-${type}`) !== key) { core.info(`Cache up-to-date.`);
upToDate = false;
break;
}
}
if (upToDate) {
core.info(`All caches up-to-date`);
return; return;
} }
const registryName = await getRegistryName();
const packages = await getPackages();
// TODO: remove this once https://github.com/actions/toolkit/pull/553 lands // TODO: remove this once https://github.com/actions/toolkit/pull/553 lands
await macOsWorkaround(); await macOsWorkaround();
await io.rmRF(external_path_default().join(paths.index, registryName, ".cache")); const registryName = await getRegistryName();
await pruneRegistryCache(registryName, packages); const packages = await getPackages();
await pruneTarget(packages); await cleanRegistry(registryName, packages);
for (const [type, { name, path: paths, key }] of Object.entries(caches)) { await cleanTarget(packages);
if (core.getState(`CACHEKEY-${type}`) === key) { core.info(`Saving paths:\n ${savePaths.join("\n ")}.`);
core.info(`${name} up-to-date.`); core.info(`Using key "${key}".`);
continue; try {
} await cache.saveCache(savePaths, key);
const start = Date.now(); }
core.startGroup(`Saving ${name}`); catch (e) {
core.info(`Saving paths:\n ${paths.join("\n ")}.`); core.info(`[warning] ${e.message}`);
core.info(`Using key "${key}".`); }
try { const duration = Math.round((Date.now() - start) / 1000);
await cache.saveCache(paths, key); if (duration) {
} core.info(`Took ${duration}s.`);
catch (e) {
core.info(`[warning] ${e.message}`);
}
const duration = Math.round((Date.now() - start) / 1000);
if (duration) {
core.info(`Took ${duration}s.`);
}
core.endGroup();
} }
} }
catch (e) { catch (e) {
@ -54782,6 +54752,15 @@ async function run() {
} }
} }
run(); run();
async function getRegistryName() {
const globber = await glob.create(`${paths.index}/**/.last-updated`, { followSymbolicLinks: false });
const files = await globber.glob();
if (files.length > 1) {
core.warning(`got multiple registries: "${files.join('", "')}"`);
}
const first = files.shift();
return external_path_default().basename(external_path_default().dirname(first));
}
async function getPackages() { async function getPackages() {
const cwd = process.cwd(); const cwd = process.cwd();
const meta = JSON.parse(await getCmdOutput("cargo", ["metadata", "--all-features", "--format-version", "1"])); const meta = JSON.parse(await getCmdOutput("cargo", ["metadata", "--all-features", "--format-version", "1"]));
@ -54792,8 +54771,9 @@ async function getPackages() {
return { name: p.name, version: p.version, targets }; return { name: p.name, version: p.version, targets };
}); });
} }
async function pruneRegistryCache(registryName, packages) { async function cleanRegistry(registryName, packages) {
var e_1, _a; var e_1, _a;
await io.rmRF(external_path_default().join(paths.index, registryName, ".cache"));
const pkgSet = new Set(packages.map((p) => `${p.name}-${p.version}.crate`)); const pkgSet = new Set(packages.map((p) => `${p.name}-${p.version}.crate`));
const dir = await external_fs_default().promises.opendir(external_path_default().join(paths.cache, registryName)); const dir = await external_fs_default().promises.opendir(external_path_default().join(paths.cache, registryName));
try { try {
@ -54814,7 +54794,7 @@ async function pruneRegistryCache(registryName, packages) {
finally { if (e_1) throw e_1.error; } finally { if (e_1) throw e_1.error; }
} }
} }
async function pruneTarget(packages) { async function cleanTarget(packages) {
var e_2, _a; var e_2, _a;
await external_fs_default().promises.unlink("./target/.rustc_info.json"); await external_fs_default().promises.unlink("./target/.rustc_info.json");
await io.rmRF("./target/debug/examples"); await io.rmRF("./target/debug/examples");
@ -54851,7 +54831,7 @@ async function pruneTarget(packages) {
})); }));
await rmExcept("./target/debug/deps", keepDeps); await rmExcept("./target/debug/deps", keepDeps);
} }
const twoWeeks = 14 * 24 * 3600 * 1000; const oneWeek = 7 * 24 * 3600 * 1000;
async function rmExcept(dirName, keepPrefix) { async function rmExcept(dirName, keepPrefix) {
var e_3, _a; var e_3, _a;
const dir = await external_fs_default().promises.opendir(dirName); const dir = await external_fs_default().promises.opendir(dirName);
@ -54865,7 +54845,7 @@ async function rmExcept(dirName, keepPrefix) {
} }
const fileName = external_path_default().join(dir.path, dirent.name); const fileName = external_path_default().join(dir.path, dirent.name);
const { mtime } = await external_fs_default().promises.stat(fileName); const { mtime } = await external_fs_default().promises.stat(fileName);
if (!keepPrefix.has(name) || Date.now() - mtime.getTime() > twoWeeks) { if (!keepPrefix.has(name) || Date.now() - mtime.getTime() > oneWeek) {
core.debug(`deleting "${fileName}"`); core.debug(`deleting "${fileName}"`);
if (dirent.isFile()) { if (dirent.isFile()) {
await external_fs_default().promises.unlink(fileName); await external_fs_default().promises.unlink(fileName);

View file

@ -2,7 +2,7 @@
"private": true, "private": true,
"name": "rust-cache", "name": "rust-cache",
"version": "1.0.2", "version": "1.0.2",
"description": "A GitHub Action that implements smart caching for rust/cargo projects", "description": "A GitHub Action that implements smart caching for rust/cargo projects with sensible defaults.",
"keywords": [ "keywords": [
"actions", "actions",
"rust", "rust",

View file

@ -6,6 +6,9 @@ import fs from "fs";
import os from "os"; import os from "os";
import path from "path"; import path from "path";
export const stateKey = "RUST_CACHE_KEY";
const stateHash = "RUST_CACHE_HASH";
const home = os.homedir(); const home = os.homedir();
export const paths = { export const paths = {
index: path.join(home, ".cargo/registry/index"), index: path.join(home, ".cargo/registry/index"),
@ -15,15 +18,9 @@ export const paths = {
}; };
interface CacheConfig { interface CacheConfig {
name: string;
paths: Array<string>; paths: Array<string>;
key: string; key: string;
restoreKeys?: Array<string>; restoreKeys: Array<string>;
}
interface Caches {
registry: CacheConfig;
target: CacheConfig;
} }
const RefKey = "GITHUB_REF"; const RefKey = "GITHUB_REF";
@ -32,41 +29,36 @@ export function isValidEvent(): boolean {
return RefKey in process.env && Boolean(process.env[RefKey]); return RefKey in process.env && Boolean(process.env[RefKey]);
} }
export async function getCaches(): Promise<Caches> { export async function getCacheConfig(): Promise<CacheConfig> {
const rustKey = await getRustKey(); let lockHash = core.getState(stateHash);
let lockHash = core.getState("lockHash");
if (!lockHash) { if (!lockHash) {
lockHash = await getLockfileHash(); lockHash = await getLockfileHash();
core.saveState("lockHash", lockHash); core.saveState(stateHash, lockHash);
}
let targetKey = core.getInput("key");
if (targetKey) {
targetKey = `${targetKey}-`;
}
const job = process.env.GITHUB_JOB;
if (job) {
targetKey = `${job}-${targetKey}`;
} }
const registry = `v0-registry`; let key = `v0-rust-`;
const target = `v0-target-${targetKey}${rustKey}`;
let inputKey = core.getInput("key");
if (inputKey) {
key += `${inputKey}-`;
}
const job = process.env.GITHUB_JOB;
if (job) {
key += `${job}-`;
}
key += await getRustKey();
return { return {
registry: { paths: [
name: "Registry", paths.index,
paths: [ paths.cache,
paths.index, // TODO: paths.git,
paths.cache, paths.target,
// TODO: paths.git, ],
], key: `${key}-${lockHash}`,
key: `${registry}-${lockHash}`, restoreKeys: [key],
restoreKeys: [registry],
},
target: {
name: "Target",
paths: [paths.target],
key: `${target}-${lockHash}`,
restoreKeys: [target],
},
}; };
} }
@ -109,17 +101,6 @@ export async function getCmdOutput(
return stdout; return stdout;
} }
export async function getRegistryName(): Promise<string> {
const globber = await glob.create(`${paths.index}/**/.last-updated`, { followSymbolicLinks: false });
const files = await globber.glob();
if (files.length > 1) {
core.warning(`got multiple registries: "${files.join('", "')}"`);
}
const first = files.shift()!;
return path.basename(path.dirname(first));
}
async function getLockfileHash(): Promise<string> { async function getLockfileHash(): Promise<string> {
const globber = await glob.create("**/Cargo.toml\n**/Cargo.lock", { followSymbolicLinks: false }); const globber = await glob.create("**/Cargo.toml\n**/Cargo.lock", { followSymbolicLinks: false });
const files = await globber.glob(); const files = await globber.glob();

View file

@ -1,6 +1,6 @@
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 { getCaches, isValidEvent } from "./common"; import { getCacheConfig, isValidEvent, stateKey } from "./common";
async function run() { async function run() {
if (!isValidEvent()) { if (!isValidEvent()) {
@ -10,28 +10,26 @@ async function run() {
try { try {
core.exportVariable("CARGO_INCREMENTAL", 0); core.exportVariable("CARGO_INCREMENTAL", 0);
const caches = await getCaches(); const start = Date.now();
for (const [type, { name, paths, key, restoreKeys }] of Object.entries(caches)) { const { paths, key, restoreKeys } = await getCacheConfig();
const start = Date.now();
core.startGroup(`Restoring ${name}`); core.info(`Restoring paths:\n ${paths.join("\n ")}.`);
core.info(`Restoring paths:\n ${paths.join("\n ")}.`); core.info(`Using keys:\n ${[key, ...restoreKeys].join("\n ")}`);
core.info(`Using keys:\n ${[key, ...restoreKeys].join("\n ")}`); try {
try { const restoreKey = await cache.restoreCache(paths, key, restoreKeys);
const restoreKey = await cache.restoreCache(paths, key, restoreKeys); if (restoreKey) {
if (restoreKey) { core.info(`Restored from cache key "${restoreKey}".`);
core.info(`Restored from cache key "${restoreKey}".`); core.saveState(stateKey, restoreKey);
core.saveState(`CACHEKEY-${type}`, restoreKey); } else {
} else { core.info("No cache found.");
core.info("No cache found.");
}
} catch (e) {
core.info(`[warning] ${e.message}`);
} }
const duration = Math.round((Date.now() - start) / 1000); } catch (e) {
if (duration) { core.info(`[warning] ${e.message}`);
core.info(`Took ${duration}s.`); }
}
core.endGroup(); const duration = Math.round((Date.now() - start) / 1000);
if (duration) {
core.info(`Took ${duration}s.`);
} }
} catch (e) { } catch (e) {
core.info(`[warning] ${e.message}`); core.info(`[warning] ${e.message}`);

View file

@ -1,10 +1,11 @@
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 * as exec from "@actions/exec";
import * as glob from "@actions/glob";
import * as io from "@actions/io"; import * as io from "@actions/io";
import fs from "fs"; import fs from "fs";
import path from "path"; import path from "path";
import { getCaches, getCmdOutput, getRegistryName, isValidEvent, paths } from "./common"; import { getCacheConfig, getCmdOutput, isValidEvent, paths, stateKey } from "./common";
async function run() { async function run() {
if (!isValidEvent()) { if (!isValidEvent()) {
@ -12,49 +13,35 @@ async function run() {
} }
try { try {
const caches = await getCaches(); const start = Date.now();
let upToDate = true; const { paths: savePaths, key } = await getCacheConfig();
for (const [type, { key }] of Object.entries(caches)) {
if (core.getState(`CACHEKEY-${type}`) !== key) { if (core.getState(stateKey) === key) {
upToDate = false; core.info(`Cache up-to-date.`);
break;
}
}
if (upToDate) {
core.info(`All caches up-to-date`);
return; return;
} }
const registryName = await getRegistryName();
const packages = await getPackages();
// TODO: remove this once https://github.com/actions/toolkit/pull/553 lands // TODO: remove this once https://github.com/actions/toolkit/pull/553 lands
await macOsWorkaround(); await macOsWorkaround();
await io.rmRF(path.join(paths.index, registryName, ".cache")); const registryName = await getRegistryName();
await pruneRegistryCache(registryName, packages); const packages = await getPackages();
await pruneTarget(packages); await cleanRegistry(registryName, packages);
for (const [type, { name, path: paths, key }] of Object.entries(caches)) { await cleanTarget(packages);
if (core.getState(`CACHEKEY-${type}`) === key) {
core.info(`${name} up-to-date.`); core.info(`Saving paths:\n ${savePaths.join("\n ")}.`);
continue; core.info(`Using key "${key}".`);
} try {
const start = Date.now(); await cache.saveCache(savePaths, key);
core.startGroup(`Saving ${name}`); } catch (e) {
core.info(`Saving paths:\n ${paths.join("\n ")}.`); core.info(`[warning] ${e.message}`);
core.info(`Using key "${key}".`); }
try {
await cache.saveCache(paths, key); const duration = Math.round((Date.now() - start) / 1000);
} catch (e) { if (duration) {
core.info(`[warning] ${e.message}`); core.info(`Took ${duration}s.`);
}
const duration = Math.round((Date.now() - start) / 1000);
if (duration) {
core.info(`Took ${duration}s.`);
}
core.endGroup();
} }
} catch (e) { } catch (e) {
core.info(`[warning] ${e.message}`); core.info(`[warning] ${e.message}`);
@ -80,6 +67,17 @@ interface Meta {
}>; }>;
} }
async function getRegistryName(): Promise<string> {
const globber = await glob.create(`${paths.index}/**/.last-updated`, { followSymbolicLinks: false });
const files = await globber.glob();
if (files.length > 1) {
core.warning(`got multiple registries: "${files.join('", "')}"`);
}
const first = files.shift()!;
return path.basename(path.dirname(first));
}
async function getPackages(): Promise<Packages> { async function getPackages(): Promise<Packages> {
const cwd = process.cwd(); const cwd = process.cwd();
const meta: Meta = JSON.parse(await getCmdOutput("cargo", ["metadata", "--all-features", "--format-version", "1"])); const meta: Meta = JSON.parse(await getCmdOutput("cargo", ["metadata", "--all-features", "--format-version", "1"]));
@ -92,7 +90,9 @@ async function getPackages(): Promise<Packages> {
}); });
} }
async function pruneRegistryCache(registryName: string, packages: Packages) { async function cleanRegistry(registryName: string, packages: Packages) {
await io.rmRF(path.join(paths.index, registryName, ".cache"));
const pkgSet = new Set(packages.map((p) => `${p.name}-${p.version}.crate`)); const pkgSet = new Set(packages.map((p) => `${p.name}-${p.version}.crate`));
const dir = await fs.promises.opendir(path.join(paths.cache, registryName)); const dir = await fs.promises.opendir(path.join(paths.cache, registryName));
@ -105,12 +105,12 @@ async function pruneRegistryCache(registryName: string, packages: Packages) {
} }
} }
async function pruneTarget(packages: Packages) { async function cleanTarget(packages: Packages) {
await fs.promises.unlink("./target/.rustc_info.json"); await fs.promises.unlink("./target/.rustc_info.json");
await io.rmRF("./target/debug/examples"); await io.rmRF("./target/debug/examples");
await io.rmRF("./target/debug/incremental"); await io.rmRF("./target/debug/incremental");
let dir: fs.Dir;
let dir: fs.Dir;
// remove all *files* from debug // remove all *files* from debug
dir = await fs.promises.opendir("./target/debug"); dir = await fs.promises.opendir("./target/debug");
for await (const dirent of dir) { for await (const dirent of dir) {
@ -137,7 +137,7 @@ async function pruneTarget(packages: Packages) {
await rmExcept("./target/debug/deps", keepDeps); await rmExcept("./target/debug/deps", keepDeps);
} }
const twoWeeks = 14 * 24 * 3600 * 1000; const oneWeek = 7 * 24 * 3600 * 1000;
async function rmExcept(dirName: string, keepPrefix: Set<string>) { async function rmExcept(dirName: string, keepPrefix: Set<string>) {
const dir = await fs.promises.opendir(dirName); const dir = await fs.promises.opendir(dirName);
@ -149,7 +149,7 @@ async function rmExcept(dirName: string, keepPrefix: Set<string>) {
} }
const fileName = path.join(dir.path, dirent.name); const fileName = path.join(dir.path, dirent.name);
const { mtime } = await fs.promises.stat(fileName); const { mtime } = await fs.promises.stat(fileName);
if (!keepPrefix.has(name) || Date.now() - mtime.getTime() > twoWeeks) { if (!keepPrefix.has(name) || Date.now() - mtime.getTime() > oneWeek) {
core.debug(`deleting "${fileName}"`); core.debug(`deleting "${fileName}"`);
if (dirent.isFile()) { if (dirent.isFile()) {
await fs.promises.unlink(fileName); await fs.promises.unlink(fileName);