From 90b667205be51435793586317467fc72f138f317 Mon Sep 17 00:00:00 2001 From: Leander Schulten Date: Fri, 2 Feb 2024 22:15:55 +0100 Subject: [PATCH 1/3] github actions cache: Use more unique key --- dist/index.js | 11 ++++++++++- src/classes/state/state-cache-storage.ts | 5 ++++- 2 files changed, 14 insertions(+), 2 deletions(-) diff --git a/dist/index.js b/dist/index.js index f2786a0f6..3a1c50025 100644 --- a/dist/index.js +++ b/dist/index.js @@ -1591,11 +1591,12 @@ exports.StateCacheStorage = void 0; const fs_1 = __importDefault(__nccwpck_require__(7147)); const path_1 = __importDefault(__nccwpck_require__(1017)); const os_1 = __importDefault(__nccwpck_require__(2037)); +const process_1 = __importDefault(__nccwpck_require__(7282)); const core = __importStar(__nccwpck_require__(2186)); const github_1 = __nccwpck_require__(5438); const plugin_retry_1 = __nccwpck_require__(6298); const cache = __importStar(__nccwpck_require__(7799)); -const CACHE_KEY = '_state'; +const CACHE_KEY = encodeURIComponent('_state' + process_1.default.env.GITHUB_WORKFLOW + '-' + process_1.default.env.GITHUB_JOB); const STATE_FILE = 'state.txt'; const STALE_DIR = '56acbeaa-1fef-4c79-8f84-7565e560fb03'; const mkTempDir = () => { @@ -89199,6 +89200,14 @@ module.exports = require("perf_hooks"); /***/ }), +/***/ 7282: +/***/ ((module) => { + +"use strict"; +module.exports = require("process"); + +/***/ }), + /***/ 5477: /***/ ((module) => { diff --git a/src/classes/state/state-cache-storage.ts b/src/classes/state/state-cache-storage.ts index b30b503b6..b1c3f40a1 100644 --- a/src/classes/state/state-cache-storage.ts +++ b/src/classes/state/state-cache-storage.ts @@ -2,12 +2,15 @@ import {IStateStorage} from '../../interfaces/state/state-storage'; import fs from 'fs'; import path from 'path'; import os from 'os'; +import process from 'process'; import * as core from '@actions/core'; import {context, getOctokit} from '@actions/github'; import {retry as octokitRetry} from '@octokit/plugin-retry'; import * as cache from '@actions/cache'; -const CACHE_KEY = '_state'; +const CACHE_KEY = encodeURIComponent( + '_state' + process.env.GITHUB_WORKFLOW + '-' + process.env.GITHUB_JOB +); const STATE_FILE = 'state.txt'; const STALE_DIR = '56acbeaa-1fef-4c79-8f84-7565e560fb03'; From 7b84952f53c8fabb90561153a686cb595fb00cb8 Mon Sep 17 00:00:00 2001 From: Leander Schulten Date: Thu, 23 May 2024 14:22:29 +0200 Subject: [PATCH 2/3] Use the hashed inputs as cache key to allow multiple steps at the same time --- dist/index.js | 57 +++++++++++++++++------- src/classes/state/state-cache-storage.ts | 20 +++++---- src/services/state.service.ts | 10 ++++- 3 files changed, 60 insertions(+), 27 deletions(-) diff --git a/dist/index.js b/dist/index.js index 3a1c50025..0f2df108f 100644 --- a/dist/index.js +++ b/dist/index.js @@ -1591,12 +1591,10 @@ exports.StateCacheStorage = void 0; const fs_1 = __importDefault(__nccwpck_require__(7147)); const path_1 = __importDefault(__nccwpck_require__(1017)); const os_1 = __importDefault(__nccwpck_require__(2037)); -const process_1 = __importDefault(__nccwpck_require__(7282)); const core = __importStar(__nccwpck_require__(2186)); const github_1 = __nccwpck_require__(5438); const plugin_retry_1 = __nccwpck_require__(6298); const cache = __importStar(__nccwpck_require__(7799)); -const CACHE_KEY = encodeURIComponent('_state' + process_1.default.env.GITHUB_WORKFLOW + '-' + process_1.default.env.GITHUB_JOB); const STATE_FILE = 'state.txt'; const STALE_DIR = '56acbeaa-1fef-4c79-8f84-7565e560fb03'; const mkTempDir = () => { @@ -1645,22 +1643,25 @@ const resetCacheWithOctokit = (cacheKey) => __awaiter(void 0, void 0, void 0, fu } }); class StateCacheStorage { + constructor(cacheKey) { + this.cacheKey = cacheKey; + } save(serializedState) { return __awaiter(this, void 0, void 0, function* () { const tmpDir = mkTempDir(); const filePath = path_1.default.join(tmpDir, STATE_FILE); fs_1.default.writeFileSync(filePath, serializedState); try { - const cacheExists = yield checkIfCacheExists(CACHE_KEY); + const cacheExists = yield checkIfCacheExists(this.cacheKey); if (cacheExists) { - yield resetCacheWithOctokit(CACHE_KEY); + yield resetCacheWithOctokit(this.cacheKey); } const fileSize = fs_1.default.statSync(filePath).size; if (fileSize === 0) { core.info(`the state will be removed`); return; } - yield cache.saveCache([path_1.default.dirname(filePath)], CACHE_KEY); + yield cache.saveCache([path_1.default.dirname(filePath)], this.cacheKey); } catch (error) { core.warning(`Saving the state was not successful due to "${error.message || 'unknown reason'}"`); @@ -1676,12 +1677,12 @@ class StateCacheStorage { const filePath = path_1.default.join(tmpDir, STATE_FILE); unlinkSafely(filePath); try { - const cacheExists = yield checkIfCacheExists(CACHE_KEY); + const cacheExists = yield checkIfCacheExists(this.cacheKey); if (!cacheExists) { core.info('The saved state was not found, the process starts from the first issue.'); return ''; } - yield cache.restoreCache([path_1.default.dirname(filePath)], CACHE_KEY); + yield cache.restoreCache([path_1.default.dirname(filePath)], this.cacheKey); if (!fs_1.default.existsSync(filePath)) { core.warning('Unknown error when unpacking the cache, the process starts from the first issue.'); return ''; @@ -2686,16 +2687,46 @@ exports.LoggerService = LoggerService; /***/ }), /***/ 6330: -/***/ ((__unused_webpack_module, exports, __nccwpck_require__) => { +/***/ (function(__unused_webpack_module, exports, __nccwpck_require__) { "use strict"; +var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { + if (k2 === undefined) k2 = k; + var desc = Object.getOwnPropertyDescriptor(m, k); + if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) { + desc = { enumerable: true, get: function() { return m[k]; } }; + } + Object.defineProperty(o, k2, desc); +}) : (function(o, m, k, k2) { + if (k2 === undefined) k2 = k; + o[k2] = m[k]; +})); +var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) { + Object.defineProperty(o, "default", { enumerable: true, value: v }); +}) : function(o, v) { + o["default"] = v; +}); +var __importStar = (this && this.__importStar) || function (mod) { + if (mod && mod.__esModule) return mod; + var result = {}; + if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k); + __setModuleDefault(result, mod); + return result; +}; Object.defineProperty(exports, "__esModule", ({ value: true })); exports.getStateInstance = void 0; +const crypto = __importStar(__nccwpck_require__(6113)); const state_1 = __nccwpck_require__(5186); const state_cache_storage_1 = __nccwpck_require__(3709); +function sha256(message) { + const hash = crypto.createHash('sha256'); + hash.update(message); + return hash.digest('hex'); +} const getStateInstance = (options) => { - const storage = new state_cache_storage_1.StateCacheStorage(); + const cacheKey = sha256(JSON.stringify(options)); + const storage = new state_cache_storage_1.StateCacheStorage(cacheKey); return new state_1.State(storage, options); }; exports.getStateInstance = getStateInstance; @@ -89200,14 +89231,6 @@ module.exports = require("perf_hooks"); /***/ }), -/***/ 7282: -/***/ ((module) => { - -"use strict"; -module.exports = require("process"); - -/***/ }), - /***/ 5477: /***/ ((module) => { diff --git a/src/classes/state/state-cache-storage.ts b/src/classes/state/state-cache-storage.ts index b1c3f40a1..176459bab 100644 --- a/src/classes/state/state-cache-storage.ts +++ b/src/classes/state/state-cache-storage.ts @@ -2,15 +2,11 @@ import {IStateStorage} from '../../interfaces/state/state-storage'; import fs from 'fs'; import path from 'path'; import os from 'os'; -import process from 'process'; import * as core from '@actions/core'; import {context, getOctokit} from '@actions/github'; import {retry as octokitRetry} from '@octokit/plugin-retry'; import * as cache from '@actions/cache'; -const CACHE_KEY = encodeURIComponent( - '_state' + process.env.GITHUB_WORKFLOW + '-' + process.env.GITHUB_JOB -); const STATE_FILE = 'state.txt'; const STALE_DIR = '56acbeaa-1fef-4c79-8f84-7565e560fb03'; @@ -68,15 +64,21 @@ const resetCacheWithOctokit = async (cacheKey: string): Promise => { } }; export class StateCacheStorage implements IStateStorage { + private cacheKey: string; + + constructor(cacheKey: string) { + this.cacheKey = cacheKey; + } + async save(serializedState: string): Promise { const tmpDir = mkTempDir(); const filePath = path.join(tmpDir, STATE_FILE); fs.writeFileSync(filePath, serializedState); try { - const cacheExists = await checkIfCacheExists(CACHE_KEY); + const cacheExists = await checkIfCacheExists(this.cacheKey); if (cacheExists) { - await resetCacheWithOctokit(CACHE_KEY); + await resetCacheWithOctokit(this.cacheKey); } const fileSize = fs.statSync(filePath).size; @@ -85,7 +87,7 @@ export class StateCacheStorage implements IStateStorage { return; } - await cache.saveCache([path.dirname(filePath)], CACHE_KEY); + await cache.saveCache([path.dirname(filePath)], this.cacheKey); } catch (error) { core.warning( `Saving the state was not successful due to "${ @@ -102,7 +104,7 @@ export class StateCacheStorage implements IStateStorage { const filePath = path.join(tmpDir, STATE_FILE); unlinkSafely(filePath); try { - const cacheExists = await checkIfCacheExists(CACHE_KEY); + const cacheExists = await checkIfCacheExists(this.cacheKey); if (!cacheExists) { core.info( 'The saved state was not found, the process starts from the first issue.' @@ -110,7 +112,7 @@ export class StateCacheStorage implements IStateStorage { return ''; } - await cache.restoreCache([path.dirname(filePath)], CACHE_KEY); + await cache.restoreCache([path.dirname(filePath)], this.cacheKey); if (!fs.existsSync(filePath)) { core.warning( diff --git a/src/services/state.service.ts b/src/services/state.service.ts index 77e403ecc..31ef93709 100644 --- a/src/services/state.service.ts +++ b/src/services/state.service.ts @@ -1,9 +1,17 @@ +import * as crypto from 'crypto'; import {IState} from '../interfaces/state/state'; import {State} from '../classes/state/state'; import {IIssuesProcessorOptions} from '../interfaces/issues-processor-options'; import {StateCacheStorage} from '../classes/state/state-cache-storage'; +function sha256(message: string): string { + const hash = crypto.createHash('sha256'); + hash.update(message); + return hash.digest('hex'); +} + export const getStateInstance = (options: IIssuesProcessorOptions): IState => { - const storage = new StateCacheStorage(); + const cacheKey = sha256(JSON.stringify(options)); + const storage = new StateCacheStorage(cacheKey); return new State(storage, options); }; From c368620c4352fb9517b2860592b3413fb9101f04 Mon Sep 17 00:00:00 2001 From: autoantwort Date: Tue, 8 Oct 2024 01:18:09 +0200 Subject: [PATCH 3/3] Don't use the token to generate a unique id --- dist/index.js | 3 ++- src/services/state.service.ts | 5 ++++- 2 files changed, 6 insertions(+), 2 deletions(-) diff --git a/dist/index.js b/dist/index.js index 0f2df108f..912b7fe9e 100644 --- a/dist/index.js +++ b/dist/index.js @@ -2725,7 +2725,8 @@ function sha256(message) { return hash.digest('hex'); } const getStateInstance = (options) => { - const cacheKey = sha256(JSON.stringify(options)); + const json = JSON.stringify(options, (key, value) => key === 'repoToken' ? undefined : value); + const cacheKey = sha256(json); const storage = new state_cache_storage_1.StateCacheStorage(cacheKey); return new state_1.State(storage, options); }; diff --git a/src/services/state.service.ts b/src/services/state.service.ts index 31ef93709..25719dadc 100644 --- a/src/services/state.service.ts +++ b/src/services/state.service.ts @@ -11,7 +11,10 @@ function sha256(message: string): string { } export const getStateInstance = (options: IIssuesProcessorOptions): IState => { - const cacheKey = sha256(JSON.stringify(options)); + const json = JSON.stringify(options, (key, value) => + key === 'repoToken' ? undefined : value + ); + const cacheKey = sha256(json); const storage = new StateCacheStorage(cacheKey); return new State(storage, options); };