mirror of https://github.com/actions/toolkit
Retry all http calls for artifact upload and download (#675)
* Retry all http calls for artifact upload and download * Extra debug information * Fix lint * Always read response body * PR Feedback * Change error message if patch call fails * Add exponential backoff when retrying * Rework tests and add diagnostic info if exception thrown * Fix lint * fix lint error for real this time * PR cleanup * 0.5.0 @actions/artifact release * Display diagnostic info if non-retryable code is hitpull/678/head
parent
73d5917a6b
commit
c861dd8859
|
@ -50,3 +50,7 @@
|
||||||
|
|
||||||
- Improved retry-ability when a partial artifact download is encountered
|
- Improved retry-ability when a partial artifact download is encountered
|
||||||
|
|
||||||
|
### 0.5.0
|
||||||
|
|
||||||
|
- Improved retry-ability for all http calls during artifact upload and download if an error is encountered
|
||||||
|
|
||||||
|
|
|
@ -71,7 +71,7 @@ describe('Download Tests', () => {
|
||||||
setupFailedResponse()
|
setupFailedResponse()
|
||||||
const downloadHttpClient = new DownloadHttpClient()
|
const downloadHttpClient = new DownloadHttpClient()
|
||||||
expect(downloadHttpClient.listArtifacts()).rejects.toThrow(
|
expect(downloadHttpClient.listArtifacts()).rejects.toThrow(
|
||||||
'Unable to list artifacts for the run'
|
'List Artifacts failed: Artifact service responded with 500'
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -113,7 +113,7 @@ describe('Download Tests', () => {
|
||||||
configVariables.getRuntimeUrl()
|
configVariables.getRuntimeUrl()
|
||||||
)
|
)
|
||||||
).rejects.toThrow(
|
).rejects.toThrow(
|
||||||
`Unable to get ContainersItems from ${configVariables.getRuntimeUrl()}`
|
`Get Container Items failed: Artifact service responded with 500`
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,114 @@
|
||||||
|
import * as http from 'http'
|
||||||
|
import * as net from 'net'
|
||||||
|
import * as core from '@actions/core'
|
||||||
|
import * as configVariables from '../src/internal/config-variables'
|
||||||
|
import {retry} from '../src/internal/requestUtils'
|
||||||
|
import {IHttpClientResponse} from '@actions/http-client/interfaces'
|
||||||
|
import {HttpClientResponse} from '@actions/http-client'
|
||||||
|
|
||||||
|
jest.mock('../src/internal/config-variables')
|
||||||
|
|
||||||
|
interface ITestResult {
|
||||||
|
responseCode: number
|
||||||
|
errorMessage: string | null
|
||||||
|
}
|
||||||
|
|
||||||
|
async function testRetry(
|
||||||
|
responseCodes: number[],
|
||||||
|
expectedResult: ITestResult
|
||||||
|
): Promise<void> {
|
||||||
|
const reverse = responseCodes.reverse() // Reverse responses since we pop from end
|
||||||
|
if (expectedResult.errorMessage) {
|
||||||
|
// we expect some exception to be thrown
|
||||||
|
expect(
|
||||||
|
retry(
|
||||||
|
'test',
|
||||||
|
async () => handleResponse(reverse.pop()),
|
||||||
|
new Map(), // extra error message for any particular http codes
|
||||||
|
configVariables.getRetryLimit()
|
||||||
|
)
|
||||||
|
).rejects.toThrow(expectedResult.errorMessage)
|
||||||
|
} else {
|
||||||
|
// we expect a correct status code to be returned
|
||||||
|
const actualResult = await retry(
|
||||||
|
'test',
|
||||||
|
async () => handleResponse(reverse.pop()),
|
||||||
|
new Map(), // extra error message for any particular http codes
|
||||||
|
configVariables.getRetryLimit()
|
||||||
|
)
|
||||||
|
expect(actualResult.message.statusCode).toEqual(expectedResult.responseCode)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function handleResponse(
|
||||||
|
testResponseCode: number | undefined
|
||||||
|
): Promise<IHttpClientResponse> {
|
||||||
|
if (!testResponseCode) {
|
||||||
|
throw new Error(
|
||||||
|
'Test incorrectly set up. reverse.pop() was called too many times so not enough test response codes were supplied'
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
return setupSingleMockResponse(testResponseCode)
|
||||||
|
}
|
||||||
|
|
||||||
|
beforeAll(async () => {
|
||||||
|
// mock all output so that there is less noise when running tests
|
||||||
|
jest.spyOn(console, 'log').mockImplementation(() => {})
|
||||||
|
jest.spyOn(core, 'debug').mockImplementation(() => {})
|
||||||
|
jest.spyOn(core, 'info').mockImplementation(() => {})
|
||||||
|
jest.spyOn(core, 'warning').mockImplementation(() => {})
|
||||||
|
jest.spyOn(core, 'error').mockImplementation(() => {})
|
||||||
|
})
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Helpers used to setup mocking for the HttpClient
|
||||||
|
*/
|
||||||
|
async function emptyMockReadBody(): Promise<string> {
|
||||||
|
return new Promise(resolve => {
|
||||||
|
resolve()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
async function setupSingleMockResponse(
|
||||||
|
statusCode: number
|
||||||
|
): Promise<IHttpClientResponse> {
|
||||||
|
const mockMessage = new http.IncomingMessage(new net.Socket())
|
||||||
|
const mockReadBody = emptyMockReadBody
|
||||||
|
mockMessage.statusCode = statusCode
|
||||||
|
return new Promise<HttpClientResponse>(resolve => {
|
||||||
|
resolve({
|
||||||
|
message: mockMessage,
|
||||||
|
readBody: mockReadBody
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
test('retry works on successful response', async () => {
|
||||||
|
await testRetry([200], {
|
||||||
|
responseCode: 200,
|
||||||
|
errorMessage: null
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
test('retry works after retryable status code', async () => {
|
||||||
|
await testRetry([503, 200], {
|
||||||
|
responseCode: 200,
|
||||||
|
errorMessage: null
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
test('retry fails after exhausting retries', async () => {
|
||||||
|
// __mocks__/config-variables caps the max retry count in tests to 2
|
||||||
|
await testRetry([503, 503, 200], {
|
||||||
|
responseCode: 200,
|
||||||
|
errorMessage: 'test failed: Artifact service responded with 503'
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
test('retry fails after non-retryable status code', async () => {
|
||||||
|
await testRetry([500, 200], {
|
||||||
|
responseCode: 500,
|
||||||
|
errorMessage: 'test failed: Artifact service responded with 500'
|
||||||
|
})
|
||||||
|
})
|
|
@ -102,7 +102,7 @@ describe('Upload Tests', () => {
|
||||||
uploadHttpClient.createArtifactInFileContainer(artifactName)
|
uploadHttpClient.createArtifactInFileContainer(artifactName)
|
||||||
).rejects.toEqual(
|
).rejects.toEqual(
|
||||||
new Error(
|
new Error(
|
||||||
`Unable to create a container for the artifact invalid-artifact-name at ${getArtifactUrl()}`
|
`Create Artifact Container failed: The artifact name invalid-artifact-name is not valid. Request URL ${getArtifactUrl()}`
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
|
@ -125,7 +125,7 @@ describe('Upload Tests', () => {
|
||||||
uploadHttpClient.createArtifactInFileContainer(artifactName)
|
uploadHttpClient.createArtifactInFileContainer(artifactName)
|
||||||
).rejects.toEqual(
|
).rejects.toEqual(
|
||||||
new Error(
|
new Error(
|
||||||
'Artifact storage quota has been hit. Unable to upload any new artifacts'
|
'Create Artifact Container failed: Artifact storage quota has been hit. Unable to upload any new artifacts'
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
|
@ -362,7 +362,9 @@ describe('Upload Tests', () => {
|
||||||
const uploadHttpClient = new UploadHttpClient()
|
const uploadHttpClient = new UploadHttpClient()
|
||||||
expect(
|
expect(
|
||||||
uploadHttpClient.patchArtifactSize(-2, 'my-artifact')
|
uploadHttpClient.patchArtifactSize(-2, 'my-artifact')
|
||||||
).rejects.toThrow('Unable to finish uploading artifact my-artifact')
|
).rejects.toThrow(
|
||||||
|
'Finalize artifact upload failed: Artifact service responded with 400'
|
||||||
|
)
|
||||||
})
|
})
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
{
|
{
|
||||||
"name": "@actions/artifact",
|
"name": "@actions/artifact",
|
||||||
"version": "0.4.2",
|
"version": "0.5.0",
|
||||||
"lockfileVersion": 1,
|
"lockfileVersion": 1,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
{
|
{
|
||||||
"name": "@actions/artifact",
|
"name": "@actions/artifact",
|
||||||
"version": "0.4.2",
|
"version": "0.5.0",
|
||||||
"preview": true,
|
"preview": true,
|
||||||
"description": "Actions artifact lib",
|
"description": "Actions artifact lib",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
|
|
|
@ -11,7 +11,8 @@ import {
|
||||||
tryGetRetryAfterValueTimeInMilliseconds,
|
tryGetRetryAfterValueTimeInMilliseconds,
|
||||||
displayHttpDiagnostics,
|
displayHttpDiagnostics,
|
||||||
getFileSize,
|
getFileSize,
|
||||||
rmFile
|
rmFile,
|
||||||
|
sleep
|
||||||
} from './utils'
|
} from './utils'
|
||||||
import {URL} from 'url'
|
import {URL} from 'url'
|
||||||
import {StatusReporter} from './status-reporter'
|
import {StatusReporter} from './status-reporter'
|
||||||
|
@ -22,6 +23,7 @@ import {HttpManager} from './http-manager'
|
||||||
import {DownloadItem} from './download-specification'
|
import {DownloadItem} from './download-specification'
|
||||||
import {getDownloadFileConcurrency, getRetryLimit} from './config-variables'
|
import {getDownloadFileConcurrency, getRetryLimit} from './config-variables'
|
||||||
import {IncomingHttpHeaders} from 'http'
|
import {IncomingHttpHeaders} from 'http'
|
||||||
|
import {retryHttpClientRequest} from './requestUtils'
|
||||||
|
|
||||||
export class DownloadHttpClient {
|
export class DownloadHttpClient {
|
||||||
// http manager is used for concurrent connections when downloading multiple files at once
|
// http manager is used for concurrent connections when downloading multiple files at once
|
||||||
|
@ -46,16 +48,11 @@ export class DownloadHttpClient {
|
||||||
// use the first client from the httpManager, `keep-alive` is not used so the connection will close immediately
|
// use the first client from the httpManager, `keep-alive` is not used so the connection will close immediately
|
||||||
const client = this.downloadHttpManager.getClient(0)
|
const client = this.downloadHttpManager.getClient(0)
|
||||||
const headers = getDownloadHeaders('application/json')
|
const headers = getDownloadHeaders('application/json')
|
||||||
const response = await client.get(artifactUrl, headers)
|
const response = await retryHttpClientRequest('List Artifacts', async () =>
|
||||||
const body: string = await response.readBody()
|
client.get(artifactUrl, headers)
|
||||||
|
|
||||||
if (isSuccessStatusCode(response.message.statusCode) && body) {
|
|
||||||
return JSON.parse(body)
|
|
||||||
}
|
|
||||||
displayHttpDiagnostics(response)
|
|
||||||
throw new Error(
|
|
||||||
`Unable to list artifacts for the run. Resource Url ${artifactUrl}`
|
|
||||||
)
|
)
|
||||||
|
const body: string = await response.readBody()
|
||||||
|
return JSON.parse(body)
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -74,15 +71,13 @@ export class DownloadHttpClient {
|
||||||
// use the first client from the httpManager, `keep-alive` is not used so the connection will close immediately
|
// use the first client from the httpManager, `keep-alive` is not used so the connection will close immediately
|
||||||
const client = this.downloadHttpManager.getClient(0)
|
const client = this.downloadHttpManager.getClient(0)
|
||||||
const headers = getDownloadHeaders('application/json')
|
const headers = getDownloadHeaders('application/json')
|
||||||
const response = await client.get(resourceUrl.toString(), headers)
|
const response = await retryHttpClientRequest(
|
||||||
|
'Get Container Items',
|
||||||
|
async () => client.get(resourceUrl.toString(), headers)
|
||||||
|
)
|
||||||
const body: string = await response.readBody()
|
const body: string = await response.readBody()
|
||||||
|
|
||||||
if (isSuccessStatusCode(response.message.statusCode) && body) {
|
|
||||||
return JSON.parse(body)
|
return JSON.parse(body)
|
||||||
}
|
}
|
||||||
displayHttpDiagnostics(response)
|
|
||||||
throw new Error(`Unable to get ContainersItems from ${resourceUrl}`)
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Concurrently downloads all the files that are part of an artifact
|
* Concurrently downloads all the files that are part of an artifact
|
||||||
|
@ -188,14 +183,14 @@ export class DownloadHttpClient {
|
||||||
core.info(
|
core.info(
|
||||||
`Backoff due to too many requests, retry #${retryCount}. Waiting for ${retryAfterValue} milliseconds before continuing the download`
|
`Backoff due to too many requests, retry #${retryCount}. Waiting for ${retryAfterValue} milliseconds before continuing the download`
|
||||||
)
|
)
|
||||||
await new Promise(resolve => setTimeout(resolve, retryAfterValue))
|
await sleep(retryAfterValue)
|
||||||
} else {
|
} else {
|
||||||
// Back off using an exponential value that depends on the retry count
|
// Back off using an exponential value that depends on the retry count
|
||||||
const backoffTime = getExponentialRetryTimeInMilliseconds(retryCount)
|
const backoffTime = getExponentialRetryTimeInMilliseconds(retryCount)
|
||||||
core.info(
|
core.info(
|
||||||
`Exponential backoff for retry #${retryCount}. Waiting for ${backoffTime} milliseconds before continuing the download`
|
`Exponential backoff for retry #${retryCount}. Waiting for ${backoffTime} milliseconds before continuing the download`
|
||||||
)
|
)
|
||||||
await new Promise(resolve => setTimeout(resolve, backoffTime))
|
await sleep(backoffTime)
|
||||||
}
|
}
|
||||||
core.info(
|
core.info(
|
||||||
`Finished backoff for retry #${retryCount}, continuing with download`
|
`Finished backoff for retry #${retryCount}, continuing with download`
|
||||||
|
|
|
@ -0,0 +1,79 @@
|
||||||
|
import {IHttpClientResponse} from '@actions/http-client/interfaces'
|
||||||
|
import {
|
||||||
|
isRetryableStatusCode,
|
||||||
|
isSuccessStatusCode,
|
||||||
|
sleep,
|
||||||
|
getExponentialRetryTimeInMilliseconds,
|
||||||
|
displayHttpDiagnostics
|
||||||
|
} from './utils'
|
||||||
|
import * as core from '@actions/core'
|
||||||
|
import {getRetryLimit} from './config-variables'
|
||||||
|
|
||||||
|
export async function retry(
|
||||||
|
name: string,
|
||||||
|
operation: () => Promise<IHttpClientResponse>,
|
||||||
|
customErrorMessages: Map<number, string>,
|
||||||
|
maxAttempts: number
|
||||||
|
): Promise<IHttpClientResponse> {
|
||||||
|
let response: IHttpClientResponse | undefined = undefined
|
||||||
|
let statusCode: number | undefined = undefined
|
||||||
|
let isRetryable = false
|
||||||
|
let errorMessage = ''
|
||||||
|
let customErrorInformation: string | undefined = undefined
|
||||||
|
let attempt = 1
|
||||||
|
|
||||||
|
while (attempt <= maxAttempts) {
|
||||||
|
try {
|
||||||
|
response = await operation()
|
||||||
|
statusCode = response.message.statusCode
|
||||||
|
|
||||||
|
if (isSuccessStatusCode(statusCode)) {
|
||||||
|
return response
|
||||||
|
}
|
||||||
|
|
||||||
|
// Extra error information that we want to display if a particular response code is hit
|
||||||
|
if (statusCode) {
|
||||||
|
customErrorInformation = customErrorMessages.get(statusCode)
|
||||||
|
}
|
||||||
|
|
||||||
|
isRetryable = isRetryableStatusCode(statusCode)
|
||||||
|
errorMessage = `Artifact service responded with ${statusCode}`
|
||||||
|
} catch (error) {
|
||||||
|
isRetryable = true
|
||||||
|
errorMessage = error.message
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!isRetryable) {
|
||||||
|
core.info(`${name} - Error is not retryable`)
|
||||||
|
if (response) {
|
||||||
|
displayHttpDiagnostics(response)
|
||||||
|
}
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
core.info(
|
||||||
|
`${name} - Attempt ${attempt} of ${maxAttempts} failed with error: ${errorMessage}`
|
||||||
|
)
|
||||||
|
|
||||||
|
await sleep(getExponentialRetryTimeInMilliseconds(attempt))
|
||||||
|
attempt++
|
||||||
|
}
|
||||||
|
|
||||||
|
if (response) {
|
||||||
|
displayHttpDiagnostics(response)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (customErrorInformation) {
|
||||||
|
throw Error(`${name} failed: ${customErrorInformation}`)
|
||||||
|
}
|
||||||
|
throw Error(`${name} failed: ${errorMessage}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function retryHttpClientRequest<T>(
|
||||||
|
name: string,
|
||||||
|
method: () => Promise<IHttpClientResponse>,
|
||||||
|
customErrorMessages: Map<number, string> = new Map(),
|
||||||
|
maxAttempts = getRetryLimit()
|
||||||
|
): Promise<IHttpClientResponse> {
|
||||||
|
return await retry(name, method, customErrorMessages, maxAttempts)
|
||||||
|
}
|
|
@ -15,11 +15,11 @@ import {
|
||||||
isRetryableStatusCode,
|
isRetryableStatusCode,
|
||||||
isSuccessStatusCode,
|
isSuccessStatusCode,
|
||||||
isThrottledStatusCode,
|
isThrottledStatusCode,
|
||||||
isForbiddenStatusCode,
|
|
||||||
displayHttpDiagnostics,
|
displayHttpDiagnostics,
|
||||||
getExponentialRetryTimeInMilliseconds,
|
getExponentialRetryTimeInMilliseconds,
|
||||||
tryGetRetryAfterValueTimeInMilliseconds,
|
tryGetRetryAfterValueTimeInMilliseconds,
|
||||||
getProperRetention
|
getProperRetention,
|
||||||
|
sleep
|
||||||
} from './utils'
|
} from './utils'
|
||||||
import {
|
import {
|
||||||
getUploadChunkSize,
|
getUploadChunkSize,
|
||||||
|
@ -31,12 +31,13 @@ import {promisify} from 'util'
|
||||||
import {URL} from 'url'
|
import {URL} from 'url'
|
||||||
import {performance} from 'perf_hooks'
|
import {performance} from 'perf_hooks'
|
||||||
import {StatusReporter} from './status-reporter'
|
import {StatusReporter} from './status-reporter'
|
||||||
import {HttpClientResponse} from '@actions/http-client/index'
|
import {HttpCodes} from '@actions/http-client'
|
||||||
import {IHttpClientResponse} from '@actions/http-client/interfaces'
|
import {IHttpClientResponse} from '@actions/http-client/interfaces'
|
||||||
import {HttpManager} from './http-manager'
|
import {HttpManager} from './http-manager'
|
||||||
import {UploadSpecification} from './upload-specification'
|
import {UploadSpecification} from './upload-specification'
|
||||||
import {UploadOptions} from './upload-options'
|
import {UploadOptions} from './upload-options'
|
||||||
import {createGZipFileOnDisk, createGZipFileInBuffer} from './upload-gzip'
|
import {createGZipFileOnDisk, createGZipFileInBuffer} from './upload-gzip'
|
||||||
|
import {retryHttpClientRequest} from './requestUtils'
|
||||||
const stat = promisify(fs.stat)
|
const stat = promisify(fs.stat)
|
||||||
|
|
||||||
export class UploadHttpClient {
|
export class UploadHttpClient {
|
||||||
|
@ -80,23 +81,28 @@ export class UploadHttpClient {
|
||||||
// use the first client from the httpManager, `keep-alive` is not used so the connection will close immediately
|
// use the first client from the httpManager, `keep-alive` is not used so the connection will close immediately
|
||||||
const client = this.uploadHttpManager.getClient(0)
|
const client = this.uploadHttpManager.getClient(0)
|
||||||
const headers = getUploadHeaders('application/json', false)
|
const headers = getUploadHeaders('application/json', false)
|
||||||
const rawResponse = await client.post(artifactUrl, data, headers)
|
|
||||||
const body: string = await rawResponse.readBody()
|
|
||||||
|
|
||||||
if (isSuccessStatusCode(rawResponse.message.statusCode) && body) {
|
// Extra information to display when a particular HTTP code is returned
|
||||||
return JSON.parse(body)
|
// If a 403 is returned when trying to create a file container, the customer has exceeded
|
||||||
} else if (isForbiddenStatusCode(rawResponse.message.statusCode)) {
|
|
||||||
// if a 403 is returned when trying to create a file container, the customer has exceeded
|
|
||||||
// their storage quota so no new artifact containers can be created
|
// their storage quota so no new artifact containers can be created
|
||||||
throw new Error(
|
const customErrorMessages: Map<number, string> = new Map([
|
||||||
`Artifact storage quota has been hit. Unable to upload any new artifacts`
|
[
|
||||||
|
HttpCodes.Forbidden,
|
||||||
|
'Artifact storage quota has been hit. Unable to upload any new artifacts'
|
||||||
|
],
|
||||||
|
[
|
||||||
|
HttpCodes.BadRequest,
|
||||||
|
`The artifact name ${artifactName} is not valid. Request URL ${artifactUrl}`
|
||||||
|
]
|
||||||
|
])
|
||||||
|
|
||||||
|
const response = await retryHttpClientRequest(
|
||||||
|
'Create Artifact Container',
|
||||||
|
async () => client.post(artifactUrl, data, headers),
|
||||||
|
customErrorMessages
|
||||||
)
|
)
|
||||||
} else {
|
const body: string = await response.readBody()
|
||||||
displayHttpDiagnostics(rawResponse)
|
return JSON.parse(body)
|
||||||
throw new Error(
|
|
||||||
`Unable to create a container for the artifact ${artifactName} at ${artifactUrl}`
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -417,13 +423,13 @@ export class UploadHttpClient {
|
||||||
core.info(
|
core.info(
|
||||||
`Backoff due to too many requests, retry #${retryCount}. Waiting for ${retryAfterValue} milliseconds before continuing the upload`
|
`Backoff due to too many requests, retry #${retryCount}. Waiting for ${retryAfterValue} milliseconds before continuing the upload`
|
||||||
)
|
)
|
||||||
await new Promise(resolve => setTimeout(resolve, retryAfterValue))
|
await sleep(retryAfterValue)
|
||||||
} else {
|
} else {
|
||||||
const backoffTime = getExponentialRetryTimeInMilliseconds(retryCount)
|
const backoffTime = getExponentialRetryTimeInMilliseconds(retryCount)
|
||||||
core.info(
|
core.info(
|
||||||
`Exponential backoff for retry #${retryCount}. Waiting for ${backoffTime} milliseconds before continuing the upload at offset ${start}`
|
`Exponential backoff for retry #${retryCount}. Waiting for ${backoffTime} milliseconds before continuing the upload at offset ${start}`
|
||||||
)
|
)
|
||||||
await new Promise(resolve => setTimeout(resolve, backoffTime))
|
await sleep(backoffTime)
|
||||||
}
|
}
|
||||||
core.info(
|
core.info(
|
||||||
`Finished backoff for retry #${retryCount}, continuing with upload`
|
`Finished backoff for retry #${retryCount}, continuing with upload`
|
||||||
|
@ -486,7 +492,6 @@ export class UploadHttpClient {
|
||||||
* Updating the size indicates that we are done uploading all the contents of the artifact
|
* Updating the size indicates that we are done uploading all the contents of the artifact
|
||||||
*/
|
*/
|
||||||
async patchArtifactSize(size: number, artifactName: string): Promise<void> {
|
async patchArtifactSize(size: number, artifactName: string): Promise<void> {
|
||||||
const headers = getUploadHeaders('application/json', false)
|
|
||||||
const resourceUrl = new URL(getArtifactUrl())
|
const resourceUrl = new URL(getArtifactUrl())
|
||||||
resourceUrl.searchParams.append('artifactName', artifactName)
|
resourceUrl.searchParams.append('artifactName', artifactName)
|
||||||
|
|
||||||
|
@ -496,25 +501,26 @@ export class UploadHttpClient {
|
||||||
|
|
||||||
// use the first client from the httpManager, `keep-alive` is not used so the connection will close immediately
|
// use the first client from the httpManager, `keep-alive` is not used so the connection will close immediately
|
||||||
const client = this.uploadHttpManager.getClient(0)
|
const client = this.uploadHttpManager.getClient(0)
|
||||||
const response: HttpClientResponse = await client.patch(
|
const headers = getUploadHeaders('application/json', false)
|
||||||
resourceUrl.toString(),
|
|
||||||
data,
|
// Extra information to display when a particular HTTP code is returned
|
||||||
headers
|
const customErrorMessages: Map<number, string> = new Map([
|
||||||
|
[
|
||||||
|
HttpCodes.NotFound,
|
||||||
|
`An Artifact with the name ${artifactName} was not found`
|
||||||
|
]
|
||||||
|
])
|
||||||
|
|
||||||
|
// TODO retry for all possible response codes, the artifact upload is pretty much complete so it at all costs we should try to finish this
|
||||||
|
const response = await retryHttpClientRequest(
|
||||||
|
'Finalize artifact upload',
|
||||||
|
async () => client.patch(resourceUrl.toString(), data, headers),
|
||||||
|
customErrorMessages
|
||||||
)
|
)
|
||||||
const body: string = await response.readBody()
|
await response.readBody()
|
||||||
if (isSuccessStatusCode(response.message.statusCode)) {
|
|
||||||
core.debug(
|
core.debug(
|
||||||
`Artifact ${artifactName} has been successfully uploaded, total size in bytes: ${size}`
|
`Artifact ${artifactName} has been successfully uploaded, total size in bytes: ${size}`
|
||||||
)
|
)
|
||||||
} else if (response.message.statusCode === 404) {
|
|
||||||
throw new Error(`An Artifact with the name ${artifactName} was not found`)
|
|
||||||
} else {
|
|
||||||
displayHttpDiagnostics(response)
|
|
||||||
core.info(body)
|
|
||||||
throw new Error(
|
|
||||||
`Unable to finish uploading artifact ${artifactName} to ${resourceUrl}`
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -65,7 +65,7 @@ export function isForbiddenStatusCode(statusCode?: number): boolean {
|
||||||
return statusCode === HttpCodes.Forbidden
|
return statusCode === HttpCodes.Forbidden
|
||||||
}
|
}
|
||||||
|
|
||||||
export function isRetryableStatusCode(statusCode?: number): boolean {
|
export function isRetryableStatusCode(statusCode: number | undefined): boolean {
|
||||||
if (!statusCode) {
|
if (!statusCode) {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
@ -335,3 +335,7 @@ export function getProperRetention(
|
||||||
}
|
}
|
||||||
return retention
|
return retention
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export async function sleep(milliseconds: number): Promise<void> {
|
||||||
|
return new Promise(resolve => setTimeout(resolve, milliseconds))
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue