InstallCommand.php 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292
  1. <?php
  2. /*
  3. * This file is part of Composer.
  4. *
  5. * (c) Nils Adermann <naderman@naderman.de>
  6. * Jordi Boggiano <j.boggiano@seld.be>
  7. *
  8. * For the full copyright and license information, please view the LICENSE
  9. * file that was distributed with this source code.
  10. */
  11. namespace Composer\Command;
  12. use Composer\Script\ScriptEvents;
  13. use Composer\Script\EventDispatcher;
  14. use Composer\Autoload\AutoloadGenerator;
  15. use Composer\Composer;
  16. use Composer\DependencyResolver;
  17. use Composer\DependencyResolver\Pool;
  18. use Composer\DependencyResolver\Request;
  19. use Composer\DependencyResolver\Operation;
  20. use Composer\Package\AliasPackage;
  21. use Composer\Package\MemoryPackage;
  22. use Composer\Package\Link;
  23. use Composer\Package\LinkConstraint\VersionConstraint;
  24. use Composer\Package\PackageInterface;
  25. use Composer\Repository\ArrayRepository;
  26. use Composer\Repository\CompositeRepository;
  27. use Composer\Repository\PlatformRepository;
  28. use Composer\Repository\RepositoryInterface;
  29. use Symfony\Component\Console\Input\InputInterface;
  30. use Symfony\Component\Console\Input\InputOption;
  31. use Symfony\Component\Console\Output\OutputInterface;
  32. use Composer\DependencyResolver\Operation\InstallOperation;
  33. use Composer\DependencyResolver\Operation\UpdateOperation;
  34. use Composer\DependencyResolver\Solver;
  35. use Composer\IO\IOInterface;
  36. /**
  37. * @author Jordi Boggiano <j.boggiano@seld.be>
  38. * @author Ryan Weaver <ryan@knplabs.com>
  39. * @author Konstantin Kudryashov <ever.zet@gmail.com>
  40. */
  41. class InstallCommand extends Command
  42. {
  43. protected function configure()
  44. {
  45. $this
  46. ->setName('install')
  47. ->setDescription('Parses the composer.json file and downloads the needed dependencies.')
  48. ->setDefinition(array(
  49. new InputOption('prefer-source', null, InputOption::VALUE_NONE, 'Forces installation from package sources when possible, including VCS information.'),
  50. new InputOption('dry-run', null, InputOption::VALUE_NONE, 'Outputs the operations but will not execute anything (implicitly enables --verbose).'),
  51. new InputOption('no-install-recommends', null, InputOption::VALUE_NONE, 'Do not install recommended packages (ignored when installing from an existing lock file).'),
  52. new InputOption('install-suggests', null, InputOption::VALUE_NONE, 'Also install suggested packages (ignored when installing from an existing lock file).'),
  53. ))
  54. ->setHelp(<<<EOT
  55. The <info>install</info> command reads the composer.json file from the
  56. current directory, processes it, and downloads and installs all the
  57. libraries and dependencies outlined in that file.
  58. <info>php composer.phar install</info>
  59. EOT
  60. )
  61. ;
  62. }
  63. protected function execute(InputInterface $input, OutputInterface $output)
  64. {
  65. $composer = $this->getComposer();
  66. $io = $this->getApplication()->getIO();
  67. $eventDispatcher = new EventDispatcher($composer, $io);
  68. return $this->install(
  69. $io,
  70. $composer,
  71. $eventDispatcher,
  72. (Boolean)$input->getOption('prefer-source'),
  73. (Boolean)$input->getOption('dry-run'),
  74. (Boolean)$input->getOption('verbose'),
  75. (Boolean)$input->getOption('no-install-recommends'),
  76. (Boolean)$input->getOption('install-suggests')
  77. );
  78. }
  79. public function install(IOInterface $io, Composer $composer, EventDispatcher $eventDispatcher, $preferSource = false, $dryRun = false, $verbose = false, $noInstallRecommends = false, $installSuggests = false, $update = false, RepositoryInterface $additionalInstalledRepository = null)
  80. {
  81. if ($dryRun) {
  82. $verbose = true;
  83. }
  84. if ($preferSource) {
  85. $composer->getDownloadManager()->setPreferSource(true);
  86. }
  87. $repoManager = $composer->getRepositoryManager();
  88. // create local repo, this contains all packages that are installed in the local project
  89. $localRepo = $repoManager->getLocalRepository();
  90. // create installed repo, this contains all local packages + platform packages (php & extensions)
  91. $installedRepo = new CompositeRepository(array($localRepo, new PlatformRepository()));
  92. if ($additionalInstalledRepository) {
  93. $installedRepo->addRepository($additionalInstalledRepository);
  94. }
  95. // prepare aliased packages
  96. if (!$update && $composer->getLocker()->isLocked()) {
  97. $aliases = $composer->getLocker()->getAliases();
  98. } else {
  99. $aliases = $composer->getPackage()->getAliases();
  100. }
  101. foreach ($aliases as $alias) {
  102. foreach ($repoManager->findPackages($alias['package'], $alias['version']) as $package) {
  103. $package->getRepository()->addPackage(new AliasPackage($package, $alias['alias']));
  104. }
  105. foreach ($repoManager->getLocalRepository()->findPackages($alias['package'], $alias['version']) as $package) {
  106. $repoManager->getLocalRepository()->addPackage(new AliasPackage($package, $alias['alias']));
  107. $repoManager->getLocalRepository()->removePackage($package);
  108. }
  109. }
  110. // creating repository pool
  111. $pool = new Pool;
  112. $pool->addRepository($installedRepo);
  113. foreach ($repoManager->getRepositories() as $repository) {
  114. $pool->addRepository($repository);
  115. }
  116. // dispatch pre event
  117. if (!$dryRun) {
  118. $eventName = $update ? ScriptEvents::PRE_UPDATE_CMD : ScriptEvents::PRE_INSTALL_CMD;
  119. $eventDispatcher->dispatchCommandEvent($eventName);
  120. }
  121. // creating requirements request
  122. $installFromLock = false;
  123. $request = new Request($pool);
  124. if ($update) {
  125. $io->write('<info>Updating dependencies</info>');
  126. $request->updateAll();
  127. $links = $this->collectLinks($composer->getPackage(), $noInstallRecommends, $installSuggests);
  128. foreach ($links as $link) {
  129. $request->install($link->getTarget(), $link->getConstraint());
  130. }
  131. } elseif ($composer->getLocker()->isLocked()) {
  132. $installFromLock = true;
  133. $io->write('<info>Installing from lock file</info>');
  134. if (!$composer->getLocker()->isFresh()) {
  135. $io->write('<warning>Your lock file is out of sync with your composer.json, run "composer.phar update" to update dependencies</warning>');
  136. }
  137. foreach ($composer->getLocker()->getLockedPackages() as $package) {
  138. $version = $package->getVersion();
  139. foreach ($aliases as $alias) {
  140. if ($alias['package'] === $package->getName() && $alias['version'] === $package->getVersion()) {
  141. $version = $alias['alias'];
  142. break;
  143. }
  144. }
  145. $constraint = new VersionConstraint('=', $version);
  146. $request->install($package->getName(), $constraint);
  147. }
  148. } else {
  149. $io->write('<info>Installing dependencies</info>');
  150. $links = $this->collectLinks($composer->getPackage(), $noInstallRecommends, $installSuggests);
  151. foreach ($links as $link) {
  152. $request->install($link->getTarget(), $link->getConstraint());
  153. }
  154. }
  155. // prepare solver
  156. $installationManager = $composer->getInstallationManager();
  157. $policy = new DependencyResolver\DefaultPolicy();
  158. $solver = new DependencyResolver\Solver($policy, $pool, $installedRepo);
  159. // solve dependencies
  160. $operations = $solver->solve($request);
  161. // force dev packages to be updated to latest reference on update
  162. if ($update) {
  163. foreach ($localRepo->getPackages() as $package) {
  164. if ($package instanceof AliasPackage) {
  165. $package = $package->getAliasOf();
  166. }
  167. // skip non-dev packages
  168. if (!$package->isDev()) {
  169. continue;
  170. }
  171. // skip packages that will be updated/uninstalled
  172. foreach ($operations as $operation) {
  173. if (('update' === $operation->getJobType() && $package === $operation->getInitialPackage())
  174. || ('uninstall' === $operation->getJobType() && $package === $operation->getPackage())
  175. ) {
  176. continue 2;
  177. }
  178. }
  179. // force update
  180. $newPackage = $repoManager->findPackage($package->getName(), $package->getVersion());
  181. if ($newPackage && $newPackage->getSourceReference() !== $package->getSourceReference()) {
  182. $operations[] = new UpdateOperation($package, $newPackage);
  183. }
  184. }
  185. }
  186. // anti-alias local repository to allow updates to work fine
  187. foreach ($repoManager->getLocalRepository()->getPackages() as $package) {
  188. if ($package instanceof AliasPackage) {
  189. $repoManager->getLocalRepository()->addPackage(clone $package->getAliasOf());
  190. $repoManager->getLocalRepository()->removePackage($package);
  191. }
  192. }
  193. // execute operations
  194. if (!$operations) {
  195. $io->write('<info>Nothing to install/update</info>');
  196. }
  197. foreach ($operations as $operation) {
  198. if ($verbose) {
  199. $io->write((string) $operation);
  200. }
  201. if (!$dryRun) {
  202. $eventDispatcher->dispatchPackageEvent(constant('Composer\Script\ScriptEvents::PRE_PACKAGE_'.strtoupper($operation->getJobType())), $operation);
  203. // if installing from lock, restore dev packages' references to their locked state
  204. if ($installFromLock) {
  205. $package = null;
  206. if ('update' === $operation->getJobType()) {
  207. $package = $operation->getTargetPackage();
  208. } elseif ('install' === $operation->getJobType()) {
  209. $package = $operation->getPackage();
  210. }
  211. if ($package && $package->isDev()) {
  212. $lockData = $composer->getLocker()->getLockData();
  213. foreach ($lockData['packages'] as $lockedPackage) {
  214. if (!empty($lockedPackage['source-reference']) && strtolower($lockedPackage['package']) === $package->getName()) {
  215. $package->setSourceReference($lockedPackage['source-reference']);
  216. break;
  217. }
  218. }
  219. }
  220. }
  221. $installationManager->execute($operation);
  222. $eventDispatcher->dispatchPackageEvent(constant('Composer\Script\ScriptEvents::POST_PACKAGE_'.strtoupper($operation->getJobType())), $operation);
  223. }
  224. }
  225. if (!$dryRun) {
  226. if ($update || !$composer->getLocker()->isLocked()) {
  227. $composer->getLocker()->setLockData($localRepo->getPackages(), $aliases);
  228. $io->write('<info>Writing lock file</info>');
  229. }
  230. $localRepo->write();
  231. $io->write('<info>Generating autoload files</info>');
  232. $generator = new AutoloadGenerator;
  233. $generator->dump($localRepo, $composer->getPackage(), $installationManager, $installationManager->getVendorPath().'/.composer');
  234. // dispatch post event
  235. $eventName = $update ? ScriptEvents::POST_UPDATE_CMD : ScriptEvents::POST_INSTALL_CMD;
  236. $eventDispatcher->dispatchCommandEvent($eventName);
  237. }
  238. }
  239. private function collectLinks(PackageInterface $package, $noInstallRecommends, $installSuggests)
  240. {
  241. $links = $package->getRequires();
  242. if (!$noInstallRecommends) {
  243. $links = array_merge($links, $package->getRecommends());
  244. }
  245. if ($installSuggests) {
  246. $links = array_merge($links, $package->getSuggests());
  247. }
  248. return $links;
  249. }
  250. }