1
0
Fork 0

Catch SIGINT/Ctrl+C during installs and run cleanups on all packages before exiting

pull/8923/head
Jordi Boggiano 2020-05-22 13:24:30 +02:00
parent 385655f02a
commit a07f9ffba9
No known key found for this signature in database
GPG Key ID: 7BBD42C429EC80BC
1 changed files with 90 additions and 47 deletions

View File

@ -171,34 +171,89 @@ class InstallationManager
public function execute(RepositoryInterface $repo, array $operations, $devMode = true, $runScripts = true) public function execute(RepositoryInterface $repo, array $operations, $devMode = true, $runScripts = true)
{ {
$promises = array(); $promises = array();
$cleanupPromises = array();
foreach ($operations as $operation) { $loop = $this->loop;
$opType = $operation->getOperationType(); $runCleanup = function () use (&$cleanupPromises, $loop) {
$promise = null; $promises = array();
if ($opType === 'install') { foreach ($cleanupPromises as $cleanup) {
$package = $operation->getPackage(); $promises[] = new \React\Promise\Promise(function ($resolve, $reject) use ($cleanup) {
$installer = $this->getInstaller($package->getType()); $promise = $cleanup();
$promise = $installer->download($package); if (null === $promise) {
} elseif ($opType === 'update') { $resolve();
$target = $operation->getTargetPackage(); } else {
$targetType = $target->getType(); $promise->then(function () use ($resolve) {
$installer = $this->getInstaller($targetType); $resolve();
$promise = $installer->download($target, $operation->getInitialPackage()); });
}
});
} }
if (!empty($promises)) {
$loop->wait($promises);
}
};
// handler Ctrl+C for unix-like systems
$handleInterrupts = function_exists('pcntl_async_signals') && function_exists('pcntl_signal');
$prevHandler = null;
if ($handleInterrupts) {
pcntl_async_signals(true);
$prevHandler = pcntl_signal_get_handler(SIGINT);
pcntl_signal(SIGINT, function ($sig) use ($runCleanup, $prevHandler) {
$runCleanup();
if (!in_array($prevHandler, array(SIG_DFL, SIG_IGN), true)) {
call_user_func($prevHandler, $sig);
}
exit(130);
});
}
try {
foreach ($operations as $index => $operation) {
$opType = $operation->getOperationType();
// ignoring alias ops as they don't need to execute anything at this stage
if (!in_array($opType, array('update', 'install', 'uninstall'))) {
continue;
}
if ($opType === 'update') {
$package = $operation->getTargetPackage();
$initialPackage = $operation->getInitialPackage();
} else {
$package = $operation->getPackage();
$initialPackage = null;
}
$installer = $this->getInstaller($package->getType());
$cleanupPromises[$index] = function () use ($opType, $installer, $package, $initialPackage) {
// avoid calling cleanup if the download was not even initialized for a package
// as without installation source configured nothing will work
if (!$package->getInstallationSource()) {
return;
}
return $installer->cleanup($opType, $package, $initialPackage);
};
if ($opType !== 'uninstall') {
$promise = $installer->download($package, $initialPackage);
if ($promise) { if ($promise) {
$promises[] = $promise; $promises[] = $promise;
} }
} }
}
// execute all downloads first
if (!empty($promises)) { if (!empty($promises)) {
$this->loop->wait($promises); $this->loop->wait($promises);
} }
$cleanupPromises = array(); foreach ($operations as $index => $operation) {
try {
foreach ($operations as $operation) {
$opType = $operation->getOperationType(); $opType = $operation->getOperationType();
// ignoring alias ops as they don't need to execute anything // ignoring alias ops as they don't need to execute anything
@ -236,13 +291,9 @@ class InstallationManager
$promise = new \React\Promise\Promise(function ($resolve, $reject) { $resolve(); }); $promise = new \React\Promise\Promise(function ($resolve, $reject) { $resolve(); });
} }
$cleanupPromise = function () use ($opType, $installer, $package, $initialPackage) {
return $installer->cleanup($opType, $package, $initialPackage);
};
$promise = $promise->then(function () use ($opType, $installManager, $repo, $operation) { $promise = $promise->then(function () use ($opType, $installManager, $repo, $operation) {
return $installManager->$opType($repo, $operation); return $installManager->$opType($repo, $operation);
})->then($cleanupPromise) })->then($cleanupPromises[$index])
->then(function () use ($opType, $runScripts, $dispatcher, $installManager, $devMode, $repo, $operations, $operation) { ->then(function () use ($opType, $runScripts, $dispatcher, $installManager, $devMode, $repo, $operations, $operation) {
$repo->write($devMode, $installManager); $repo->write($devMode, $installManager);
@ -256,35 +307,27 @@ class InstallationManager
throw $e; throw $e;
}); });
$cleanupPromises[] = $cleanupPromise;
$promises[] = $promise; $promises[] = $promise;
} }
// execute all prepare => installs/updates/removes => cleanup steps
if (!empty($promises)) { if (!empty($promises)) {
$this->loop->wait($promises); $this->loop->wait($promises);
} }
} catch (\Exception $e) { } catch (\Exception $e) {
$promises = array(); $runCleanup();
foreach ($cleanupPromises as $cleanup) {
$promises[] = new \React\Promise\Promise(function ($resolve, $reject) use ($cleanup) {
$promise = $cleanup();
if (null === $promise) {
$resolve();
} else {
$promise->then(function () use ($resolve) {
$resolve();
});
}
});
}
if (!empty($promises)) { if ($handleInterrupts) {
$this->loop->wait($promises); pcntl_signal(SIGINT, $prevHandler);
} }
throw $e; throw $e;
} }
if ($handleInterrupts) {
pcntl_signal(SIGINT, $prevHandler);
}
// do a last write so that we write the repository even if nothing changed // do a last write so that we write the repository even if nothing changed
// as that can trigger an update of some files like InstalledVersions.php if // as that can trigger an update of some files like InstalledVersions.php if
// running a new composer version // running a new composer version