From d1ef618239c78473a711625358ad059e74c4355b Mon Sep 17 00:00:00 2001 From: Rico Huijbers Date: Wed, 22 May 2019 10:56:15 +0200 Subject: [PATCH] fix(jsii-pacmak): retry .NET build a couple of times (#509) Try to survive NuGet's failing due to race conditions. --- packages/jsii-pacmak/lib/targets/dotnet.ts | 3 +- packages/jsii-pacmak/lib/util.ts | 79 +++++++++++++++------- 2 files changed, 55 insertions(+), 27 deletions(-) diff --git a/packages/jsii-pacmak/lib/targets/dotnet.ts b/packages/jsii-pacmak/lib/targets/dotnet.ts index 863028d2d6..7b92496953 100644 --- a/packages/jsii-pacmak/lib/targets/dotnet.ts +++ b/packages/jsii-pacmak/lib/targets/dotnet.ts @@ -53,10 +53,11 @@ export default class Dotnet extends Target { const packageId: string = pkg.jsii.targets.dotnet.packageId; const project: string = path.join(packageId, `${packageId}.csproj`); + // Add retry as NuGet on Ubuntu is prone to failing due to race conditions await shell( 'dotnet', [ 'build', project, '-c', 'Release' ], - { cwd: sourceDir } + { cwd: sourceDir, retry: true } ); await this.copyFiles( diff --git a/packages/jsii-pacmak/lib/util.ts b/packages/jsii-pacmak/lib/util.ts index 1eb5719f37..1c2556ace6 100644 --- a/packages/jsii-pacmak/lib/util.ts +++ b/packages/jsii-pacmak/lib/util.ts @@ -4,6 +4,15 @@ import spec = require('jsii-spec'); import path = require('path'); import logging = require('./logging'); +export interface ShellOptions extends SpawnOptions { + /** + * Retry execution up to 3 times if it fails + * + * @default false + */ + retry?: boolean; +} + /** * Given an npm package directory and a dependency name, returns the package directory of the dep. * @param packageDir the root of the package declaring the dependency. @@ -15,33 +24,51 @@ export function resolveDependencyDirectory(packageDir: string, dependencyName: s return path.dirname(require.resolve(`${dependencyName}/package.json`, { paths: lookupPaths })); } -export function shell(cmd: string, args: string[], options: SpawnOptions): Promise { - return new Promise((resolve, reject) => { - logging.debug(cmd, args.join(' '), JSON.stringify(options)); - const child = spawn(cmd, args, { ...options, shell: true, env: { ...process.env, ...options.env || {} }, stdio: ['ignore', 'pipe', 'pipe'] }); - const stdout = new Array(); - const stderr = new Array(); - child.stdout.on('data', chunk => { - if (logging.level >= logging.LEVEL_VERBOSE) { - process.stderr.write(chunk); // notice - we emit all build output to stderr - } - stdout.push(Buffer.from(chunk)); - }); - child.stderr.on('data', chunk => { - if (logging.level >= logging.LEVEL_VERBOSE) { - process.stderr.write(chunk); - } - stderr.push(Buffer.from(chunk)); - }); - child.once('error', reject); - child.once('exit', (code, signal) => { - const out = Buffer.concat(stdout).toString('utf-8'); - if (code === 0) { return resolve(out); } - const err = Buffer.concat(stderr).toString('utf-8'); - if (code != null) { return reject(new Error(`Process exited with status ${code}\n${out}\n${err}`)); } - reject(new Error(`Process terminated by signal ${signal}\n${out}\n${err}`)); +export async function shell(cmd: string, args: string[], options: ShellOptions): Promise { + function spawn1() { + return new Promise((resolve, reject) => { + logging.debug(cmd, args.join(' '), JSON.stringify(options)); + const child = spawn(cmd, args, { + ...options, + shell: true, + env: { ...process.env, ...options.env || {} }, + stdio: ['ignore', 'pipe', 'pipe'] + }); + const stdout = new Array(); + const stderr = new Array(); + child.stdout.on('data', chunk => { + if (logging.level >= logging.LEVEL_VERBOSE) { + process.stderr.write(chunk); // notice - we emit all build output to stderr + } + stdout.push(Buffer.from(chunk)); + }); + child.stderr.on('data', chunk => { + if (logging.level >= logging.LEVEL_VERBOSE) { + process.stderr.write(chunk); + } + stderr.push(Buffer.from(chunk)); + }); + child.once('error', reject); + child.once('exit', (code, signal) => { + const out = Buffer.concat(stdout).toString('utf-8'); + if (code === 0) { return resolve(out); } + const err = Buffer.concat(stderr).toString('utf-8'); + if (code != null) { return reject(new Error(`Process exited with status ${code}\n${out}\n${err}`)); } + reject(new Error(`Process terminated by signal ${signal}\n${out}\n${err}`)); + }); }); - }); + } + + let attempts = options.retry ? 3 : 1; + while (true) { + attempts--; + try { + return spawn1(); + } catch (e) { + if (attempts === 0) { throw e; } + logging.info(`${e.message} (retrying)`); + } + } } /**