1
0
Fork 0

Add more tests for restoreCacheV2

pull/1876/head
Bassem Dghaidi 2024-11-25 03:53:03 -08:00 committed by GitHub
parent 27dfd2c41c
commit 4de30f744e
3 changed files with 216 additions and 179 deletions

View File

@ -3,11 +3,12 @@ import * as path from 'path'
import * as tar from '../src/internal/tar' import * as tar from '../src/internal/tar'
import * as config from '../src/internal/config' import * as config from '../src/internal/config'
import * as cacheUtils from '../src/internal/cacheUtils' import * as cacheUtils from '../src/internal/cacheUtils'
import * as cacheHttpClient from '../src/internal/cacheHttpClient' import * as downloadCacheModule from '../src/internal/blob/download-cache'
import { restoreCache } from '../src/cache' import {restoreCache} from '../src/cache'
import { CacheFilename, CompressionMethod } from '../src/internal/constants' import {CacheFilename, CompressionMethod} from '../src/internal/constants'
import { ArtifactCacheEntry } from '../src/internal/contracts' import {CacheServiceClientJSON} from '../src/generated/results/api/v1/cache.twirp'
import { CacheServiceClientJSON } from '../src/generated/results/api/v1/cache.twirp' import {BlobDownloadResponseParsed} from '@azure/storage-blob'
// import {executePromisesSequentially} from '@azure/ms-rest-js'
jest.mock('../src/internal/cacheHttpClient') jest.mock('../src/internal/cacheHttpClient')
jest.mock('../src/internal/cacheUtils') jest.mock('../src/internal/cacheUtils')
@ -15,222 +16,257 @@ jest.mock('../src/internal/config')
jest.mock('../src/internal/tar') jest.mock('../src/internal/tar')
beforeAll(() => { beforeAll(() => {
jest.spyOn(console, 'log').mockImplementation(() => { }) jest.spyOn(console, 'log').mockImplementation(() => {})
jest.spyOn(core, 'debug').mockImplementation(() => { }) jest.spyOn(core, 'debug').mockImplementation(() => {})
jest.spyOn(core, 'info').mockImplementation(() => { }) jest.spyOn(core, 'info').mockImplementation(() => {})
jest.spyOn(core, 'warning').mockImplementation(() => { }) jest.spyOn(core, 'warning').mockImplementation(() => {})
jest.spyOn(core, 'error').mockImplementation(() => { }) jest.spyOn(core, 'error').mockImplementation(() => {})
jest.spyOn(cacheUtils, 'getCacheFileName').mockImplementation(cm => { jest.spyOn(cacheUtils, 'getCacheFileName').mockImplementation(cm => {
const actualUtils = jest.requireActual('../src/internal/cacheUtils') const actualUtils = jest.requireActual('../src/internal/cacheUtils')
return actualUtils.getCacheFileName(cm) return actualUtils.getCacheFileName(cm)
}) })
// Ensure that we're using v2 for these tests // Ensure that we're using v2 for these tests
jest.spyOn(config, 'getCacheServiceVersion').mockReturnValue('v2') jest.spyOn(config, 'getCacheServiceVersion').mockReturnValue('v2')
}) })
test('restore with no path should fail', async () => { test('restore with no path should fail', async () => {
const paths: string[] = [] const paths: string[] = []
const key = 'node-test' const key = 'node-test'
await expect(restoreCache(paths, key)).rejects.toThrowError( await expect(restoreCache(paths, key)).rejects.toThrowError(
`Path Validation Error: At least one directory or file path is required` `Path Validation Error: At least one directory or file path is required`
) )
}) })
test('restore with too many keys should fail', async () => { test('restore with too many keys should fail', async () => {
const paths = ['node_modules'] const paths = ['node_modules']
const key = 'node-test' const key = 'node-test'
const restoreKeys = [...Array(20).keys()].map(x => x.toString()) const restoreKeys = [...Array(20).keys()].map(x => x.toString())
await expect(restoreCache(paths, key, restoreKeys)).rejects.toThrowError( await expect(restoreCache(paths, key, restoreKeys)).rejects.toThrowError(
`Key Validation Error: Keys are limited to a maximum of 10.` `Key Validation Error: Keys are limited to a maximum of 10.`
) )
}) })
test('restore with large key should fail', async () => { test('restore with large key should fail', async () => {
const paths = ['node_modules'] const paths = ['node_modules']
const key = 'foo'.repeat(512) // Over the 512 character limit const key = 'foo'.repeat(512) // Over the 512 character limit
await expect(restoreCache(paths, key)).rejects.toThrowError( await expect(restoreCache(paths, key)).rejects.toThrowError(
`Key Validation Error: ${key} cannot be larger than 512 characters.` `Key Validation Error: ${key} cannot be larger than 512 characters.`
) )
}) })
test('restore with invalid key should fail', async () => { test('restore with invalid key should fail', async () => {
const paths = ['node_modules'] const paths = ['node_modules']
const key = 'comma,comma' const key = 'comma,comma'
await expect(restoreCache(paths, key)).rejects.toThrowError( await expect(restoreCache(paths, key)).rejects.toThrowError(
`Key Validation Error: ${key} cannot contain commas.` `Key Validation Error: ${key} cannot contain commas.`
) )
}) })
test('restore with no cache found', async () => { test('restore with no cache found', async () => {
const paths = ['node_modules'] const paths = ['node_modules']
const key = 'node-test' const key = 'node-test'
jest jest
.spyOn(CacheServiceClientJSON.prototype, 'GetCacheEntryDownloadURL') .spyOn(CacheServiceClientJSON.prototype, 'GetCacheEntryDownloadURL')
.mockReturnValue(Promise.resolve({ ok: false, signedDownloadUrl: '' })) .mockReturnValue(Promise.resolve({ok: false, signedDownloadUrl: ''}))
const cacheKey = await restoreCache(paths, key) const cacheKey = await restoreCache(paths, key)
expect(cacheKey).toBe(undefined) expect(cacheKey).toBe(undefined)
}) })
test('restore with server error should fail', async () => { test('restore with server error should fail', async () => {
const paths = ['node_modules'] const paths = ['node_modules']
const key = 'node-test' const key = 'node-test'
const logWarningMock = jest.spyOn(core, 'warning') const logWarningMock = jest.spyOn(core, 'warning')
jest jest
.spyOn(CacheServiceClientJSON.prototype, 'GetCacheEntryDownloadURL') .spyOn(CacheServiceClientJSON.prototype, 'GetCacheEntryDownloadURL')
.mockImplementation(() => { .mockImplementation(() => {
throw new Error('HTTP Error Occurred') throw new Error('HTTP Error Occurred')
}) })
const cacheKey = await restoreCache(paths, key) const cacheKey = await restoreCache(paths, key)
expect(cacheKey).toBe(undefined) expect(cacheKey).toBe(undefined)
expect(logWarningMock).toHaveBeenCalledTimes(1) expect(logWarningMock).toHaveBeenCalledTimes(1)
expect(logWarningMock).toHaveBeenCalledWith( expect(logWarningMock).toHaveBeenCalledWith(
'Failed to restore: HTTP Error Occurred' 'Failed to restore: HTTP Error Occurred'
) )
}) })
// test('restore with restore keys and no cache found', async () => { test('restore with restore keys and no cache found', async () => {
// const paths = ['node_modules'] const paths = ['node_modules']
// const key = 'node-test' const key = 'node-test'
// const restoreKey = 'node-' const restoreKey = 'node-'
const logWarningMock = jest.spyOn(core, 'warning')
// jest jest
// .spyOn(CacheServiceClientJSON.prototype, 'GetCacheEntryDownloadURL') .spyOn(CacheServiceClientJSON.prototype, 'GetCacheEntryDownloadURL')
// .mockImplementation(() => { .mockReturnValue(Promise.resolve({ok: false, signedDownloadUrl: ''}))
// return Promise.resolve(null)
// })
// jest.spyOn(cacheHttpClient, 'getCacheEntry').mockImplementation(async () => {
// return Promise.resolve(null)
// })
// const cacheKey = await restoreCache(paths, key, [restoreKey]) const cacheKey = await restoreCache(paths, key, [restoreKey])
// expect(cacheKey).toBe(undefined) expect(cacheKey).toBe(undefined)
// }) expect(logWarningMock).toHaveBeenCalledWith(
`Cache not found for keys: ${[key, restoreKey].join(', ')}`
)
})
// test('restore with gzip compressed cache found', async () => { test('restore with gzip compressed cache found', async () => {
// const paths = ['node_modules'] const paths = ['node_modules']
// const key = 'node-test' const key = 'node-test'
const logInfoMock = jest.spyOn(core, 'info')
const compressionMethod = CompressionMethod.Gzip
const signedDownloadUrl = 'https://blob-storage.local?signed=true'
const cacheVersion =
'd90f107aaeb22920dba0c637a23c37b5bc497b4dfa3b07fe3f79bf88a273c11b'
// const cacheEntry: ArtifactCacheEntry = { const getCacheVersionMock = jest.spyOn(cacheUtils, 'getCacheVersion')
// cacheKey: key, getCacheVersionMock.mockReturnValue(cacheVersion)
// scope: 'refs/heads/main',
// archiveLocation: 'www.actionscache.test/download'
// }
// const getCacheMock = jest.spyOn(cacheHttpClient, 'getCacheEntry')
// getCacheMock.mockImplementation(async () => {
// return Promise.resolve(cacheEntry)
// })
// const tempPath = '/foo/bar' const compressionMethodMock = jest.spyOn(cacheUtils, 'getCompressionMethod')
compressionMethodMock.mockReturnValue(Promise.resolve(compressionMethod))
// const createTempDirectoryMock = jest.spyOn(cacheUtils, 'createTempDirectory') const getCacheDownloadURLMock = jest.spyOn(
// createTempDirectoryMock.mockImplementation(async () => { CacheServiceClientJSON.prototype,
// return Promise.resolve(tempPath) 'GetCacheEntryDownloadURL'
// }) )
getCacheDownloadURLMock.mockReturnValue(
Promise.resolve({ok: true, signedDownloadUrl})
)
// const archivePath = path.join(tempPath, CacheFilename.Gzip) const tempPath = '/foo/bar'
// const downloadCacheMock = jest.spyOn(cacheHttpClient, 'downloadCache')
// const fileSize = 142 const createTempDirectoryMock = jest.spyOn(cacheUtils, 'createTempDirectory')
// const getArchiveFileSizeInBytesMock = jest createTempDirectoryMock.mockImplementation(async () => {
// .spyOn(cacheUtils, 'getArchiveFileSizeInBytes') return Promise.resolve(tempPath)
// .mockReturnValue(fileSize) })
// const extractTarMock = jest.spyOn(tar, 'extractTar') const archivePath = path.join(tempPath, CacheFilename.Gzip)
// const unlinkFileMock = jest.spyOn(cacheUtils, 'unlinkFile') const downloadCacheFileMock = jest.spyOn(
downloadCacheModule,
'downloadCacheFile'
)
downloadCacheFileMock.mockReturnValue(
Promise.resolve({} as BlobDownloadResponseParsed)
)
// const compression = CompressionMethod.Gzip const fileSize = 142
// const getCompressionMock = jest const getArchiveFileSizeInBytesMock = jest
// .spyOn(cacheUtils, 'getCompressionMethod') .spyOn(cacheUtils, 'getArchiveFileSizeInBytes')
// .mockReturnValue(Promise.resolve(compression)) .mockReturnValue(fileSize)
// const cacheKey = await restoreCache(paths, key) const extractTarMock = jest.spyOn(tar, 'extractTar')
const unlinkFileMock = jest.spyOn(cacheUtils, 'unlinkFile')
// expect(cacheKey).toBe(key) const cacheKey = await restoreCache(paths, key)
// expect(getCacheMock).toHaveBeenCalledWith([key], paths, {
// compressionMethod: compression,
// enableCrossOsArchive: false
// })
// expect(createTempDirectoryMock).toHaveBeenCalledTimes(1)
// expect(downloadCacheMock).toHaveBeenCalledWith(
// cacheEntry.archiveLocation,
// archivePath,
// undefined
// )
// expect(getArchiveFileSizeInBytesMock).toHaveBeenCalledWith(archivePath)
// expect(extractTarMock).toHaveBeenCalledTimes(1) expect(cacheKey).toBe(key)
// expect(extractTarMock).toHaveBeenCalledWith(archivePath, compression) expect(getCacheVersionMock).toHaveBeenCalledWith(
paths,
compressionMethod,
false
)
expect(getCacheDownloadURLMock).toHaveBeenCalledWith({
key,
restoreKeys: [],
version: cacheVersion
})
expect(createTempDirectoryMock).toHaveBeenCalledTimes(1)
expect(downloadCacheFileMock).toHaveBeenCalledWith(
signedDownloadUrl,
archivePath
)
expect(getArchiveFileSizeInBytesMock).toHaveBeenCalledWith(archivePath)
expect(logInfoMock).toHaveBeenCalledWith(`Cache Size: ~0 MB (142 B)`)
// expect(unlinkFileMock).toHaveBeenCalledTimes(1) expect(extractTarMock).toHaveBeenCalledTimes(1)
// expect(unlinkFileMock).toHaveBeenCalledWith(archivePath) expect(extractTarMock).toHaveBeenCalledWith(archivePath, compressionMethod)
// expect(getCompressionMock).toHaveBeenCalledTimes(1) expect(unlinkFileMock).toHaveBeenCalledTimes(1)
// }) expect(unlinkFileMock).toHaveBeenCalledWith(archivePath)
// test('restore with zstd compressed cache found', async () => { expect(compressionMethodMock).toHaveBeenCalledTimes(1)
// const paths = ['node_modules'] })
// const key = 'node-test'
// const infoMock = jest.spyOn(core, 'info') test('restore with zstd compressed cache found', async () => {
const paths = ['node_modules']
const key = 'node-test'
const logInfoMock = jest.spyOn(core, 'info')
const compressionMethod = CompressionMethod.Zstd
const signedDownloadUrl = 'https://blob-storage.local?signed=true'
const cacheVersion =
'8e2e96a184cb0cd6b48285b176c06a418f3d7fce14c29d9886fd1bb4f05c513d'
// const cacheEntry: ArtifactCacheEntry = { const getCacheVersionMock = jest.spyOn(cacheUtils, 'getCacheVersion')
// cacheKey: key, getCacheVersionMock.mockReturnValue(cacheVersion)
// scope: 'refs/heads/main',
// archiveLocation: 'www.actionscache.test/download'
// }
// const getCacheMock = jest.spyOn(cacheHttpClient, 'getCacheEntry')
// getCacheMock.mockImplementation(async () => {
// return Promise.resolve(cacheEntry)
// })
// const tempPath = '/foo/bar'
// const createTempDirectoryMock = jest.spyOn(cacheUtils, 'createTempDirectory') const compressionMethodMock = jest.spyOn(cacheUtils, 'getCompressionMethod')
// createTempDirectoryMock.mockImplementation(async () => { compressionMethodMock.mockReturnValue(Promise.resolve(compressionMethod))
// return Promise.resolve(tempPath)
// })
// const archivePath = path.join(tempPath, CacheFilename.Zstd) const getCacheDownloadURLMock = jest.spyOn(
// const downloadCacheMock = jest.spyOn(cacheHttpClient, 'downloadCache') CacheServiceClientJSON.prototype,
'GetCacheEntryDownloadURL'
)
getCacheDownloadURLMock.mockReturnValue(
Promise.resolve({ok: true, signedDownloadUrl})
)
// const fileSize = 62915000 const tempPath = '/foo/bar'
// const getArchiveFileSizeInBytesMock = jest
// .spyOn(cacheUtils, 'getArchiveFileSizeInBytes')
// .mockReturnValue(fileSize)
// const extractTarMock = jest.spyOn(tar, 'extractTar') const createTempDirectoryMock = jest.spyOn(cacheUtils, 'createTempDirectory')
// const compression = CompressionMethod.Zstd createTempDirectoryMock.mockImplementation(async () => {
// const getCompressionMock = jest return Promise.resolve(tempPath)
// .spyOn(cacheUtils, 'getCompressionMethod') })
// .mockReturnValue(Promise.resolve(compression))
// const cacheKey = await restoreCache(paths, key) const archivePath = path.join(tempPath, CacheFilename.Zstd)
const downloadCacheFileMock = jest.spyOn(
downloadCacheModule,
'downloadCacheFile'
)
downloadCacheFileMock.mockReturnValue(
Promise.resolve({} as BlobDownloadResponseParsed)
)
// expect(cacheKey).toBe(key) const fileSize = 62915000
// expect(getCacheMock).toHaveBeenCalledWith([key], paths, { const getArchiveFileSizeInBytesMock = jest
// compressionMethod: compression, .spyOn(cacheUtils, 'getArchiveFileSizeInBytes')
// enableCrossOsArchive: false .mockReturnValue(fileSize)
// })
// expect(createTempDirectoryMock).toHaveBeenCalledTimes(1)
// expect(downloadCacheMock).toHaveBeenCalledWith(
// cacheEntry.archiveLocation,
// archivePath,
// undefined
// )
// expect(getArchiveFileSizeInBytesMock).toHaveBeenCalledWith(archivePath)
// expect(infoMock).toHaveBeenCalledWith(`Cache Size: ~60 MB (62915000 B)`)
// expect(extractTarMock).toHaveBeenCalledTimes(1) const extractTarMock = jest.spyOn(tar, 'extractTar')
// expect(extractTarMock).toHaveBeenCalledWith(archivePath, compression) const unlinkFileMock = jest.spyOn(cacheUtils, 'unlinkFile')
// expect(getCompressionMock).toHaveBeenCalledTimes(1)
// }) const cacheKey = await restoreCache(paths, key)
expect(cacheKey).toBe(key)
expect(getCacheVersionMock).toHaveBeenCalledWith(
paths,
compressionMethod,
false
)
expect(getCacheDownloadURLMock).toHaveBeenCalledWith({
key,
restoreKeys: [],
version: cacheVersion
})
expect(createTempDirectoryMock).toHaveBeenCalledTimes(1)
expect(downloadCacheFileMock).toHaveBeenCalledWith(
signedDownloadUrl,
archivePath
)
expect(getArchiveFileSizeInBytesMock).toHaveBeenCalledWith(archivePath)
expect(logInfoMock).toHaveBeenCalledWith(`Cache Size: ~60 MB (62915000 B)`)
expect(extractTarMock).toHaveBeenCalledTimes(1)
expect(extractTarMock).toHaveBeenCalledWith(archivePath, compressionMethod)
expect(unlinkFileMock).toHaveBeenCalledTimes(1)
expect(unlinkFileMock).toHaveBeenCalledWith(archivePath)
expect(compressionMethodMock).toHaveBeenCalledTimes(1)
})
// test('restore with cache found for restore key', async () => { // test('restore with cache found for restore key', async () => {
// const paths = ['node_modules'] // const paths = ['node_modules']

