Installer.php 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424
  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;
  12. use Composer\Autoload\AutoloadGenerator;
  13. use Composer\DependencyResolver\DefaultPolicy;
  14. use Composer\DependencyResolver\Operation\UpdateOperation;
  15. use Composer\DependencyResolver\Pool;
  16. use Composer\DependencyResolver\Request;
  17. use Composer\DependencyResolver\Solver;
  18. use Composer\Downloader\DownloadManager;
  19. use Composer\Installer\InstallationManager;
  20. use Composer\IO\IOInterface;
  21. use Composer\Package\AliasPackage;
  22. use Composer\Package\Link;
  23. use Composer\Package\LinkConstraint\VersionConstraint;
  24. use Composer\Package\Locker;
  25. use Composer\Package\PackageInterface;
  26. use Composer\Repository\CompositeRepository;
  27. use Composer\Repository\PlatformRepository;
  28. use Composer\Repository\RepositoryInterface;
  29. use Composer\Repository\RepositoryManager;
  30. use Composer\Script\EventDispatcher;
  31. use Composer\Script\ScriptEvents;
  32. /**
  33. * @author Jordi Boggiano <j.boggiano@seld.be>
  34. * @author Beau Simensen <beau@dflydev.com>
  35. * @author Konstantin Kudryashov <ever.zet@gmail.com>
  36. */
  37. class Installer
  38. {
  39. /**
  40. * @var IOInterface
  41. */
  42. protected $io;
  43. /**
  44. * @var PackageInterface
  45. */
  46. protected $package;
  47. /**
  48. * @var DownloadManager
  49. */
  50. protected $downloadManager;
  51. /**
  52. * @var RepositoryManager
  53. */
  54. protected $repositoryManager;
  55. /**
  56. * @var Locker
  57. */
  58. protected $locker;
  59. /**
  60. * @var InstallationManager
  61. */
  62. protected $installationManager;
  63. /**
  64. * @var EventDispatcher
  65. */
  66. protected $eventDispatcher;
  67. protected $preferSource = false;
  68. protected $dryRun = false;
  69. protected $verbose = false;
  70. protected $installRecommends = true;
  71. protected $installSuggests = false;
  72. protected $update = false;
  73. /**
  74. * @var RepositoryInterface
  75. */
  76. protected $additionalInstalledRepository;
  77. /**
  78. * Constructor
  79. *
  80. * @param IOInterface $io
  81. * @param PackageInterface $package
  82. * @param DownloadManager $downloadManager
  83. * @param RepositoryManager $repositoryManager
  84. * @param Locker $locker
  85. * @param InstallationManager $installationManager
  86. * @param EventDispatcher $eventDispatcher
  87. */
  88. public function __construct(IOInterface $io, PackageInterface $package, DownloadManager $downloadManager, RepositoryManager $repositoryManager, Locker $locker, InstallationManager $installationManager, EventDispatcher $eventDispatcher)
  89. {
  90. $this->io = $io;
  91. $this->package = $package;
  92. $this->downloadManager = $downloadManager;
  93. $this->repositoryManager = $repositoryManager;
  94. $this->locker = $locker;
  95. $this->installationManager = $installationManager;
  96. $this->eventDispatcher = $eventDispatcher;
  97. }
  98. /**
  99. * Run installation (or update)
  100. */
  101. public function run()
  102. {
  103. if ($this->dryRun) {
  104. $this->verbose = true;
  105. }
  106. if ($this->preferSource) {
  107. $this->downloadManager->setPreferSource(true);
  108. }
  109. // create local repo, this contains all packages that are installed in the local project
  110. $localRepo = $this->repositoryManager->getLocalRepository();
  111. // create installed repo, this contains all local packages + platform packages (php & extensions)
  112. $installedRepo = new CompositeRepository(array($localRepo, new PlatformRepository()));
  113. if ($this->additionalInstalledRepository) {
  114. $installedRepo->addRepository($this->additionalInstalledRepository);
  115. }
  116. // prepare aliased packages
  117. if (!$this->update && $this->locker->isLocked()) {
  118. $aliases = $this->locker->getAliases();
  119. } else {
  120. $aliases = $this->package->getAliases();
  121. }
  122. foreach ($aliases as $alias) {
  123. foreach ($this->repositoryManager->findPackages($alias['package'], $alias['version']) as $package) {
  124. $package->getRepository()->addPackage(new AliasPackage($package, $alias['alias_normalized'], $alias['alias']));
  125. }
  126. foreach ($this->repositoryManager->getLocalRepository()->findPackages($alias['package'], $alias['version']) as $package) {
  127. $this->repositoryManager->getLocalRepository()->addPackage(new AliasPackage($package, $alias['alias_normalized'], $alias['alias']));
  128. $this->repositoryManager->getLocalRepository()->removePackage($package);
  129. }
  130. }
  131. // creating repository pool
  132. $pool = new Pool;
  133. $pool->addRepository($installedRepo);
  134. foreach ($this->repositoryManager->getRepositories() as $repository) {
  135. $pool->addRepository($repository);
  136. }
  137. // dispatch pre event
  138. if (!$this->dryRun) {
  139. $eventName = $this->update ? ScriptEvents::PRE_UPDATE_CMD : ScriptEvents::PRE_INSTALL_CMD;
  140. $this->eventDispatcher->dispatchCommandEvent($eventName);
  141. }
  142. // creating requirements request
  143. $installFromLock = false;
  144. $request = new Request($pool);
  145. if ($this->update) {
  146. $this->io->write('<info>Updating dependencies</info>');
  147. $request->updateAll();
  148. $links = $this->collectLinks();
  149. foreach ($links as $link) {
  150. $request->install($link->getTarget(), $link->getConstraint());
  151. }
  152. } elseif ($this->locker->isLocked()) {
  153. $installFromLock = true;
  154. $this->io->write('<info>Installing from lock file</info>');
  155. if (!$this->locker->isFresh()) {
  156. $this->io->write('<warning>Your lock file is out of sync with your composer.json, run "composer.phar update" to update dependencies</warning>');
  157. }
  158. foreach ($this->locker->getLockedPackages() as $package) {
  159. $version = $package->getVersion();
  160. foreach ($aliases as $alias) {
  161. if ($alias['package'] === $package->getName() && $alias['version'] === $package->getVersion()) {
  162. $version = $alias['alias'];
  163. break;
  164. }
  165. }
  166. $constraint = new VersionConstraint('=', $version);
  167. $request->install($package->getName(), $constraint);
  168. }
  169. } else {
  170. $this->io->write('<info>Installing dependencies</info>');
  171. $links = $this->collectLinks();
  172. foreach ($links as $link) {
  173. $request->install($link->getTarget(), $link->getConstraint());
  174. }
  175. }
  176. // prepare solver
  177. $policy = new DefaultPolicy();
  178. $solver = new Solver($policy, $pool, $installedRepo);
  179. // solve dependencies
  180. $operations = $solver->solve($request);
  181. // force dev packages to be updated to latest reference on update
  182. if ($this->update) {
  183. foreach ($localRepo->getPackages() as $package) {
  184. if ($package instanceof AliasPackage) {
  185. $package = $package->getAliasOf();
  186. }
  187. // skip non-dev packages
  188. if (!$package->isDev()) {
  189. continue;
  190. }
  191. // skip packages that will be updated/uninstalled
  192. foreach ($operations as $operation) {
  193. if (('update' === $operation->getJobType() && $package === $operation->getInitialPackage())
  194. || ('uninstall' === $operation->getJobType() && $package === $operation->getPackage())
  195. ) {
  196. continue 2;
  197. }
  198. }
  199. // force update
  200. $newPackage = $this->repositoryManager->findPackage($package->getName(), $package->getVersion());
  201. if ($newPackage && $newPackage->getSourceReference() !== $package->getSourceReference()) {
  202. $operations[] = new UpdateOperation($package, $newPackage);
  203. }
  204. }
  205. }
  206. // anti-alias local repository to allow updates to work fine
  207. foreach ($this->repositoryManager->getLocalRepository()->getPackages() as $package) {
  208. if ($package instanceof AliasPackage) {
  209. $this->repositoryManager->getLocalRepository()->addPackage(clone $package->getAliasOf());
  210. $this->repositoryManager->getLocalRepository()->removePackage($package);
  211. }
  212. }
  213. // execute operations
  214. if (!$operations) {
  215. $this->io->write('<info>Nothing to install/update</info>');
  216. }
  217. foreach ($operations as $operation) {
  218. if ($this->verbose) {
  219. $this->io->write((string) $operation);
  220. }
  221. if (!$this->dryRun) {
  222. $this->eventDispatcher->dispatchPackageEvent(constant('Composer\Script\ScriptEvents::PRE_PACKAGE_'.strtoupper($operation->getJobType())), $operation);
  223. // if installing from lock, restore dev packages' references to their locked state
  224. if ($installFromLock) {
  225. $package = null;
  226. if ('update' === $operation->getJobType()) {
  227. $package = $operation->getTargetPackage();
  228. } elseif ('install' === $operation->getJobType()) {
  229. $package = $operation->getPackage();
  230. }
  231. if ($package && $package->isDev()) {
  232. $lockData = $this->locker->getLockData();
  233. foreach ($lockData['packages'] as $lockedPackage) {
  234. if (!empty($lockedPackage['source-reference']) && strtolower($lockedPackage['package']) === $package->getName()) {
  235. $package->setSourceReference($lockedPackage['source-reference']);
  236. break;
  237. }
  238. }
  239. }
  240. }
  241. $this->installationManager->execute($operation);
  242. $this->eventDispatcher->dispatchPackageEvent(constant('Composer\Script\ScriptEvents::POST_PACKAGE_'.strtoupper($operation->getJobType())), $operation);
  243. }
  244. }
  245. if (!$this->dryRun) {
  246. if ($this->update || !$this->locker->isLocked()) {
  247. $this->locker->setLockData($localRepo->getPackages(), $aliases);
  248. $this->io->write('<info>Writing lock file</info>');
  249. }
  250. $localRepo->write();
  251. $this->io->write('<info>Generating autoload files</info>');
  252. $generator = new AutoloadGenerator;
  253. $generator->dump($localRepo, $this->package, $this->installationManager, $this->installationManager->getVendorPath().'/.composer');
  254. // dispatch post event
  255. $eventName = $this->update ? ScriptEvents::POST_UPDATE_CMD : ScriptEvents::POST_INSTALL_CMD;
  256. $this->eventDispatcher->dispatchCommandEvent($eventName);
  257. }
  258. }
  259. private function collectLinks()
  260. {
  261. $links = $this->package->getRequires();
  262. if ($this->installRecommends) {
  263. $links = array_merge($links, $this->package->getRecommends());
  264. }
  265. if ($this->installSuggests) {
  266. $links = array_merge($links, $this->package->getSuggests());
  267. }
  268. return $links;
  269. }
  270. /**
  271. * Create Installer
  272. *
  273. * @param IOInterface $io
  274. * @param Composer $composer
  275. * @param EventDispatcher $eventDispatcher
  276. * @return Installer
  277. */
  278. static public function create(IOInterface $io, Composer $composer, EventDispatcher $eventDispatcher = null)
  279. {
  280. $eventDispatcher = $eventDispatcher ?: new EventDispatcher($composer, $io);
  281. return new static(
  282. $io,
  283. $composer->getPackage(),
  284. $composer->getDownloadManager(),
  285. $composer->getRepositoryManager(),
  286. $composer->getLocker(),
  287. $composer->getInstallationManager(),
  288. $eventDispatcher
  289. );
  290. }
  291. public function setAdditionalInstalledRepository(RepositoryInterface $additionalInstalledRepository)
  292. {
  293. $this->additionalInstalledRepository = $additionalInstalledRepository;
  294. return $this;
  295. }
  296. /**
  297. * wether to run in drymode or not
  298. *
  299. * @param boolean $dryRun
  300. * @return Installer
  301. */
  302. public function setDryRun($dryRun=true)
  303. {
  304. $this->dryRun = (boolean) $dryRun;
  305. return $this;
  306. }
  307. /**
  308. * install recommend packages
  309. *
  310. * @param boolean $noInstallRecommends
  311. * @return Installer
  312. */
  313. public function setInstallRecommends($installRecommends=true)
  314. {
  315. $this->installRecommends = (boolean) $installRecommends;
  316. return $this;
  317. }
  318. /**
  319. * also install suggested packages
  320. *
  321. * @param boolean $installSuggests
  322. * @return Installer
  323. */
  324. public function setInstallSuggests($installSuggests=true)
  325. {
  326. $this->installSuggests = (boolean) $installSuggests;
  327. return $this;
  328. }
  329. /**
  330. * prefer source installation
  331. *
  332. * @param boolean $preferSource
  333. * @return Installer
  334. */
  335. public function setPreferSource($preferSource=true)
  336. {
  337. $this->preferSource = (boolean) $preferSource;
  338. return $this;
  339. }
  340. /**
  341. * update packages
  342. *
  343. * @param boolean $update
  344. * @return Installer
  345. */
  346. public function setUpdate($update=true)
  347. {
  348. $this->update = (boolean) $update;
  349. return $this;
  350. }
  351. /**
  352. * run in verbose mode
  353. *
  354. * @param boolean $verbose
  355. * @return Installer
  356. */
  357. public function setVerbose($verbose=true)
  358. {
  359. $this->verbose = (boolean) $verbose;
  360. return $this;
  361. }
  362. }