123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506 |
- <?php
- /*
- * This file is part of Composer.
- *
- * (c) Nils Adermann <naderman@naderman.de>
- * Jordi Boggiano <j.boggiano@seld.be>
- *
- * For the full copyright and license information, please view the LICENSE
- * file that was distributed with this source code.
- */
- namespace Composer\Installer;
- use Composer\IO\IOInterface;
- use Composer\Package\PackageInterface;
- use Composer\Package\AliasPackage;
- use Composer\Repository\RepositoryInterface;
- use Composer\Repository\InstalledRepositoryInterface;
- use Composer\DependencyResolver\Operation\OperationInterface;
- use Composer\DependencyResolver\Operation\InstallOperation;
- use Composer\DependencyResolver\Operation\UpdateOperation;
- use Composer\DependencyResolver\Operation\UninstallOperation;
- use Composer\DependencyResolver\Operation\MarkAliasInstalledOperation;
- use Composer\DependencyResolver\Operation\MarkAliasUninstalledOperation;
- use Composer\EventDispatcher\EventDispatcher;
- use Composer\Util\StreamContextFactory;
- use Composer\Util\Loop;
- /**
- * Package operation manager.
- *
- * @author Konstantin Kudryashov <ever.zet@gmail.com>
- * @author Jordi Boggiano <j.boggiano@seld.be>
- * @author Nils Adermann <naderman@naderman.de>
- */
- class InstallationManager
- {
- private $installers = array();
- private $cache = array();
- private $notifiablePackages = array();
- private $loop;
- private $io;
- private $eventDispatcher;
- public function __construct(Loop $loop, IOInterface $io, EventDispatcher $eventDispatcher = null)
- {
- $this->loop = $loop;
- $this->io = $io;
- $this->eventDispatcher = $eventDispatcher;
- }
- public function reset()
- {
- $this->notifiablePackages = array();
- }
- /**
- * Adds installer
- *
- * @param InstallerInterface $installer installer instance
- */
- public function addInstaller(InstallerInterface $installer)
- {
- array_unshift($this->installers, $installer);
- $this->cache = array();
- }
- /**
- * Removes installer
- *
- * @param InstallerInterface $installer installer instance
- */
- public function removeInstaller(InstallerInterface $installer)
- {
- if (false !== ($key = array_search($installer, $this->installers, true))) {
- array_splice($this->installers, $key, 1);
- $this->cache = array();
- }
- }
- /**
- * Disables plugins.
- *
- * We prevent any plugins from being instantiated by simply
- * deactivating the installer for them. This ensure that no third-party
- * code is ever executed.
- */
- public function disablePlugins()
- {
- foreach ($this->installers as $i => $installer) {
- if (!$installer instanceof PluginInstaller) {
- continue;
- }
- unset($this->installers[$i]);
- }
- }
- /**
- * Returns installer for a specific package type.
- *
- * @param string $type package type
- *
- * @throws \InvalidArgumentException if installer for provided type is not registered
- * @return InstallerInterface
- */
- public function getInstaller($type)
- {
- $type = strtolower($type);
- if (isset($this->cache[$type])) {
- return $this->cache[$type];
- }
- foreach ($this->installers as $installer) {
- if ($installer->supports($type)) {
- return $this->cache[$type] = $installer;
- }
- }
- throw new \InvalidArgumentException('Unknown installer type: '.$type);
- }
- /**
- * Checks whether provided package is installed in one of the registered installers.
- *
- * @param InstalledRepositoryInterface $repo repository in which to check
- * @param PackageInterface $package package instance
- *
- * @return bool
- */
- public function isPackageInstalled(InstalledRepositoryInterface $repo, PackageInterface $package)
- {
- if ($package instanceof AliasPackage) {
- return $repo->hasPackage($package) && $this->isPackageInstalled($repo, $package->getAliasOf());
- }
- return $this->getInstaller($package->getType())->isInstalled($repo, $package);
- }
- /**
- * Install binary for the given package.
- * If the installer associated to this package doesn't handle that function, it'll do nothing.
- *
- * @param PackageInterface $package Package instance
- */
- public function ensureBinariesPresence(PackageInterface $package)
- {
- try {
- $installer = $this->getInstaller($package->getType());
- } catch (\InvalidArgumentException $e) {
- // no installer found for the current package type (@see `getInstaller()`)
- return;
- }
- // if the given installer support installing binaries
- if ($installer instanceof BinaryPresenceInterface) {
- $installer->ensureBinariesPresence($package);
- }
- }
- /**
- * Executes solver operation.
- *
- * @param RepositoryInterface $repo repository in which to add/remove/update packages
- * @param OperationInterface[] $operations operations to execute
- * @param bool $devMode whether the install is being run in dev mode
- * @param bool $operation whether to dispatch script events
- */
- public function execute(RepositoryInterface $repo, array $operations, $devMode = true, $runScripts = true)
- {
- $promises = array();
- $cleanupPromises = array();
- $loop = $this->loop;
- $runCleanup = function () use (&$cleanupPromises, $loop) {
- $promises = array();
- 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)) {
- $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) {
- $promises[] = $promise;
- }
- }
- }
- // execute all downloads first
- if (!empty($promises)) {
- $this->loop->wait($promises);
- }
- foreach ($operations as $index => $operation) {
- $opType = $operation->getOperationType();
- // ignoring alias ops as they don't need to execute anything
- if (!in_array($opType, array('update', 'install', 'uninstall'))) {
- // output alias ops in debug verbosity as they have no output otherwise
- if ($this->io->isDebug()) {
- $this->io->writeError(' - ' . $operation->show(false));
- }
- $this->$opType($repo, $operation);
- continue;
- }
- if ($opType === 'update') {
- $package = $operation->getTargetPackage();
- $initialPackage = $operation->getInitialPackage();
- } else {
- $package = $operation->getPackage();
- $initialPackage = null;
- }
- $installer = $this->getInstaller($package->getType());
- $event = 'Composer\Installer\PackageEvents::PRE_PACKAGE_'.strtoupper($opType);
- if (defined($event) && $runScripts && $this->eventDispatcher) {
- $this->eventDispatcher->dispatchPackageEvent(constant($event), $devMode, $repo, $operations, $operation);
- }
- $dispatcher = $this->eventDispatcher;
- $installManager = $this;
- $loop = $this->loop;
- $io = $this->io;
- $promise = $installer->prepare($opType, $package, $initialPackage);
- if (null === $promise) {
- $promise = new \React\Promise\Promise(function ($resolve, $reject) { $resolve(); });
- }
- $promise = $promise->then(function () use ($opType, $installManager, $repo, $operation) {
- return $installManager->$opType($repo, $operation);
- })->then($cleanupPromises[$index])
- ->then(function () use ($opType, $runScripts, $dispatcher, $installManager, $devMode, $repo, $operations, $operation) {
- $repo->write($devMode, $installManager);
- $event = 'Composer\Installer\PackageEvents::POST_PACKAGE_'.strtoupper($opType);
- if (defined($event) && $runScripts && $dispatcher) {
- $dispatcher->dispatchPackageEvent(constant($event), $devMode, $repo, $operations, $operation);
- }
- }, function ($e) use ($opType, $package, $io) {
- $io->writeError(' <error>' . ucfirst($opType) .' of '.$package->getPrettyName().' failed</error>');
- throw $e;
- });
- $promises[] = $promise;
- }
- // execute all prepare => installs/updates/removes => cleanup steps
- if (!empty($promises)) {
- $this->loop->wait($promises);
- }
- } catch (\Exception $e) {
- $runCleanup();
- if ($handleInterrupts) {
- pcntl_signal(SIGINT, $prevHandler);
- }
- throw $e;
- }
- if ($handleInterrupts) {
- pcntl_signal(SIGINT, $prevHandler);
- }
- // 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
- // running a new composer version
- $repo->write($devMode, $this);
- }
- /**
- * Executes install operation.
- *
- * @param RepositoryInterface $repo repository in which to check
- * @param InstallOperation $operation operation instance
- */
- public function install(RepositoryInterface $repo, InstallOperation $operation)
- {
- $package = $operation->getPackage();
- $installer = $this->getInstaller($package->getType());
- $promise = $installer->install($repo, $package);
- $this->markForNotification($package);
- return $promise;
- }
- /**
- * Executes update operation.
- *
- * @param RepositoryInterface $repo repository in which to check
- * @param UpdateOperation $operation operation instance
- */
- public function update(RepositoryInterface $repo, UpdateOperation $operation)
- {
- $initial = $operation->getInitialPackage();
- $target = $operation->getTargetPackage();
- $initialType = $initial->getType();
- $targetType = $target->getType();
- if ($initialType === $targetType) {
- $installer = $this->getInstaller($initialType);
- $promise = $installer->update($repo, $initial, $target);
- $this->markForNotification($target);
- } else {
- $this->getInstaller($initialType)->uninstall($repo, $initial);
- $installer = $this->getInstaller($targetType);
- $promise = $installer->install($repo, $target);
- }
- return $promise;
- }
- /**
- * Uninstalls package.
- *
- * @param RepositoryInterface $repo repository in which to check
- * @param UninstallOperation $operation operation instance
- */
- public function uninstall(RepositoryInterface $repo, UninstallOperation $operation)
- {
- $package = $operation->getPackage();
- $installer = $this->getInstaller($package->getType());
- return $installer->uninstall($repo, $package);
- }
- /**
- * Executes markAliasInstalled operation.
- *
- * @param RepositoryInterface $repo repository in which to check
- * @param MarkAliasInstalledOperation $operation operation instance
- */
- public function markAliasInstalled(RepositoryInterface $repo, MarkAliasInstalledOperation $operation)
- {
- $package = $operation->getPackage();
- if (!$repo->hasPackage($package)) {
- $repo->addPackage(clone $package);
- }
- }
- /**
- * Executes markAlias operation.
- *
- * @param RepositoryInterface $repo repository in which to check
- * @param MarkAliasUninstalledOperation $operation operation instance
- */
- public function markAliasUninstalled(RepositoryInterface $repo, MarkAliasUninstalledOperation $operation)
- {
- $package = $operation->getPackage();
- $repo->removePackage($package);
- }
- /**
- * Returns the installation path of a package
- *
- * @param PackageInterface $package
- * @return string path
- */
- public function getInstallPath(PackageInterface $package)
- {
- $installer = $this->getInstaller($package->getType());
- return $installer->getInstallPath($package);
- }
- public function notifyInstalls(IOInterface $io)
- {
- foreach ($this->notifiablePackages as $repoUrl => $packages) {
- $repositoryName = parse_url($repoUrl, PHP_URL_HOST);
- if ($io->hasAuthentication($repositoryName)) {
- $auth = $io->getAuthentication($repositoryName);
- $authStr = base64_encode($auth['username'] . ':' . $auth['password']);
- $authHeader = 'Authorization: Basic '.$authStr;
- }
- // non-batch API, deprecated
- if (strpos($repoUrl, '%package%')) {
- foreach ($packages as $package) {
- $url = str_replace('%package%', $package->getPrettyName(), $repoUrl);
- $params = array(
- 'version' => $package->getPrettyVersion(),
- 'version_normalized' => $package->getVersion(),
- );
- $opts = array('http' =>
- array(
- 'method' => 'POST',
- 'header' => array('Content-type: application/x-www-form-urlencoded'),
- 'content' => http_build_query($params, '', '&'),
- 'timeout' => 3,
- ),
- );
- if (isset($authHeader)) {
- $opts['http']['header'][] = $authHeader;
- }
- $context = StreamContextFactory::getContext($url, $opts);
- @file_get_contents($url, false, $context);
- }
- continue;
- }
- $postData = array('downloads' => array());
- foreach ($packages as $package) {
- $postData['downloads'][] = array(
- 'name' => $package->getPrettyName(),
- 'version' => $package->getVersion(),
- );
- }
- $opts = array('http' =>
- array(
- 'method' => 'POST',
- 'header' => array('Content-Type: application/json'),
- 'content' => json_encode($postData),
- 'timeout' => 6,
- ),
- );
- if (isset($authHeader)) {
- $opts['http']['header'][] = $authHeader;
- }
- $context = StreamContextFactory::getContext($repoUrl, $opts);
- @file_get_contents($repoUrl, false, $context);
- }
- $this->reset();
- }
- private function markForNotification(PackageInterface $package)
- {
- if ($package->getNotificationUrl()) {
- $this->notifiablePackages[$package->getNotificationUrl()][$package->getName()] = $package;
- }
- }
- }
|