View File

@ -3,18 +3,18 @@ import * as path from 'path'
import * as utils from './internal/cacheUtils' import * as utils from './internal/cacheUtils'
import * as cacheHttpClient from './internal/cacheHttpClient' import * as cacheHttpClient from './internal/cacheHttpClient'
import * as cacheTwirpClient from './internal/shared/cacheTwirpClient' import * as cacheTwirpClient from './internal/shared/cacheTwirpClient'
import { getCacheServiceVersion, isGhes } from './internal/config' import {getCacheServiceVersion, isGhes} from './internal/config'
import { DownloadOptions, UploadOptions } from './options' import {DownloadOptions, UploadOptions} from './options'
import { createTar, extractTar, listTar } from './internal/tar' import {createTar, extractTar, listTar} from './internal/tar'
import { import {
CreateCacheEntryRequest, CreateCacheEntryRequest,
FinalizeCacheEntryUploadRequest, FinalizeCacheEntryUploadRequest,
FinalizeCacheEntryUploadResponse, FinalizeCacheEntryUploadResponse,
GetCacheEntryDownloadURLRequest GetCacheEntryDownloadURLRequest
} from './generated/results/api/v1/cache' } from './generated/results/api/v1/cache'
import { CacheFileSizeLimit } from './internal/constants' import {CacheFileSizeLimit} from './internal/constants'
import { uploadCacheFile } from './internal/blob/upload-cache' import {uploadCacheFile} from './internal/blob/upload-cache'
import { downloadCacheFile } from './internal/blob/download-cache' import {downloadCacheFile} from './internal/blob/download-cache'
export class ValidationError extends Error { export class ValidationError extends Error {
constructor(message: string) { constructor(message: string) {
super(message) super(message)
@ -405,9 +405,9 @@ async function saveCacheV1(
} else if (reserveCacheResponse?.statusCode === 400) { } else if (reserveCacheResponse?.statusCode === 400) {
throw new Error( throw new Error(
reserveCacheResponse?.error?.message ?? reserveCacheResponse?.error?.message ??
`Cache size of ~${Math.round( `Cache size of ~${Math.round(
archiveFileSize / (1024 * 1024) archiveFileSize / (1024 * 1024)
)} MB (${archiveFileSize} B) is over the data cap limit, not saving cache.` )} MB (${archiveFileSize} B) is over the data cap limit, not saving cache.`
) )
} else { } else {
throw new ReserveCacheError( throw new ReserveCacheError(

View File

@ -3,13 +3,14 @@ import * as core from '@actions/core'
import { import {
BlobClient, BlobClient,
BlockBlobClient, BlockBlobClient,
BlobDownloadOptions BlobDownloadOptions,
BlobDownloadResponseParsed
} from '@azure/storage-blob' } from '@azure/storage-blob'
export async function downloadCacheFile( export async function downloadCacheFile(
signedUploadURL: string, signedUploadURL: string,
archivePath: string archivePath: string
): Promise<{}> { ): Promise<BlobDownloadResponseParsed> {
const downloadOptions: BlobDownloadOptions = { const downloadOptions: BlobDownloadOptions = {
maxRetryRequests: 5 maxRetryRequests: 5
} }