Skip to content

Commit

Permalink
[build] automatically retry all downloads (elastic#119642)
Browse files Browse the repository at this point in the history
  • Loading branch information
Spencer authored and TinLe committed Dec 22, 2021
1 parent c14dda7 commit e87243e
Show file tree
Hide file tree
Showing 8 changed files with 374 additions and 213 deletions.
179 changes: 125 additions & 54 deletions src/dev/build/lib/download.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,11 +8,12 @@

import { openSync, writeSync, unlinkSync, closeSync } from 'fs';
import { dirname } from 'path';
import { setTimeout } from 'timers/promises';

import chalk from 'chalk';
import { createHash } from 'crypto';
import Axios from 'axios';
import { ToolingLog } from '@kbn/dev-utils';
import { ToolingLog, isAxiosResponseError } from '@kbn/dev-utils';

// https://github.com/axios/axios/tree/ffea03453f77a8176c51554d5f6c3c6829294649/lib/adapters
// @ts-expect-error untyped internal module used to prevent axios from using xhr adapter in tests
Expand All @@ -30,81 +31,151 @@ function tryUnlink(path: string) {
}
}

interface DownloadOptions {
interface DownloadToDiskOptions {
log: ToolingLog;
url: string;
destination: string;
shaChecksum: string;
shaAlgorithm: string;
retries?: number;
maxAttempts?: number;
retryDelaySecMultiplier?: number;
}
export async function download(options: DownloadOptions): Promise<void> {
const { log, url, destination, shaChecksum, shaAlgorithm, retries = 0 } = options;

export async function downloadToDisk({
log,
url,
destination,
shaChecksum,
shaAlgorithm,
maxAttempts = 1,
retryDelaySecMultiplier = 5,
}: DownloadToDiskOptions) {
if (!shaChecksum) {
throw new Error(`${shaAlgorithm} checksum of ${url} not provided, refusing to download.`);
}

// mkdirp and open file outside of try/catch, we don't retry for those errors
await mkdirp(dirname(destination));
const fileHandle = openSync(destination, 'w');
if (maxAttempts < 1) {
throw new Error(`[maxAttempts=${maxAttempts}] must be >= 1`);
}

let error;
try {
log.debug(`Attempting download of ${url}`, chalk.dim(shaAlgorithm));
let attempt = 0;
while (true) {
attempt += 1;

const response = await Axios.request({
url,
responseType: 'stream',
adapter: AxiosHttpAdapter,
});
// mkdirp and open file outside of try/catch, we don't retry for those errors
await mkdirp(dirname(destination));
const fileHandle = openSync(destination, 'w');

if (response.status !== 200) {
throw new Error(`Unexpected status code ${response.status} when downloading ${url}`);
}
let error;
try {
log.debug(
`[${attempt}/${maxAttempts}] Attempting download of ${url}`,
chalk.dim(shaAlgorithm)
);

const hash = createHash(shaAlgorithm);
await new Promise((resolve, reject) => {
response.data.on('data', (chunk: Buffer) => {
hash.update(chunk);
writeSync(fileHandle, chunk);
const response = await Axios.request({
url,
responseType: 'stream',
adapter: AxiosHttpAdapter,
});

response.data.on('error', reject);
response.data.on('end', resolve);
});
if (response.status !== 200) {
throw new Error(`Unexpected status code ${response.status} when downloading ${url}`);
}

const downloadedSha = hash.digest('hex');
if (downloadedSha !== shaChecksum) {
throw new Error(
`Downloaded checksum ${downloadedSha} does not match the expected ${shaAlgorithm} checksum.`
);
const hash = createHash(shaAlgorithm);
await new Promise((resolve, reject) => {
response.data.on('data', (chunk: Buffer) => {
hash.update(chunk);
writeSync(fileHandle, chunk);
});

response.data.on('error', reject);
response.data.on('end', resolve);
});

const downloadedSha = hash.digest('hex');
if (downloadedSha !== shaChecksum) {
throw new Error(
`Downloaded checksum ${downloadedSha} does not match the expected ${shaAlgorithm} checksum.`
);
}
} catch (_error) {
error = _error;
} finally {
closeSync(fileHandle);
}
} catch (_error) {
error = _error;
} finally {
closeSync(fileHandle);
}

if (!error) {
log.debug(`Downloaded ${url} and verified checksum`);
return;
}
if (!error) {
log.debug(`Downloaded ${url} and verified checksum`);
return;
}

log.debug(`Download failed: ${error.message}`);
log.debug(`Download failed: ${error.message}`);

// cleanup downloaded data and log error
log.debug(`Deleting downloaded data at ${destination}`);
tryUnlink(destination);
// cleanup downloaded data and log error
log.debug(`Deleting downloaded data at ${destination}`);
tryUnlink(destination);

// retry if we have retries left
if (retries > 0) {
log.debug(`Retrying - ${retries} attempt remaining`);
return await download({
...options,
retries: retries - 1,
});
// retry if we have retries left
if (attempt < maxAttempts) {
const sec = attempt * retryDelaySecMultiplier;
log.info(`Retrying in ${sec} seconds`);
await setTimeout(sec * 1000);
continue;
}

throw error;
}
}

interface DownloadToStringOptions {
log: ToolingLog;
url: string;
expectStatus?: number;
maxAttempts?: number;
retryDelaySecMultiplier?: number;
}
export async function downloadToString({
log,
url,
expectStatus,
maxAttempts = 3,
retryDelaySecMultiplier = 5,
}: DownloadToStringOptions) {
let attempt = 0;
while (true) {
try {
attempt += 1;
log.debug(`[${attempt}/${maxAttempts}] Attempting download to string of [${url}]`);

const resp = await Axios.request<string>({
url,
method: 'GET',
adapter: AxiosHttpAdapter,
responseType: 'text',
validateStatus: !expectStatus ? undefined : (status) => status === expectStatus,
});

log.success(`Downloaded [${url}]`);
return resp.data;
} catch (error) {
log.warning(`Download failed: ${error.message}`);
if (isAxiosResponseError(error)) {
log.debug(
`[${error.response.status}/${error.response.statusText}] response: ${error.response.data}`
);
} else {
log.debug('received no response');
}

if ((maxAttempts ?? 3) > attempt) {
const sec = (retryDelaySecMultiplier ?? 5) * attempt;
log.info(`Retrying in ${sec} seconds`);
await setTimeout(sec * 1000);
continue;
}

throw error;
throw error;
}
}
}
Loading

0 comments on commit e87243e

Please sign in to comment.